/* * Unix SMB/CIFS implementation. * RPC Pipe client / server routines * * Copyright (C) Andrew Tridgell 1992-1997, * Copyright (C) Gerald (Jerry) Carter 2006. * Copyright (C) Guenther Deschner 2007-2008. * * 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 . */ /* This is the implementation of the wks interface. */ #include "includes.h" #include "libnet/libnet.h" #include "../libcli/auth/libcli_auth.h" #include "../librpc/gen_ndr/srv_wkssvc.h" #undef DBGC_CLASS #define DBGC_CLASS DBGC_RPC_SRV #ifdef HAVE_GETUTXENT #include static char **get_logged_on_userlist(TALLOC_CTX *mem_ctx) { char **users = NULL; int num_users = 0; struct utmpx *u; while ((u = getutxent()) != NULL) { char **tmp; if (u->ut_type != USER_PROCESS) { continue; } tmp = talloc_realloc(mem_ctx, users, char *, num_users+1); if (tmp == NULL) { return NULL; } users = tmp; users[num_users] = talloc_strdup(users, u->ut_user); if (users[num_users] == NULL) { TALLOC_FREE(users); return NULL; } num_users += 1; } return users; } #else static char **get_logged_on_userlist(TALLOC_CTX *mem_ctx) { return NULL; } #endif /******************************************************************* Fill in the values for the struct wkssvc_NetWkstaInfo100. ********************************************************************/ static struct wkssvc_NetWkstaInfo100 *create_wks_info_100(TALLOC_CTX *mem_ctx) { struct wkssvc_NetWkstaInfo100 *info100; info100 = talloc(mem_ctx, struct wkssvc_NetWkstaInfo100); if (info100 == NULL) { return NULL; } info100->platform_id = PLATFORM_ID_NT; /* unknown */ info100->version_major = lp_major_announce_version(); info100->version_minor = lp_minor_announce_version(); info100->server_name = talloc_asprintf_strupper_m( info100, "%s", global_myname()); info100->domain_name = talloc_asprintf_strupper_m( info100, "%s", lp_workgroup()); return info100; } static struct wkssvc_NetWkstaInfo101 *create_wks_info_101(TALLOC_CTX *mem_ctx) { struct wkssvc_NetWkstaInfo101 *info101; info101 = talloc(mem_ctx, struct wkssvc_NetWkstaInfo101); if (info101 == NULL) { return NULL; } info101->platform_id = PLATFORM_ID_NT; /* unknown */ info101->version_major = lp_major_announce_version(); info101->version_minor = lp_minor_announce_version(); info101->server_name = talloc_asprintf_strupper_m( info101, "%s", global_myname()); info101->domain_name = talloc_asprintf_strupper_m( info101, "%s", lp_workgroup()); info101->lan_root = NULL; return info101; } static struct wkssvc_NetWkstaInfo102 *create_wks_info_102(TALLOC_CTX *mem_ctx) { struct wkssvc_NetWkstaInfo102 *info102; char **users; info102 = talloc(mem_ctx, struct wkssvc_NetWkstaInfo102); if (info102 == NULL) { return NULL; } info102->platform_id = PLATFORM_ID_NT; /* unknown */ info102->version_major = lp_major_announce_version(); info102->version_minor = lp_minor_announce_version(); info102->server_name = talloc_asprintf_strupper_m( info102, "%s", global_myname()); info102->domain_name = talloc_asprintf_strupper_m( info102, "%s", lp_workgroup()); info102->lan_root = NULL; users = get_logged_on_userlist(talloc_tos()); info102->logged_on_users = talloc_array_length(users); TALLOC_FREE(users); return info102; } /******************************************************************** only supports info level 100 at the moment. ********************************************************************/ WERROR _wkssvc_NetWkstaGetInfo(pipes_struct *p, struct wkssvc_NetWkstaGetInfo *r) { switch (r->in.level) { case 100: r->out.info->info100 = create_wks_info_100(p->mem_ctx); if (r->out.info->info100 == NULL) { return WERR_NOMEM; } break; case 101: r->out.info->info101 = create_wks_info_101(p->mem_ctx); if (r->out.info->info101 == NULL) { return WERR_NOMEM; } break; case 102: r->out.info->info102 = create_wks_info_102(p->mem_ctx); if (r->out.info->info102 == NULL) { return WERR_NOMEM; } break; default: return WERR_UNKNOWN_LEVEL; } return WERR_OK; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetWkstaSetInfo(pipes_struct *p, struct wkssvc_NetWkstaSetInfo *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } static struct wkssvc_NetWkstaEnumUsersCtr0 *create_enum_users0( TALLOC_CTX *mem_ctx) { struct wkssvc_NetWkstaEnumUsersCtr0 *ctr0; char **users; int i, num_users; ctr0 = talloc(mem_ctx, struct wkssvc_NetWkstaEnumUsersCtr0); if (ctr0 == NULL) { return NULL; } users = get_logged_on_userlist(talloc_tos()); if (users == NULL) { TALLOC_FREE(ctr0); return NULL; } num_users = talloc_array_length(users); ctr0->entries_read = num_users; ctr0->user0 = talloc_array(ctr0, struct wkssvc_NetrWkstaUserInfo0, num_users); if (ctr0->user0 == NULL) { TALLOC_FREE(ctr0); TALLOC_FREE(users); return NULL; } for (i=0; iuser0[i].user_name = talloc_move(ctr0->user0, &users[i]); } TALLOC_FREE(users); return ctr0; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetWkstaEnumUsers(pipes_struct *p, struct wkssvc_NetWkstaEnumUsers *r) { if (r->in.info->level != 0) { return WERR_UNKNOWN_LEVEL; } r->out.info->ctr.user0 = create_enum_users0(p->mem_ctx); if (r->out.info->ctr.user0 == NULL) { return WERR_NOMEM; } r->out.info->level = r->in.info->level; *r->out.entries_read = r->out.info->ctr.user0->entries_read; *r->out.resume_handle = 0; return WERR_OK; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrWkstaUserGetInfo(pipes_struct *p, struct wkssvc_NetrWkstaUserGetInfo *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrWkstaUserSetInfo(pipes_struct *p, struct wkssvc_NetrWkstaUserSetInfo *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetWkstaTransportEnum(pipes_struct *p, struct wkssvc_NetWkstaTransportEnum *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrWkstaTransportAdd(pipes_struct *p, struct wkssvc_NetrWkstaTransportAdd *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrWkstaTransportDel(pipes_struct *p, struct wkssvc_NetrWkstaTransportDel *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrUseAdd(pipes_struct *p, struct wkssvc_NetrUseAdd *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrUseGetInfo(pipes_struct *p, struct wkssvc_NetrUseGetInfo *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrUseDel(pipes_struct *p, struct wkssvc_NetrUseDel *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrUseEnum(pipes_struct *p, struct wkssvc_NetrUseEnum *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrMessageBufferSend(pipes_struct *p, struct wkssvc_NetrMessageBufferSend *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrWorkstationStatisticsGet(pipes_struct *p, struct wkssvc_NetrWorkstationStatisticsGet *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrLogonDomainNameAdd(pipes_struct *p, struct wkssvc_NetrLogonDomainNameAdd *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrLogonDomainNameDel(pipes_struct *p, struct wkssvc_NetrLogonDomainNameDel *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrJoinDomain(pipes_struct *p, struct wkssvc_NetrJoinDomain *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrUnjoinDomain(pipes_struct *p, struct wkssvc_NetrUnjoinDomain *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrRenameMachineInDomain(pipes_struct *p, struct wkssvc_NetrRenameMachineInDomain *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrValidateName(pipes_struct *p, struct wkssvc_NetrValidateName *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrGetJoinInformation(pipes_struct *p, struct wkssvc_NetrGetJoinInformation *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrGetJoinableOus(pipes_struct *p, struct wkssvc_NetrGetJoinableOus *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** _wkssvc_NetrJoinDomain2 ********************************************************************/ WERROR _wkssvc_NetrJoinDomain2(pipes_struct *p, struct wkssvc_NetrJoinDomain2 *r) { struct libnet_JoinCtx *j = NULL; char *cleartext_pwd = NULL; char *admin_domain = NULL; char *admin_account = NULL; WERROR werr; struct nt_user_token *token = p->server_info->ptok; if (!r->in.domain_name) { return WERR_INVALID_PARAM; } if (!r->in.admin_account || !r->in.encrypted_password) { return WERR_INVALID_PARAM; } if (!user_has_privileges(token, &se_machine_account) && !nt_token_check_domain_rid(token, DOMAIN_GROUP_RID_ADMINS) && !nt_token_check_sid(&global_sid_Builtin_Administrators, token)) { DEBUG(5,("_wkssvc_NetrJoinDomain2: account doesn't have " "sufficient privileges\n")); return WERR_ACCESS_DENIED; } if ((r->in.join_flags & WKSSVC_JOIN_FLAGS_MACHINE_PWD_PASSED) || (r->in.join_flags & WKSSVC_JOIN_FLAGS_JOIN_UNSECURE)) { return WERR_NOT_SUPPORTED; } werr = decode_wkssvc_join_password_buffer( p->mem_ctx, r->in.encrypted_password, &p->server_info->user_session_key, &cleartext_pwd); if (!W_ERROR_IS_OK(werr)) { return werr; } split_domain_user(p->mem_ctx, r->in.admin_account, &admin_domain, &admin_account); werr = libnet_init_JoinCtx(p->mem_ctx, &j); if (!W_ERROR_IS_OK(werr)) { return werr; } j->in.domain_name = r->in.domain_name; j->in.account_ou = r->in.account_ou; j->in.join_flags = r->in.join_flags; j->in.admin_account = admin_account; j->in.admin_password = cleartext_pwd; j->in.debug = true; j->in.modify_config = lp_config_backend_is_registry(); j->in.msg_ctx = smbd_messaging_context(); become_root(); werr = libnet_Join(p->mem_ctx, j); unbecome_root(); if (!W_ERROR_IS_OK(werr)) { DEBUG(5,("_wkssvc_NetrJoinDomain2: libnet_Join failed with: %s\n", j->out.error_string ? j->out.error_string : win_errstr(werr))); } TALLOC_FREE(j); return werr; } /******************************************************************** _wkssvc_NetrUnjoinDomain2 ********************************************************************/ WERROR _wkssvc_NetrUnjoinDomain2(pipes_struct *p, struct wkssvc_NetrUnjoinDomain2 *r) { struct libnet_UnjoinCtx *u = NULL; char *cleartext_pwd = NULL; char *admin_domain = NULL; char *admin_account = NULL; WERROR werr; struct nt_user_token *token = p->server_info->ptok; if (!r->in.account || !r->in.encrypted_password) { return WERR_INVALID_PARAM; } if (!user_has_privileges(token, &se_machine_account) && !nt_token_check_domain_rid(token, DOMAIN_GROUP_RID_ADMINS) && !nt_token_check_sid(&global_sid_Builtin_Administrators, token)) { DEBUG(5,("_wkssvc_NetrUnjoinDomain2: account doesn't have " "sufficient privileges\n")); return WERR_ACCESS_DENIED; } werr = decode_wkssvc_join_password_buffer( p->mem_ctx, r->in.encrypted_password, &p->server_info->user_session_key, &cleartext_pwd); if (!W_ERROR_IS_OK(werr)) { return werr; } split_domain_user(p->mem_ctx, r->in.account, &admin_domain, &admin_account); werr = libnet_init_UnjoinCtx(p->mem_ctx, &u); if (!W_ERROR_IS_OK(werr)) { return werr; } u->in.domain_name = lp_realm(); u->in.unjoin_flags = r->in.unjoin_flags | WKSSVC_JOIN_FLAGS_JOIN_TYPE; u->in.admin_account = admin_account; u->in.admin_password = cleartext_pwd; u->in.debug = true; u->in.modify_config = lp_config_backend_is_registry(); u->in.msg_ctx = smbd_messaging_context(); become_root(); werr = libnet_Unjoin(p->mem_ctx, u); unbecome_root(); if (!W_ERROR_IS_OK(werr)) { DEBUG(5,("_wkssvc_NetrUnjoinDomain2: libnet_Unjoin failed with: %s\n", u->out.error_string ? u->out.error_string : win_errstr(werr))); } TALLOC_FREE(u); return werr; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrRenameMachineInDomain2(pipes_struct *p, struct wkssvc_NetrRenameMachineInDomain2 *r) { /* for now just return not supported */ return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrValidateName2(pipes_struct *p, struct wkssvc_NetrValidateName2 *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrGetJoinableOus2(pipes_struct *p, struct wkssvc_NetrGetJoinableOus2 *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrAddAlternateComputerName(pipes_struct *p, struct wkssvc_NetrAddAlternateComputerName *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrRemoveAlternateComputerName(pipes_struct *p, struct wkssvc_NetrRemoveAlternateComputerName *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrSetPrimaryComputername(pipes_struct *p, struct wkssvc_NetrSetPrimaryComputername *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; } /******************************************************************** ********************************************************************/ WERROR _wkssvc_NetrEnumerateComputerNames(pipes_struct *p, struct wkssvc_NetrEnumerateComputerNames *r) { /* FIXME: Add implementation code here */ p->rng_fault_state = True; return WERR_NOT_SUPPORTED; }