2007-12-17 10:34:29 +01:00
/*
2007-04-09 10:38:55 +00:00
* Unix SMB / CIFS implementation .
* Authentication utility functions
* Copyright ( C ) Andrew Tridgell 1992 - 1998
* Copyright ( C ) Andrew Bartlett 2001
* Copyright ( C ) Jeremy Allison 2000 - 2001
* Copyright ( C ) Rafal Szczesniak 2002
* Copyright ( C ) Volker Lendecke 2006
* Copyright ( C ) Michael Adam 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
2007-07-09 19:25:36 +00:00
* the Free Software Foundation ; either version 3 of the License , or
2007-04-09 10:38:55 +00:00
* ( at your option ) any later version .
2007-12-17 10:34:29 +01:00
*
2007-04-09 10:38:55 +00:00
* 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 .
2007-12-17 10:34:29 +01:00
*
2007-04-09 10:38:55 +00:00
* 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/>.
2007-04-09 10:38:55 +00:00
*/
/* functions moved from auth/auth_util.c to minimize linker deps */
# include "includes.h"
2011-03-25 02:28:05 +01:00
# include "auth.h"
2010-08-05 02:25:37 +02:00
# include "secrets.h"
2010-08-18 12:24:35 +02:00
# include "memcache.h"
2010-08-05 15:14:04 +02:00
# include "../librpc/gen_ndr/netlogon.h"
2010-10-12 15:27:50 +11:00
# include "../libcli/security/security.h"
2011-03-02 16:06:32 +01:00
# include "../lib/util/util_pw.h"
2011-03-22 16:50:02 +01:00
# include "passdb.h"
2011-03-25 11:56:52 +01:00
# include "lib/privileges.h"
2010-08-05 15:14:04 +02:00
2007-04-09 10:38:55 +00:00
/****************************************************************************
2010-08-26 22:08:22 +10:00
Check for a SID in an struct security_token
2007-04-09 10:38:55 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-08-26 22:08:22 +10:00
bool nt_token_check_sid ( const struct dom_sid * sid , const struct security_token * token )
2007-04-09 10:38:55 +00:00
{
if ( ! sid | | ! token )
return False ;
2007-12-17 10:34:29 +01:00
2010-09-17 14:55:56 +10:00
return security_token_has_sid ( token , sid ) ;
2007-04-09 10:38:55 +00:00
}
2010-08-26 22:08:22 +10:00
bool nt_token_check_domain_rid ( struct security_token * token , uint32 rid )
2007-04-09 10:38:55 +00:00
{
2010-05-21 11:25:01 +10:00
struct dom_sid domain_sid ;
2007-04-09 10:38:55 +00:00
2007-12-17 10:34:29 +01:00
/* if we are a domain member, the get the domain SID, else for
2007-04-09 10:38:55 +00:00
a DC or standalone server , use our own SID */
if ( lp_server_role ( ) = = ROLE_DOMAIN_MEMBER ) {
if ( ! secrets_fetch_domain_sid ( lp_workgroup ( ) ,
& domain_sid ) ) {
DEBUG ( 1 , ( " nt_token_check_domain_rid: Cannot lookup "
" SID for domain [%s] \n " , lp_workgroup ( ) ) ) ;
return False ;
}
2007-12-17 10:34:29 +01:00
}
2007-04-09 10:38:55 +00:00
else
sid_copy ( & domain_sid , get_global_sam_sid ( ) ) ;
sid_append_rid ( & domain_sid , rid ) ;
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
return nt_token_check_sid ( & domain_sid , token ) ; \
}
/******************************************************************************
Create a token for the root user to be used internally by smbd .
This is similar to running under the context of the LOCAL_SYSTEM account
in Windows . This is a read - only token . Do not modify it or free ( ) it .
2010-09-25 15:44:55 -07:00
Create a copy if you need to change it .
2007-04-09 10:38:55 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-08-26 22:08:22 +10:00
struct security_token * get_root_nt_token ( void )
2007-04-09 10:38:55 +00:00
{
2010-08-26 20:04:11 +10:00
struct security_token * token , * for_cache ;
2010-05-21 11:25:01 +10:00
struct dom_sid u_sid , g_sid ;
2007-04-09 10:38:55 +00:00
struct passwd * pw ;
2007-12-28 17:24:39 +01:00
void * cache_data ;
2007-12-17 10:34:29 +01:00
2007-12-28 17:24:39 +01:00
cache_data = memcache_lookup_talloc (
NULL , SINGLETON_CACHE_TALLOC ,
2008-10-13 05:20:26 +02:00
data_blob_string_const_null ( " root_nt_token " ) ) ;
2007-12-28 17:24:39 +01:00
if ( cache_data ! = NULL ) {
return talloc_get_type_abort (
2010-08-26 20:04:11 +10:00
cache_data , struct security_token ) ;
2007-12-28 17:24:39 +01:00
}
2007-04-09 10:38:55 +00:00
2009-08-19 16:55:26 -07:00
if ( ! ( pw = sys_getpwuid ( 0 ) ) ) {
2009-08-22 09:40:58 -07:00
if ( ! ( pw = sys_getpwnam ( " root " ) ) ) {
DEBUG ( 0 , ( " get_root_nt_token: both sys_getpwuid(0) "
" and sys_getpwnam( \" root \" ) failed! \n " ) ) ;
return NULL ;
}
2007-04-09 10:38:55 +00:00
}
2007-12-17 10:34:29 +01:00
/* get the user and primary group SIDs; although the
2007-04-09 10:38:55 +00:00
BUILTIN \ Administrators SId is really the one that matters here */
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
uid_to_sid ( & u_sid , pw - > pw_uid ) ;
gid_to_sid ( & g_sid , pw - > pw_gid ) ;
2010-09-25 15:46:36 -07:00
token = create_local_nt_token ( talloc_tos ( ) , & u_sid , False ,
2007-04-09 10:38:55 +00:00
1 , & global_sid_Builtin_Administrators ) ;
2007-12-28 17:24:39 +01:00
2010-08-30 12:47:29 +10:00
security_token_set_privilege ( token , SEC_PRIV_DISK_OPERATOR ) ;
2008-04-15 20:41:14 +02:00
2008-11-14 13:42:54 +01:00
for_cache = token ;
2007-12-28 17:24:39 +01:00
memcache_add_talloc (
NULL , SINGLETON_CACHE_TALLOC ,
2008-11-14 13:42:54 +01:00
data_blob_string_const_null ( " root_nt_token " ) , & for_cache ) ;
2007-12-28 17:24:39 +01:00
2007-04-09 10:38:55 +00:00
return token ;
}
/*
* Add alias SIDs from memberships within the partially created token SID list
*/
2010-05-21 11:25:01 +10:00
NTSTATUS add_aliases ( const struct dom_sid * domain_sid ,
2010-08-26 20:04:11 +10:00
struct security_token * token )
2007-04-09 10:38:55 +00:00
{
uint32 * aliases ;
size_t i , num_aliases ;
NTSTATUS status ;
TALLOC_CTX * tmp_ctx ;
if ( ! ( tmp_ctx = talloc_init ( " add_aliases " ) ) ) {
return NT_STATUS_NO_MEMORY ;
}
aliases = NULL ;
num_aliases = 0 ;
status = pdb_enum_alias_memberships ( tmp_ctx , domain_sid ,
2010-08-31 09:32:52 +10:00
token - > sids ,
2007-04-09 10:38:55 +00:00
token - > num_sids ,
& aliases , & num_aliases ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 10 , ( " pdb_enum_alias_memberships failed: %s \n " ,
nt_errstr ( status ) ) ) ;
2008-01-09 00:11:31 +01:00
goto done ;
2007-04-09 10:38:55 +00:00
}
for ( i = 0 ; i < num_aliases ; i + + ) {
2010-05-21 11:25:01 +10:00
struct dom_sid alias_sid ;
2007-04-09 10:38:55 +00:00
sid_compose ( & alias_sid , domain_sid , aliases [ i ] ) ;
2008-01-09 00:11:31 +01:00
status = add_sid_to_array_unique ( token , & alias_sid ,
2010-08-31 09:32:52 +10:00
& token - > sids ,
2008-01-09 00:11:31 +01:00
& token - > num_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2007-04-09 10:38:55 +00:00
DEBUG ( 0 , ( " add_sid_to_array failed \n " ) ) ;
2008-01-09 00:11:31 +01:00
goto done ;
2007-04-09 10:38:55 +00:00
}
}
2008-01-09 00:11:31 +01:00
done :
2007-04-09 10:38:55 +00:00
TALLOC_FREE ( tmp_ctx ) ;
return NT_STATUS_OK ;
}
/*******************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-08-26 20:04:11 +10:00
static NTSTATUS add_builtin_administrators ( struct security_token * token ,
2010-05-21 11:25:01 +10:00
const struct dom_sid * dom_sid )
2007-04-09 10:38:55 +00:00
{
2010-05-21 11:25:01 +10:00
struct dom_sid domadm ;
2008-01-09 00:11:31 +01:00
NTSTATUS status ;
2007-04-09 10:38:55 +00:00
/* nothing to do if we aren't in a domain */
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
if ( ! ( IS_DC | | lp_server_role ( ) = = ROLE_DOMAIN_MEMBER ) ) {
return NT_STATUS_OK ;
}
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
/* Find the Domain Admins SID */
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
if ( IS_DC ) {
sid_copy ( & domadm , get_global_sam_sid ( ) ) ;
} else {
2008-07-23 20:42:32 -07:00
sid_copy ( & domadm , dom_sid ) ;
2007-04-09 10:38:55 +00:00
}
2010-05-17 22:04:24 +02:00
sid_append_rid ( & domadm , DOMAIN_RID_ADMINS ) ;
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
/* Add Administrators if the user beloongs to Domain Admins */
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
if ( nt_token_check_sid ( & domadm , token ) ) {
2008-01-09 00:11:31 +01:00
status = add_sid_to_array ( token ,
& global_sid_Builtin_Administrators ,
2010-08-31 09:32:52 +10:00
& token - > sids , & token - > num_sids ) ;
2008-01-09 00:11:31 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2007-04-09 10:38:55 +00:00
}
}
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
return NT_STATUS_OK ;
}
2010-08-26 20:04:11 +10:00
static NTSTATUS finalize_local_nt_token ( struct security_token * result ,
2010-05-28 16:16:53 -04:00
bool is_guest ) ;
NTSTATUS create_local_nt_token_from_info3 ( TALLOC_CTX * mem_ctx ,
bool is_guest ,
struct netr_SamInfo3 * info3 ,
struct extra_auth_info * extra ,
2010-08-26 20:04:11 +10:00
struct security_token * * ntok )
2010-05-28 16:16:53 -04:00
{
2010-08-26 20:04:11 +10:00
struct security_token * usrtok = NULL ;
2010-05-28 16:16:53 -04:00
NTSTATUS status ;
int i ;
DEBUG ( 10 , ( " Create local NT token for %s \n " ,
info3 - > base . account_name . string ) ) ;
2010-08-26 20:04:11 +10:00
usrtok = talloc_zero ( mem_ctx , struct security_token ) ;
2010-05-28 16:16:53 -04:00
if ( ! usrtok ) {
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
/* Add the user and primary group sid FIRST */
/* check if the user rid is the special "Domain Guests" rid.
* If so pick the first sid for the extra sids instead as it
* is a local fake account */
2010-08-31 09:32:52 +10:00
usrtok - > sids = talloc_array ( usrtok , struct dom_sid , 2 ) ;
if ( ! usrtok - > sids ) {
2010-05-28 16:16:53 -04:00
TALLOC_FREE ( usrtok ) ;
return NT_STATUS_NO_MEMORY ;
}
usrtok - > num_sids = 2 ;
/* USER SID */
if ( info3 - > base . rid = = ( uint32_t ) ( - 1 ) ) {
/* this is a signal the user was fake and generated,
* the actual SID we want to use is stored in the extra
* sids */
if ( is_null_sid ( & extra - > user_sid ) ) {
/* we couldn't find the user sid, bail out */
DEBUG ( 3 , ( " Invalid user SID \n " ) ) ;
TALLOC_FREE ( usrtok ) ;
return NT_STATUS_UNSUCCESSFUL ;
}
2010-08-31 09:32:52 +10:00
sid_copy ( & usrtok - > sids [ 0 ] , & extra - > user_sid ) ;
2010-05-28 16:16:53 -04:00
} else {
2010-08-31 09:32:52 +10:00
sid_copy ( & usrtok - > sids [ 0 ] , info3 - > base . domain_sid ) ;
sid_append_rid ( & usrtok - > sids [ 0 ] , info3 - > base . rid ) ;
2010-05-28 16:16:53 -04:00
}
/* GROUP SID */
if ( info3 - > base . primary_gid = = ( uint32_t ) ( - 1 ) ) {
/* this is a signal the user was fake and generated,
* the actual SID we want to use is stored in the extra
* sids */
if ( is_null_sid ( & extra - > pgid_sid ) ) {
/* we couldn't find the user sid, bail out */
DEBUG ( 3 , ( " Invalid group SID \n " ) ) ;
TALLOC_FREE ( usrtok ) ;
return NT_STATUS_UNSUCCESSFUL ;
}
2010-08-31 09:32:52 +10:00
sid_copy ( & usrtok - > sids [ 1 ] , & extra - > pgid_sid ) ;
2010-05-28 16:16:53 -04:00
} else {
2010-08-31 09:32:52 +10:00
sid_copy ( & usrtok - > sids [ 1 ] , info3 - > base . domain_sid ) ;
sid_append_rid ( & usrtok - > sids [ 1 ] ,
2010-05-28 16:16:53 -04:00
info3 - > base . primary_gid ) ;
}
/* Now the SIDs we got from authentication. These are the ones from
* the info3 struct or from the pdb_enum_group_memberships , depending
* on who authenticated the user .
* Note that we start the for loop at " 1 " here , we already added the
* first group sid as primary above . */
for ( i = 0 ; i < info3 - > base . groups . count ; i + + ) {
struct dom_sid tmp_sid ;
sid_copy ( & tmp_sid , info3 - > base . domain_sid ) ;
sid_append_rid ( & tmp_sid , info3 - > base . groups . rids [ i ] . rid ) ;
status = add_sid_to_array_unique ( usrtok , & tmp_sid ,
2010-08-31 09:32:52 +10:00
& usrtok - > sids ,
2010-05-28 16:16:53 -04:00
& usrtok - > num_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 3 , ( " Failed to add SID to nt token \n " ) ) ;
TALLOC_FREE ( usrtok ) ;
return status ;
}
}
/* now also add extra sids if they are not the special user/group
* sids */
for ( i = 0 ; i < info3 - > sidcount ; i + + ) {
status = add_sid_to_array_unique ( usrtok ,
info3 - > sids [ i ] . sid ,
2010-08-31 09:32:52 +10:00
& usrtok - > sids ,
2010-05-28 16:16:53 -04:00
& usrtok - > num_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 3 , ( " Failed to add SID to nt token \n " ) ) ;
TALLOC_FREE ( usrtok ) ;
return status ;
}
}
status = finalize_local_nt_token ( usrtok , is_guest ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 3 , ( " Failed to finalize nt token \n " ) ) ;
TALLOC_FREE ( usrtok ) ;
return status ;
}
* ntok = usrtok ;
return NT_STATUS_OK ;
}
2007-04-09 10:38:55 +00:00
/*******************************************************************
Create a NT token for the user , expanding local aliases
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-08-26 20:04:11 +10:00
struct security_token * create_local_nt_token ( TALLOC_CTX * mem_ctx ,
2010-05-21 11:25:01 +10:00
const struct dom_sid * user_sid ,
2007-10-18 17:40:25 -07:00
bool is_guest ,
2007-04-09 10:38:55 +00:00
int num_groupsids ,
2010-05-21 11:25:01 +10:00
const struct dom_sid * groupsids )
2007-04-09 10:38:55 +00:00
{
2010-08-26 20:04:11 +10:00
struct security_token * result = NULL ;
2007-04-09 10:38:55 +00:00
int i ;
NTSTATUS status ;
2007-12-15 21:11:36 +01:00
DEBUG ( 10 , ( " Create local NT token for %s \n " ,
sid_string_dbg ( user_sid ) ) ) ;
2007-04-09 10:38:55 +00:00
2011-06-07 11:44:43 +10:00
if ( ! ( result = talloc_zero ( mem_ctx , struct security_token ) ) ) {
2007-04-09 10:38:55 +00:00
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
return NULL ;
}
/* Add the user and primary group sid */
2008-01-09 00:11:31 +01:00
status = add_sid_to_array ( result , user_sid ,
2010-08-31 09:32:52 +10:00
& result - > sids , & result - > num_sids ) ;
2008-01-09 00:11:31 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-05-28 16:16:53 -04:00
TALLOC_FREE ( result ) ;
2007-04-09 10:38:55 +00:00
return NULL ;
}
/* For guest, num_groupsids may be zero. */
if ( num_groupsids ) {
2008-01-09 00:11:31 +01:00
status = add_sid_to_array ( result , & groupsids [ 0 ] ,
2010-08-31 09:32:52 +10:00
& result - > sids ,
2008-01-09 00:11:31 +01:00
& result - > num_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-05-28 16:16:53 -04:00
TALLOC_FREE ( result ) ;
2007-04-09 10:38:55 +00:00
return NULL ;
}
}
2007-12-17 10:34:29 +01:00
2010-05-28 16:16:53 -04:00
/* Now the SIDs we got from authentication. These are the ones from
* the info3 struct or from the pdb_enum_group_memberships , depending
* on who authenticated the user .
* Note that we start the for loop at " 1 " here , we already added the
* first group sid as primary above . */
for ( i = 1 ; i < num_groupsids ; i + + ) {
status = add_sid_to_array_unique ( result , & groupsids [ i ] ,
2010-08-31 09:32:52 +10:00
& result - > sids ,
2010-05-28 16:16:53 -04:00
& result - > num_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( result ) ;
return NULL ;
}
}
status = finalize_local_nt_token ( result , is_guest ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( result ) ;
return NULL ;
}
return result ;
}
2010-08-26 20:04:11 +10:00
static NTSTATUS finalize_local_nt_token ( struct security_token * result ,
2010-05-28 16:16:53 -04:00
bool is_guest )
{
struct dom_sid dom_sid ;
gid_t gid ;
NTSTATUS status ;
2007-04-09 10:38:55 +00:00
/* Add in BUILTIN sids */
2007-12-17 10:34:29 +01:00
2008-01-09 00:11:31 +01:00
status = add_sid_to_array ( result , & global_sid_World ,
2010-08-31 09:32:52 +10:00
& result - > sids , & result - > num_sids ) ;
2008-01-09 00:11:31 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-05-28 16:16:53 -04:00
return status ;
2007-04-09 10:38:55 +00:00
}
2008-01-09 00:11:31 +01:00
status = add_sid_to_array ( result , & global_sid_Network ,
2010-08-31 09:32:52 +10:00
& result - > sids , & result - > num_sids ) ;
2008-01-09 00:11:31 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-05-28 16:16:53 -04:00
return status ;
2007-04-09 10:38:55 +00:00
}
if ( is_guest ) {
2008-01-09 00:11:31 +01:00
status = add_sid_to_array ( result , & global_sid_Builtin_Guests ,
2010-08-31 09:32:52 +10:00
& result - > sids ,
2008-01-09 00:11:31 +01:00
& result - > num_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-05-28 16:16:53 -04:00
return status ;
2007-04-09 10:38:55 +00:00
}
} else {
2008-01-09 00:11:31 +01:00
status = add_sid_to_array ( result ,
& global_sid_Authenticated_Users ,
2010-08-31 09:32:52 +10:00
& result - > sids ,
2008-01-09 00:11:31 +01:00
& result - > num_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-05-28 16:16:53 -04:00
return status ;
2007-04-09 10:38:55 +00:00
}
}
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
/* Deal with the BUILTIN\Administrators group. If the SID can
2007-12-17 10:34:29 +01:00
be resolved then assume that the add_aliasmem ( S - 1 - 5 - 32 )
2007-04-09 10:38:55 +00:00
handled it . */
2008-07-23 20:42:32 -07:00
if ( ! sid_to_gid ( & global_sid_Builtin_Administrators , & gid ) ) {
2007-12-17 10:34:29 +01:00
2008-07-23 20:42:32 -07:00
become_root ( ) ;
if ( ! secrets_fetch_domain_sid ( lp_workgroup ( ) , & dom_sid ) ) {
status = NT_STATUS_OK ;
DEBUG ( 3 , ( " Failed to fetch domain sid for %s \n " ,
lp_workgroup ( ) ) ) ;
} else {
status = create_builtin_administrators ( & dom_sid ) ;
2007-04-09 10:38:55 +00:00
}
2008-07-23 20:42:32 -07:00
unbecome_root ( ) ;
if ( NT_STATUS_EQUAL ( status , NT_STATUS_PROTOCOL_UNREACHABLE ) ) {
/* Add BUILTIN\Administrators directly to token. */
status = add_builtin_administrators ( result , & dom_sid ) ;
2007-04-09 10:38:55 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-07-23 20:42:32 -07:00
DEBUG ( 3 , ( " Failed to check for local "
" Administrators membership (%s) \n " ,
nt_errstr ( status ) ) ) ;
2007-12-17 10:34:29 +01:00
}
2008-07-23 20:42:32 -07:00
} else if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 2 , ( " WARNING: Failed to create "
" BUILTIN \\ Administrators group! Can "
" Winbind allocate gids? \n " ) ) ;
2007-12-17 10:34:29 +01:00
}
2007-04-09 10:38:55 +00:00
}
/* Deal with the BUILTIN\Users group. If the SID can
2007-12-17 10:34:29 +01:00
be resolved then assume that the add_aliasmem ( S - 1 - 5 - 32 )
2007-04-09 10:38:55 +00:00
handled it . */
2008-07-23 20:33:15 -07:00
if ( ! sid_to_gid ( & global_sid_Builtin_Users , & gid ) ) {
2007-12-17 10:34:29 +01:00
2008-07-23 20:33:15 -07:00
become_root ( ) ;
if ( ! secrets_fetch_domain_sid ( lp_workgroup ( ) , & dom_sid ) ) {
status = NT_STATUS_OK ;
DEBUG ( 3 , ( " Failed to fetch domain sid for %s \n " ,
lp_workgroup ( ) ) ) ;
} else {
status = create_builtin_users ( & dom_sid ) ;
}
unbecome_root ( ) ;
if ( ! NT_STATUS_EQUAL ( status , NT_STATUS_PROTOCOL_UNREACHABLE ) & &
! NT_STATUS_IS_OK ( status ) )
{
DEBUG ( 2 , ( " WARNING: Failed to create BUILTIN \\ Users group! "
" Can Winbind allocate gids? \n " ) ) ;
2007-04-09 10:38:55 +00:00
}
}
/* Deal with local groups */
2007-12-17 10:34:29 +01:00
2007-04-09 10:38:55 +00:00
if ( lp_winbind_nested_groups ( ) ) {
2007-10-10 08:27:56 +00:00
become_root ( ) ;
2007-04-09 10:38:55 +00:00
/* Now add the aliases. First the one from our local SAM */
status = add_aliases ( get_global_sam_sid ( ) , result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2007-12-10 11:47:17 +01:00
unbecome_root ( ) ;
2010-05-28 16:16:53 -04:00
return status ;
2007-04-09 10:38:55 +00:00
}
/* Finally the builtin ones */
status = add_aliases ( & global_sid_Builtin , result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2007-12-10 11:47:17 +01:00
unbecome_root ( ) ;
2010-05-28 16:16:53 -04:00
return status ;
2007-04-09 10:38:55 +00:00
}
2007-10-10 08:27:56 +00:00
unbecome_root ( ) ;
2007-12-17 10:34:29 +01:00
}
2007-04-09 10:38:55 +00:00
2010-05-28 16:16:53 -04:00
/* Add privileges based on current user sids */
2007-04-09 10:38:55 +00:00
2010-08-31 11:18:11 +10:00
get_privileges_for_sids ( & result - > privilege_mask , result - > sids ,
2007-04-09 10:38:55 +00:00
result - > num_sids ) ;
2010-05-28 16:16:53 -04:00
return NT_STATUS_OK ;
2007-04-09 10:38:55 +00:00
}
2007-05-07 12:15:11 +00:00
/****************************************************************************
prints a UNIX ' token ' to debug output .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void debug_unix_user_token ( int dbg_class , int dbg_lev , uid_t uid , gid_t gid ,
int n_groups , gid_t * groups )
{
int i ;
DEBUGC ( dbg_class , dbg_lev ,
( " UNIX token of user %ld \n " , ( long int ) uid ) ) ;
DEBUGADDC ( dbg_class , dbg_lev ,
( " Primary group is %ld and contains %i supplementary "
" groups \n " , ( long int ) gid , n_groups ) ) ;
for ( i = 0 ; i < n_groups ; i + + )
2007-12-17 10:34:29 +01:00
DEBUGADDC ( dbg_class , dbg_lev , ( " Group[%3i]: %ld \n " , i ,
2007-05-07 12:15:11 +00:00
( long int ) groups [ i ] ) ) ;
}
2010-04-11 13:48:55 +02:00
/*
* Create an artificial NT token given just a username . ( Initially intended
* for force user )
*
* We go through lookup_name ( ) to avoid problems we had with ' winbind use
* default domain ' .
*
* We have 3 cases :
*
* unmapped unix users : Go directly to nss to find the user ' s group .
*
* A passdb user : The list of groups is provided by pdb_enum_group_memberships .
*
* If the user is provided by winbind , the primary gid is set to " domain
* users " of the user's domain. For an explanation why this is necessary, see
* the thread starting at
* http : //lists.samba.org/archive/samba-technical/2006-January/044803.html.
*/
NTSTATUS create_token_from_username ( TALLOC_CTX * mem_ctx , const char * username ,
bool is_guest ,
uid_t * uid , gid_t * gid ,
char * * found_username ,
2010-08-26 20:04:11 +10:00
struct security_token * * token )
2010-04-11 13:48:55 +02:00
{
NTSTATUS result = NT_STATUS_NO_SUCH_USER ;
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
2010-05-21 11:25:01 +10:00
struct dom_sid user_sid ;
2010-04-11 13:48:55 +02:00
enum lsa_SidType type ;
gid_t * gids ;
2010-05-21 11:25:01 +10:00
struct dom_sid * group_sids ;
struct dom_sid unix_group_sid ;
2010-08-26 20:54:13 +10:00
uint32_t num_group_sids ;
2011-02-21 10:30:28 +01:00
uint32_t num_gids ;
uint32_t i ;
2010-04-11 13:48:55 +02:00
if ( ! lookup_name_smbconf ( tmp_ctx , username , LOOKUP_NAME_ALL ,
NULL , NULL , & user_sid , & type ) ) {
DEBUG ( 1 , ( " lookup_name_smbconf for %s failed \n " , username ) ) ;
goto done ;
}
if ( type ! = SID_NAME_USER ) {
DEBUG ( 1 , ( " %s is a %s, not a user \n " , username ,
sid_type_lookup ( type ) ) ) ;
goto done ;
}
if ( sid_check_is_in_our_domain ( & user_sid ) ) {
bool ret ;
2011-02-21 10:30:28 +01:00
uint32_t pdb_num_group_sids ;
2010-04-11 13:48:55 +02:00
/* This is a passdb user, so ask passdb */
struct samu * sam_acct = NULL ;
if ( ! ( sam_acct = samu_new ( tmp_ctx ) ) ) {
result = NT_STATUS_NO_MEMORY ;
goto done ;
}
become_root ( ) ;
ret = pdb_getsampwsid ( sam_acct , & user_sid ) ;
unbecome_root ( ) ;
if ( ! ret ) {
DEBUG ( 1 , ( " pdb_getsampwsid(%s) for user %s failed \n " ,
sid_string_dbg ( & user_sid ) , username ) ) ;
DEBUGADD ( 1 , ( " Fall back to unix user %s \n " , username ) ) ;
goto unix_user ;
}
result = pdb_enum_group_memberships ( tmp_ctx , sam_acct ,
& group_sids , & gids ,
2010-08-26 20:54:13 +10:00
& pdb_num_group_sids ) ;
2010-04-11 13:48:55 +02:00
if ( ! NT_STATUS_IS_OK ( result ) ) {
DEBUG ( 1 , ( " enum_group_memberships failed for %s (%s): "
" %s \n " , username , sid_string_dbg ( & user_sid ) ,
nt_errstr ( result ) ) ) ;
DEBUGADD ( 1 , ( " Fall back to unix user %s \n " , username ) ) ;
goto unix_user ;
}
2010-08-26 20:54:13 +10:00
num_group_sids = pdb_num_group_sids ;
2010-04-11 13:48:55 +02:00
/* see the smb_panic() in pdb_default_enum_group_memberships */
SMB_ASSERT ( num_group_sids > 0 ) ;
* gid = gids [ 0 ] ;
/* Ensure we're returning the found_username on the right context. */
* found_username = talloc_strdup ( mem_ctx ,
pdb_get_username ( sam_acct ) ) ;
/*
* If the SID from lookup_name ( ) was the guest sid , passdb knows
* about the mapping of guest sid to lp_guestaccount ( )
* username and will return the unix_pw info for a guest
* user . Use it if it ' s there , else lookup the * uid details
2010-10-20 08:16:23 -07:00
* using Get_Pwnam_alloc ( ) . See bug # 6291 for details . JRA .
2010-04-11 13:48:55 +02:00
*/
/* We must always assign the *uid. */
if ( sam_acct - > unix_pw = = NULL ) {
2010-10-20 08:16:23 -07:00
struct passwd * pwd = Get_Pwnam_alloc ( sam_acct , * found_username ) ;
2010-04-11 13:48:55 +02:00
if ( ! pwd ) {
2010-10-20 08:16:23 -07:00
DEBUG ( 10 , ( " Get_Pwnam_alloc failed for %s \n " ,
2010-04-11 13:48:55 +02:00
* found_username ) ) ;
result = NT_STATUS_NO_SUCH_USER ;
goto done ;
}
result = samu_set_unix ( sam_acct , pwd ) ;
if ( ! NT_STATUS_IS_OK ( result ) ) {
DEBUG ( 10 , ( " samu_set_unix failed for %s \n " ,
* found_username ) ) ;
result = NT_STATUS_NO_SUCH_USER ;
goto done ;
}
}
* uid = sam_acct - > unix_pw - > pw_uid ;
} else if ( sid_check_is_in_unix_users ( & user_sid ) ) {
2011-02-21 10:30:28 +01:00
uint32_t getgroups_num_group_sids ;
2010-04-11 13:48:55 +02:00
/* This is a unix user not in passdb. We need to ask nss
* directly , without consulting passdb */
struct passwd * pass ;
/*
* This goto target is used as a fallback for the passdb
* case . The concrete bug report is when passdb gave us an
* unmapped gid .
*/
unix_user :
if ( ! sid_to_uid ( & user_sid , uid ) ) {
DEBUG ( 1 , ( " unix_user case, sid_to_uid for %s (%s) failed \n " ,
username , sid_string_dbg ( & user_sid ) ) ) ;
result = NT_STATUS_NO_SUCH_USER ;
goto done ;
}
uid_to_unix_users_sid ( * uid , & user_sid ) ;
pass = getpwuid_alloc ( tmp_ctx , * uid ) ;
if ( pass = = NULL ) {
DEBUG ( 1 , ( " getpwuid(%u) for user %s failed \n " ,
( unsigned int ) * uid , username ) ) ;
goto done ;
}
if ( ! getgroups_unix_user ( tmp_ctx , username , pass - > pw_gid ,
2010-08-26 20:54:13 +10:00
& gids , & getgroups_num_group_sids ) ) {
2010-04-11 13:48:55 +02:00
DEBUG ( 1 , ( " getgroups_unix_user for user %s failed \n " ,
username ) ) ;
goto done ;
}
2010-08-26 20:54:13 +10:00
num_group_sids = getgroups_num_group_sids ;
2010-04-11 13:48:55 +02:00
if ( num_group_sids ) {
2011-06-07 11:30:12 +10:00
group_sids = talloc_array ( tmp_ctx , struct dom_sid , num_group_sids ) ;
2010-04-11 13:48:55 +02:00
if ( group_sids = = NULL ) {
2011-06-07 11:30:12 +10:00
DEBUG ( 1 , ( " talloc_array failed \n " ) ) ;
2010-04-11 13:48:55 +02:00
result = NT_STATUS_NO_MEMORY ;
goto done ;
}
} else {
group_sids = NULL ;
}
for ( i = 0 ; i < num_group_sids ; i + + ) {
gid_to_sid ( & group_sids [ i ] , gids [ i ] ) ;
}
/* In getgroups_unix_user we always set the primary gid */
SMB_ASSERT ( num_group_sids > 0 ) ;
* gid = gids [ 0 ] ;
/* Ensure we're returning the found_username on the right context. */
* found_username = talloc_strdup ( mem_ctx , pass - > pw_name ) ;
} else {
/* This user is from winbind, force the primary gid to the
* user ' s " domain users " group . Under certain circumstances
* ( user comes from NT4 ) , this might be a loss of
* information . But we can not rely on winbind getting the
* correct info . AD might prohibit winbind looking up that
* information . */
/* We must always assign the *uid. */
if ( ! sid_to_uid ( & user_sid , uid ) ) {
DEBUG ( 1 , ( " winbindd case, sid_to_uid for %s (%s) failed \n " ,
username , sid_string_dbg ( & user_sid ) ) ) ;
result = NT_STATUS_NO_SUCH_USER ;
goto done ;
}
num_group_sids = 1 ;
2011-06-07 11:30:12 +10:00
group_sids = talloc_array ( tmp_ctx , struct dom_sid , num_group_sids ) ;
2010-04-11 13:48:55 +02:00
if ( group_sids = = NULL ) {
2011-06-07 11:30:12 +10:00
DEBUG ( 1 , ( " talloc_array failed \n " ) ) ;
2010-04-11 13:48:55 +02:00
result = NT_STATUS_NO_MEMORY ;
goto done ;
}
sid_copy ( & group_sids [ 0 ] , & user_sid ) ;
2011-03-10 16:19:17 +01:00
sid_split_rid ( & group_sids [ 0 ] , NULL ) ;
2010-05-17 22:04:24 +02:00
sid_append_rid ( & group_sids [ 0 ] , DOMAIN_RID_USERS ) ;
2010-04-11 13:48:55 +02:00
if ( ! sid_to_gid ( & group_sids [ 0 ] , gid ) ) {
DEBUG ( 1 , ( " sid_to_gid(%s) failed \n " ,
sid_string_dbg ( & group_sids [ 0 ] ) ) ) ;
goto done ;
}
gids = gid ;
/* Ensure we're returning the found_username on the right context. */
* found_username = talloc_strdup ( mem_ctx , username ) ;
}
/* Add the "Unix Group" SID for each gid to catch mapped groups
and their Unix equivalent . This is to solve the backwards
compatibility problem of ' valid users = + ntadmin ' where
ntadmin has been paired with " Domain Admins " in the group
mapping table . Otherwise smb . conf would need to be changed
to ' valid user = " Domain Admins " ' . - - jerry */
num_gids = num_group_sids ;
for ( i = 0 ; i < num_gids ; i + + ) {
gid_t high , low ;
/* don't pickup anything managed by Winbind */
if ( lp_idmap_gid ( & low , & high ) & & ( gids [ i ] > = low ) & & ( gids [ i ] < = high ) )
continue ;
2010-07-11 17:30:53 +02:00
gid_to_unix_groups_sid ( gids [ i ] , & unix_group_sid ) ;
2010-04-11 13:48:55 +02:00
result = add_sid_to_array_unique ( tmp_ctx , & unix_group_sid ,
& group_sids , & num_group_sids ) ;
if ( ! NT_STATUS_IS_OK ( result ) ) {
goto done ;
}
}
/* Ensure we're creating the nt_token on the right context. */
* token = create_local_nt_token ( mem_ctx , & user_sid ,
is_guest , num_group_sids , group_sids ) ;
if ( ( * token = = NULL ) | | ( * found_username = = NULL ) ) {
result = NT_STATUS_NO_MEMORY ;
goto done ;
}
result = NT_STATUS_OK ;
done :
TALLOC_FREE ( tmp_ctx ) ;
return result ;
}
/***************************************************************************
Build upon create_token_from_username :
Expensive helper function to figure out whether a user given its name is
member of a particular group .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-05-21 11:25:01 +10:00
bool user_in_group_sid ( const char * username , const struct dom_sid * group_sid )
2010-04-11 13:48:55 +02:00
{
NTSTATUS status ;
uid_t uid ;
gid_t gid ;
char * found_username ;
2010-08-26 20:04:11 +10:00
struct security_token * token ;
2010-04-11 13:48:55 +02:00
bool result ;
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
status = create_token_from_username ( mem_ctx , username , False ,
& uid , & gid , & found_username ,
& token ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 10 , ( " could not create token for %s \n " , username ) ) ;
TALLOC_FREE ( mem_ctx ) ;
return False ;
}
2010-09-17 14:55:56 +10:00
result = security_token_has_sid ( token , group_sid ) ;
2010-04-11 13:48:55 +02:00
TALLOC_FREE ( mem_ctx ) ;
return result ;
}
bool user_in_group ( const char * username , const char * groupname )
{
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2010-05-21 11:25:01 +10:00
struct dom_sid group_sid ;
2010-04-11 13:48:55 +02:00
bool ret ;
ret = lookup_name ( mem_ctx , groupname , LOOKUP_NAME_ALL ,
NULL , NULL , & group_sid , NULL ) ;
TALLOC_FREE ( mem_ctx ) ;
if ( ! ret ) {
DEBUG ( 10 , ( " lookup_name for (%s) failed. \n " , groupname ) ) ;
return False ;
}
return user_in_group_sid ( username , & group_sid ) ;
}
2007-04-09 10:38:55 +00:00
/* END */