2007-11-30 20:49:21 +03:00
/*
* Unix SMB / CIFS implementation .
* NetApi Join Support
* Copyright ( C ) Guenther Deschner 2007
*
* 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
* the Free Software Foundation ; either version 3 of the License , or
* ( 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
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2007-12-20 22:59:27 +03:00
# include "lib/netapi/joindomain.h"
2007-11-30 22:33:51 +03:00
extern const char * opt_user_name ;
extern const char * opt_workgroup ;
extern const char * opt_password ;
2007-11-30 20:49:21 +03:00
2007-12-11 23:23:40 +03:00
static WERROR NetJoinDomainLocal ( TALLOC_CTX * mem_ctx ,
const char * server_name ,
const char * domain_name ,
const char * account_ou ,
const char * Account ,
const char * password ,
uint32_t join_flags )
{
2007-12-11 23:32:16 +03:00
struct libnet_JoinCtx * r = NULL ;
WERROR werr ;
werr = libnet_init_JoinCtx ( mem_ctx , & r ) ;
W_ERROR_NOT_OK_RETURN ( werr ) ;
if ( ! server_name | | ! domain_name ) {
return WERR_INVALID_PARAM ;
}
r - > in . server_name = talloc_strdup ( mem_ctx , server_name ) ;
W_ERROR_HAVE_NO_MEMORY ( r - > in . server_name ) ;
r - > in . domain_name = talloc_strdup ( mem_ctx , domain_name ) ;
W_ERROR_HAVE_NO_MEMORY ( r - > in . domain_name ) ;
if ( account_ou ) {
r - > in . account_ou = talloc_strdup ( mem_ctx , account_ou ) ;
W_ERROR_HAVE_NO_MEMORY ( r - > in . account_ou ) ;
}
if ( Account ) {
r - > in . admin_account = talloc_strdup ( mem_ctx , Account ) ;
W_ERROR_HAVE_NO_MEMORY ( r - > in . admin_account ) ;
}
if ( password ) {
r - > in . password = talloc_strdup ( mem_ctx , password ) ;
W_ERROR_HAVE_NO_MEMORY ( r - > in . password ) ;
}
r - > in . join_flags = join_flags ;
r - > in . modify_config = true ;
return libnet_Join ( mem_ctx , r ) ;
2007-12-11 23:23:40 +03:00
}
static WERROR NetJoinDomainRemote ( TALLOC_CTX * mem_ctx ,
const char * server_name ,
const char * domain_name ,
const char * account_ou ,
const char * Account ,
const char * password ,
uint32_t join_flags )
2007-11-30 20:49:21 +03:00
{
struct cli_state * cli = NULL ;
struct rpc_pipe_client * pipe_cli = NULL ;
struct wkssvc_PasswordBuffer encrypted_password ;
NTSTATUS status ;
WERROR werr ;
2007-12-01 13:41:44 +03:00
unsigned int old_timeout = 0 ;
2007-11-30 20:49:21 +03:00
2007-11-30 21:56:41 +03:00
ZERO_STRUCT ( encrypted_password ) ;
2007-11-30 22:33:51 +03:00
status = cli_full_connection ( & cli , NULL , server_name ,
NULL , 0 ,
" IPC$ " , " IPC " ,
opt_user_name , opt_workgroup ,
opt_password , 0 , Undefined , NULL ) ;
2007-11-30 20:49:21 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
werr = ntstatus_to_werror ( status ) ;
goto done ;
}
2007-11-30 21:55:40 +03:00
old_timeout = cli_set_timeout ( cli , 60000 ) ;
2007-11-30 20:49:21 +03:00
pipe_cli = cli_rpc_pipe_open_noauth ( cli , PI_WKSSVC ,
& status ) ;
if ( ! pipe_cli ) {
werr = ntstatus_to_werror ( status ) ;
goto done ;
} ;
2007-11-30 21:56:41 +03:00
if ( password ) {
encode_wkssvc_join_password_buffer ( mem_ctx ,
password ,
& cli - > user_session_key ,
& encrypted_password ) ;
}
2007-11-30 20:49:21 +03:00
2007-11-30 21:55:40 +03:00
old_timeout = cli_set_timeout ( cli , 60000 ) ;
2007-11-30 20:49:21 +03:00
status = rpccli_wkssvc_NetrJoinDomain2 ( pipe_cli , mem_ctx ,
server_name , domain_name ,
account_ou , Account ,
& encrypted_password ,
2007-12-03 20:40:09 +03:00
join_flags , & werr ) ;
2007-11-30 20:49:21 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
werr = ntstatus_to_werror ( status ) ;
2007-11-30 21:56:41 +03:00
goto done ;
2007-11-30 20:49:21 +03:00
}
done :
2007-11-30 21:56:41 +03:00
if ( cli ) {
cli_set_timeout ( cli , old_timeout ) ;
cli_shutdown ( cli ) ;
}
2007-12-11 23:23:40 +03:00
return werr ;
}
WERROR NetJoinDomain ( const char * server_name ,
const char * domain_name ,
const char * account_ou ,
const char * Account ,
const char * password ,
uint32_t join_flags )
{
TALLOC_CTX * mem_ctx = NULL ;
WERROR werr ;
mem_ctx = talloc_init ( " NetJoinDomain " ) ;
if ( ! mem_ctx ) {
werr = WERR_NOMEM ;
goto done ;
}
if ( ! domain_name ) {
werr = WERR_INVALID_PARAM ;
goto done ;
}
if ( ! server_name | | is_myname_or_ipaddr ( server_name ) ) {
const char * dc = NULL ;
/* FIXME: DsGetDcName */
if ( server_name = = NULL ) {
dc = domain_name ;
} else {
dc = domain_name ;
}
werr = NetJoinDomainLocal ( mem_ctx ,
dc ,
domain_name ,
account_ou ,
Account ,
password ,
join_flags ) ;
goto done ;
}
werr = NetJoinDomainRemote ( mem_ctx ,
server_name ,
domain_name ,
account_ou ,
Account ,
password ,
join_flags ) ;
done :
2007-11-30 20:49:21 +03:00
TALLOC_FREE ( mem_ctx ) ;
return werr ;
}
2007-11-30 21:57:08 +03:00
WERROR NetUnjoinDomain ( const char * server_name ,
const char * account ,
const char * password ,
uint32_t unjoin_flags )
{
TALLOC_CTX * mem_ctx = NULL ;
struct cli_state * cli = NULL ;
struct rpc_pipe_client * pipe_cli = NULL ;
struct wkssvc_PasswordBuffer encrypted_password ;
NTSTATUS status ;
WERROR werr ;
2007-12-01 13:41:44 +03:00
unsigned int old_timeout = 0 ;
2007-11-30 21:57:08 +03:00
ZERO_STRUCT ( encrypted_password ) ;
mem_ctx = talloc_init ( " NetUnjoinDomain " ) ;
if ( ! mem_ctx ) {
werr = WERR_NOMEM ;
goto done ;
}
if ( ! server_name | | is_myname_or_ipaddr ( server_name ) ) {
werr = WERR_NOT_SUPPORTED ;
goto done ;
}
2007-11-30 22:33:51 +03:00
status = cli_full_connection ( & cli , NULL , server_name ,
NULL , 0 ,
" IPC$ " , " IPC " ,
opt_user_name , opt_workgroup ,
opt_password , 0 , Undefined , NULL ) ;
2007-11-30 21:57:08 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
werr = ntstatus_to_werror ( status ) ;
goto done ;
}
old_timeout = cli_set_timeout ( cli , 60000 ) ;
pipe_cli = cli_rpc_pipe_open_noauth ( cli , PI_WKSSVC ,
& status ) ;
if ( ! pipe_cli ) {
werr = ntstatus_to_werror ( status ) ;
goto done ;
} ;
if ( password ) {
encode_wkssvc_join_password_buffer ( mem_ctx ,
password ,
& cli - > user_session_key ,
& encrypted_password ) ;
}
old_timeout = cli_set_timeout ( cli , 60000 ) ;
status = rpccli_wkssvc_NetrUnjoinDomain2 ( pipe_cli , mem_ctx ,
server_name ,
account ,
& encrypted_password ,
2007-12-03 20:40:09 +03:00
unjoin_flags ,
& werr ) ;
2007-11-30 21:57:08 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
werr = ntstatus_to_werror ( status ) ;
goto done ;
}
done :
if ( cli ) {
cli_set_timeout ( cli , old_timeout ) ;
cli_shutdown ( cli ) ;
}
TALLOC_FREE ( mem_ctx ) ;
return werr ;
}
2007-12-06 21:04:49 +03:00
WERROR NetGetJoinInformation ( const char * server_name ,
const char * * name_buffer ,
uint16_t * name_type )
{
TALLOC_CTX * mem_ctx = NULL ;
struct cli_state * cli = NULL ;
struct rpc_pipe_client * pipe_cli = NULL ;
NTSTATUS status ;
WERROR werr ;
mem_ctx = talloc_init ( " NetGetJoinInformation " ) ;
if ( ! mem_ctx ) {
werr = WERR_NOMEM ;
goto done ;
}
status = cli_full_connection ( & cli , NULL , server_name ,
NULL , 0 ,
" IPC$ " , " IPC " ,
opt_user_name , opt_workgroup ,
opt_password , 0 , Undefined , NULL ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
werr = ntstatus_to_werror ( status ) ;
goto done ;
}
pipe_cli = cli_rpc_pipe_open_noauth ( cli , PI_WKSSVC ,
& status ) ;
if ( ! pipe_cli ) {
werr = ntstatus_to_werror ( status ) ;
goto done ;
} ;
status = rpccli_wkssvc_NetrGetJoinInformation ( pipe_cli , mem_ctx ,
server_name ,
name_buffer ,
( enum wkssvc_NetJoinStatus * ) name_type ,
& werr ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
werr = ntstatus_to_werror ( status ) ;
goto done ;
}
done :
if ( cli ) {
cli_shutdown ( cli ) ;
}
TALLOC_FREE ( mem_ctx ) ;
return werr ;
}