2006-02-03 22:19:41 +00:00
/*
* Unix SMB / CIFS implementation .
* Shell around net rpc subcommands
* Copyright ( C ) Volker Lendecke 2006
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
2007-07-09 19:25:36 +00:00
* the Free Software Foundation ; either version 3 of the License , or
2006-02-03 22:19:41 +00:00
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
2007-07-10 05:23:25 +00:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2006-02-03 22:19:41 +00:00
*/
# include "includes.h"
# include "utils/net.h"
2011-02-28 10:19:44 +01:00
# include "rpc_client/cli_pipe.h"
2010-05-28 02:18:21 +02:00
# include "../librpc/gen_ndr/ndr_samr.h"
2010-08-03 00:03:03 +02:00
# include "lib/netapi/netapi.h"
2010-11-21 20:40:50 +11:00
# include "lib/netapi/netapi_net.h"
2010-10-01 10:34:14 +02:00
# include "../libcli/smbreadline/smbreadline.h"
2011-05-06 11:47:43 +02:00
# include "libsmb/libsmb.h"
2018-12-08 15:05:53 +01:00
# include "libcli/security/dom_sid.h"
2006-02-03 22:19:41 +00:00
2021-01-13 09:31:41 +01:00
# include <popt.h>
2008-05-09 23:22:12 +02:00
static NTSTATUS rpc_sh_info ( struct net_context * c ,
TALLOC_CTX * mem_ctx , struct rpc_sh_ctx * ctx ,
2006-02-03 22:19:41 +00:00
struct rpc_pipe_client * pipe_hnd ,
int argc , const char * * argv )
{
2008-05-09 23:22:12 +02:00
return rpc_info_internals ( c , ctx - > domain_sid , ctx - > domain_name ,
2006-02-03 22:19:41 +00:00
ctx - > cli , pipe_hnd , mem_ctx ,
argc , argv ) ;
}
static struct rpc_sh_ctx * this_ctx ;
static char * * completion_fn ( const char * text , int start , int end )
{
char * * cmds = NULL ;
int n_cmds = 0 ;
struct rpc_sh_cmd * c ;
if ( start ! = 0 ) {
return NULL ;
}
2024-09-30 10:34:17 +02:00
ADD_TO_MALLOC_ARRAY ( char * , SMB_STRDUP ( text ) , & cmds , & n_cmds ) ;
2006-02-03 22:19:41 +00:00
for ( c = this_ctx - > cmds ; c - > name ! = NULL ; c + + ) {
2007-10-18 17:40:25 -07:00
bool match = ( strncmp ( text , c - > name , strlen ( text ) ) = = 0 ) ;
2006-02-03 22:19:41 +00:00
if ( match ) {
2024-10-04 15:12:22 +02:00
ADD_TO_MALLOC_ARRAY ( char * , SMB_STRDUP ( c - > name ) ,
& cmds , & n_cmds ) ;
2006-02-03 22:19:41 +00:00
}
}
if ( n_cmds = = 2 ) {
SAFE_FREE ( cmds [ 0 ] ) ;
cmds [ 0 ] = cmds [ 1 ] ;
n_cmds - = 1 ;
}
2024-09-30 10:34:17 +02:00
ADD_TO_MALLOC_ARRAY ( char * , NULL , & cmds , & n_cmds ) ;
2006-02-03 22:19:41 +00:00
return cmds ;
}
2008-05-09 23:22:12 +02:00
static NTSTATUS net_sh_run ( struct net_context * c ,
struct rpc_sh_ctx * ctx , struct rpc_sh_cmd * cmd ,
2006-02-03 22:19:41 +00:00
int argc , const char * * argv )
{
TALLOC_CTX * mem_ctx ;
2009-11-12 13:56:33 -08:00
struct rpc_pipe_client * pipe_hnd = NULL ;
2006-02-03 22:19:41 +00:00
NTSTATUS status ;
mem_ctx = talloc_new ( ctx ) ;
if ( mem_ctx = = NULL ) {
2009-08-10 20:38:33 +02:00
d_fprintf ( stderr , _ ( " talloc_new failed \n " ) ) ;
2006-02-03 22:19:41 +00:00
return NT_STATUS_NO_MEMORY ;
}
2013-05-24 13:29:28 +02:00
status = cli_rpc_pipe_open_noauth ( ctx - > cli , cmd - > table ,
2009-11-08 19:37:26 +01:00
& pipe_hnd ) ;
2008-07-20 11:04:31 +02:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-08-10 20:38:33 +02:00
d_fprintf ( stderr , _ ( " Could not open pipe: %s \n " ) ,
2006-02-03 22:19:41 +00:00
nt_errstr ( status ) ) ;
return status ;
}
2008-05-09 23:22:12 +02:00
status = cmd - > fn ( c , mem_ctx , ctx , pipe_hnd , argc , argv ) ;
2006-02-03 22:19:41 +00:00
2008-04-20 13:51:46 +02:00
TALLOC_FREE ( pipe_hnd ) ;
2006-02-03 22:19:41 +00:00
talloc_destroy ( mem_ctx ) ;
return status ;
}
2008-05-09 23:22:12 +02:00
static bool net_sh_process ( struct net_context * c ,
struct rpc_sh_ctx * ctx ,
2006-02-03 22:19:41 +00:00
int argc , const char * * argv )
{
2008-05-09 23:22:12 +02:00
struct rpc_sh_cmd * cmd ;
2006-02-03 22:19:41 +00:00
struct rpc_sh_ctx * new_ctx ;
NTSTATUS status ;
if ( argc = = 0 ) {
2008-05-12 11:53:23 +02:00
return true ;
2006-02-03 22:19:41 +00:00
}
if ( ctx = = this_ctx ) {
/* We've been called from the cmd line */
if ( strequal ( argv [ 0 ] , " .. " ) & &
( this_ctx - > parent ! = NULL ) ) {
new_ctx = this_ctx - > parent ;
2006-02-20 17:59:58 +00:00
TALLOC_FREE ( this_ctx ) ;
2006-02-03 22:19:41 +00:00
this_ctx = new_ctx ;
2008-05-12 11:53:23 +02:00
return true ;
2006-02-03 22:19:41 +00:00
}
}
2009-09-02 21:09:13 +02:00
if ( strequal ( argv [ 0 ] , " exit " ) | |
strequal ( argv [ 0 ] , " quit " ) | |
strequal ( argv [ 0 ] , " q " ) ) {
2008-05-12 11:53:23 +02:00
return false ;
2006-06-19 10:02:04 +00:00
}
2006-02-03 22:19:41 +00:00
if ( strequal ( argv [ 0 ] , " help " ) | | strequal ( argv [ 0 ] , " ? " ) ) {
2008-05-09 23:22:12 +02:00
for ( cmd = ctx - > cmds ; cmd - > name ! = NULL ; cmd + + ) {
2006-02-03 22:19:41 +00:00
if ( ctx ! = this_ctx ) {
d_printf ( " %s " , ctx - > whoami ) ;
}
2008-05-09 23:22:12 +02:00
d_printf ( " %-15s %s \n " , cmd - > name , cmd - > help ) ;
2006-02-03 22:19:41 +00:00
}
2008-05-12 11:53:23 +02:00
return true ;
2006-02-03 22:19:41 +00:00
}
2008-05-09 23:22:12 +02:00
for ( cmd = ctx - > cmds ; cmd - > name ! = NULL ; cmd + + ) {
if ( strequal ( cmd - > name , argv [ 0 ] ) ) {
2006-02-03 22:19:41 +00:00
break ;
}
}
2008-05-09 23:22:12 +02:00
if ( cmd - > name = = NULL ) {
2006-02-03 22:19:41 +00:00
/* None found */
2009-08-10 20:38:33 +02:00
d_fprintf ( stderr , _ ( " %s: unknown cmd \n " ) , argv [ 0 ] ) ;
2008-05-12 11:53:23 +02:00
return true ;
2006-02-03 22:19:41 +00:00
}
2011-06-07 11:38:41 +10:00
new_ctx = talloc ( ctx , struct rpc_sh_ctx ) ;
2006-02-03 22:19:41 +00:00
if ( new_ctx = = NULL ) {
2009-08-10 20:38:33 +02:00
d_fprintf ( stderr , _ ( " talloc failed \n " ) ) ;
2008-05-12 11:53:23 +02:00
return false ;
2006-02-03 22:19:41 +00:00
}
new_ctx - > cli = ctx - > cli ;
new_ctx - > whoami = talloc_asprintf ( new_ctx , " %s %s " ,
2008-05-09 23:22:12 +02:00
ctx - > whoami , cmd - > name ) ;
new_ctx - > thiscmd = talloc_strdup ( new_ctx , cmd - > name ) ;
2006-02-03 22:19:41 +00:00
2008-05-09 23:22:12 +02:00
if ( cmd - > sub ! = NULL ) {
new_ctx - > cmds = cmd - > sub ( c , new_ctx , ctx ) ;
2006-02-03 22:19:41 +00:00
} else {
new_ctx - > cmds = NULL ;
}
new_ctx - > parent = ctx ;
new_ctx - > domain_name = ctx - > domain_name ;
new_ctx - > domain_sid = ctx - > domain_sid ;
argc - = 1 ;
argv + = 1 ;
2008-05-09 23:22:12 +02:00
if ( cmd - > sub ! = NULL ) {
2006-02-03 22:19:41 +00:00
if ( argc = = 0 ) {
this_ctx = new_ctx ;
2008-05-12 11:53:23 +02:00
return true ;
2006-02-03 22:19:41 +00:00
}
2008-05-09 23:22:12 +02:00
return net_sh_process ( c , new_ctx , argc , argv ) ;
2006-02-03 22:19:41 +00:00
}
2008-05-09 23:22:12 +02:00
status = net_sh_run ( c , new_ctx , cmd , argc , argv ) ;
2006-02-03 22:19:41 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-08-10 20:38:33 +02:00
d_fprintf ( stderr , _ ( " %s failed: %s \n " ) , new_ctx - > whoami ,
2006-02-03 22:19:41 +00:00
nt_errstr ( status ) ) ;
}
2008-05-12 11:53:23 +02:00
return true ;
2006-02-03 22:19:41 +00:00
}
2006-02-20 21:58:33 +00:00
static struct rpc_sh_cmd sh_cmds [ 6 ] = {
2012-01-10 21:53:42 +11:00
{ " info " , NULL , & ndr_table_samr , rpc_sh_info ,
2009-08-10 20:38:33 +02:00
N_ ( " Print information about the domain connected to " ) } ,
2006-02-20 21:58:33 +00:00
2009-11-08 19:37:26 +01:00
{ " rights " , net_rpc_rights_cmds , 0 , NULL ,
2009-08-10 20:38:33 +02:00
N_ ( " List/Grant/Revoke user rights " ) } ,
2006-02-20 21:58:33 +00:00
2009-11-08 19:37:26 +01:00
{ " share " , net_rpc_share_cmds , 0 , NULL ,
2009-08-10 20:38:33 +02:00
N_ ( " List/Add/Remove etc shares " ) } ,
2006-02-20 21:58:33 +00:00
2009-11-08 19:37:26 +01:00
{ " user " , net_rpc_user_cmds , 0 , NULL ,
2009-08-10 20:38:33 +02:00
N_ ( " List/Add/Remove user info " ) } ,
2006-02-20 21:58:33 +00:00
2009-11-08 19:37:26 +01:00
{ " account " , net_rpc_acct_cmds , 0 , NULL ,
2009-08-10 20:38:33 +02:00
N_ ( " Show/Change account policy settings " ) } ,
2006-02-20 21:58:33 +00:00
2009-11-08 19:37:26 +01:00
{ NULL , NULL , 0 , NULL , NULL }
2006-02-20 21:58:33 +00:00
} ;
2008-05-09 23:22:12 +02:00
int net_rpc_shell ( struct net_context * c , int argc , const char * * argv )
2006-02-03 22:19:41 +00:00
{
NTSTATUS status ;
struct rpc_sh_ctx * ctx ;
2018-12-08 15:05:53 +01:00
struct dom_sid_buf buf ;
2021-03-24 14:37:26 +01:00
NET_API_STATUS net_api_status ;
2006-02-03 22:19:41 +00:00
2008-05-21 10:27:59 +02:00
if ( argc ! = 0 | | c - > display_usage ) {
2010-01-19 11:43:54 +01:00
d_printf ( " %s \n net rpc shell \n " , _ ( " Usage: " ) ) ;
2006-02-03 22:19:41 +00:00
return - 1 ;
}
2023-09-07 14:53:22 +02:00
net_api_status = libnetapi_net_init ( & c - > netapi_ctx , c - > lp_ctx , c - > creds ) ;
2021-03-24 14:37:26 +01:00
if ( net_api_status ! = 0 ) {
return - 1 ;
2008-08-27 13:54:49 +02:00
}
2011-06-07 11:38:41 +10:00
ctx = talloc ( NULL , struct rpc_sh_ctx ) ;
2006-02-03 22:19:41 +00:00
if ( ctx = = NULL ) {
2009-08-10 20:38:33 +02:00
d_fprintf ( stderr , _ ( " talloc failed \n " ) ) ;
2006-02-03 22:19:41 +00:00
return - 1 ;
}
2008-05-09 23:22:12 +02:00
status = net_make_ipc_connection ( c , 0 , & ( ctx - > cli ) ) ;
2007-09-17 15:11:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-08-10 20:38:33 +02:00
d_fprintf ( stderr , _ ( " Could not open connection: %s \n " ) ,
2007-09-17 15:11:20 +00:00
nt_errstr ( status ) ) ;
2006-02-03 22:19:41 +00:00
return - 1 ;
}
ctx - > cmds = sh_cmds ;
ctx - > whoami = " net rpc " ;
ctx - > parent = NULL ;
status = net_get_remote_domain_sid ( ctx - > cli , ctx , & ctx - > domain_sid ,
& ctx - > domain_name ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return - 1 ;
}
2009-08-10 20:38:33 +02:00
d_printf ( _ ( " Talking to domain %s (%s) \n " ) , ctx - > domain_name ,
2018-12-08 15:05:53 +01:00
dom_sid_str_buf ( ctx - > domain_sid , & buf ) ) ;
2007-12-06 17:16:33 -08:00
2006-02-03 22:19:41 +00:00
this_ctx = ctx ;
while ( 1 ) {
2007-12-06 17:16:33 -08:00
char * prompt = NULL ;
char * line = NULL ;
2006-02-03 22:19:41 +00:00
int ret ;
2007-12-06 17:16:33 -08:00
if ( asprintf ( & prompt , " %s> " , this_ctx - > whoami ) < 0 ) {
break ;
}
2006-02-03 22:19:41 +00:00
line = smb_readline ( prompt , NULL , completion_fn ) ;
SAFE_FREE ( prompt ) ;
if ( line = = NULL ) {
break ;
}
ret = poptParseArgvString ( line , & argc , & argv ) ;
2006-02-04 21:44:57 +00:00
if ( ret = = POPT_ERROR_NOARG ) {
2007-12-06 17:16:33 -08:00
SAFE_FREE ( line ) ;
2006-02-04 21:44:57 +00:00
continue ;
}
2006-02-03 22:19:41 +00:00
if ( ret ! = 0 ) {
2009-08-10 20:38:33 +02:00
d_fprintf ( stderr , _ ( " cmdline invalid: %s \n " ) ,
2006-02-03 22:19:41 +00:00
poptStrerror ( ret ) ) ;
2007-12-06 17:16:33 -08:00
SAFE_FREE ( line ) ;
2008-05-12 11:53:23 +02:00
return false ;
2006-02-03 22:19:41 +00:00
}
if ( ( line [ 0 ] ! = ' \n ' ) & &
2008-05-09 23:22:12 +02:00
( ! net_sh_process ( c , this_ctx , argc , argv ) ) ) {
2007-12-06 17:16:33 -08:00
SAFE_FREE ( line ) ;
2006-02-03 22:19:41 +00:00
break ;
}
2007-12-06 17:16:33 -08:00
SAFE_FREE ( line ) ;
2006-02-03 22:19:41 +00:00
}
cli_shutdown ( ctx - > cli ) ;
2006-02-20 17:59:58 +00:00
TALLOC_FREE ( ctx ) ;
2006-02-03 22:19:41 +00:00
return 0 ;
}