2006-12-12 14:52:13 +00:00
/*
* idmap_rid : static map between Active Directory / NT RIDs and RFC 2307 accounts
* Copyright ( C ) Guenther Deschner , 2004
* Copyright ( C ) Sumit Bose , 2004
*
* 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-12-12 14:52:13 +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-12-12 14:52:13 +00:00
*
*/
# include "includes.h"
2006-12-12 15:16:26 +00:00
# include "winbindd.h"
2006-12-12 14:52:13 +00:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_IDMAP
struct idmap_rid_context {
2007-01-22 16:54:02 +00:00
const char * domain_name ;
2006-12-12 14:52:13 +00:00
uint32_t low_id ;
uint32_t high_id ;
uint32_t base_rid ;
} ;
2007-01-22 16:54:02 +00:00
/******************************************************************************
compat params can ' t be used because of the completely different way
we support multiple domains in the new idmap
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-04-19 22:26:09 +00:00
static NTSTATUS idmap_rid_initialize ( struct idmap_domain * dom )
2006-12-12 14:52:13 +00:00
{
NTSTATUS ret ;
struct idmap_rid_context * ctx ;
char * config_option = NULL ;
const char * range ;
2007-06-16 18:59:02 +00:00
uid_t low_uid = 0 ;
uid_t high_uid = 0 ;
gid_t low_gid = 0 ;
gid_t high_gid = 0 ;
2006-12-12 14:52:13 +00:00
2007-04-28 12:50:35 +00:00
if ( ( ctx = TALLOC_ZERO_P ( dom , struct idmap_rid_context ) ) = = NULL ) {
2006-12-12 14:52:13 +00:00
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
config_option = talloc_asprintf ( ctx , " idmap config %s " , dom - > name ) ;
if ( ! config_option ) {
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
ret = NT_STATUS_NO_MEMORY ;
goto failed ;
}
range = lp_parm_const_string ( - 1 , config_option , " range " , NULL ) ;
2007-01-22 16:54:02 +00:00
if ( ! range | |
2006-12-12 14:52:13 +00:00
( sscanf ( range , " %u - %u " , & ctx - > low_id , & ctx - > high_id ) ! = 2 ) | |
2007-01-22 16:54:02 +00:00
( ctx - > low_id > ctx - > high_id ) )
{
2006-12-12 14:52:13 +00:00
ctx - > low_id = 0 ;
ctx - > high_id = 0 ;
}
2007-06-16 18:59:02 +00:00
/* lets see if the range is defined by the old idmap uid/idmap gid */
if ( ! ctx - > low_id & & ! ctx - > high_id ) {
if ( lp_idmap_uid ( & low_uid , & high_uid ) ) {
ctx - > low_id = low_uid ;
ctx - > high_id = high_uid ;
}
if ( lp_idmap_gid ( & low_gid , & high_gid ) ) {
if ( ( ctx - > low_id ! = low_gid ) | |
( ctx - > high_id ! = high_uid ) ) {
2007-06-26 18:18:44 +00:00
DEBUG ( 1 , ( " ERROR: idmap uid range must match idmap gid range \n " ) ) ;
2007-06-16 18:59:02 +00:00
ret = NT_STATUS_UNSUCCESSFUL ;
goto failed ;
}
}
}
if ( ! ctx - > low_id | | ! ctx - > high_id ) {
DEBUG ( 1 , ( " ERROR: Invalid configuration, ID range missing or invalid \n " ) ) ;
2006-12-12 14:52:13 +00:00
ret = NT_STATUS_UNSUCCESSFUL ;
goto failed ;
}
ctx - > base_rid = lp_parm_int ( - 1 , config_option , " base_rid " , 0 ) ;
2007-01-22 16:54:02 +00:00
ctx - > domain_name = talloc_strdup ( ctx , dom - > name ) ;
2006-12-12 14:52:13 +00:00
dom - > private_data = ctx ;
2007-06-16 18:59:02 +00:00
dom - > initialized = True ;
2006-12-12 14:52:13 +00:00
talloc_free ( config_option ) ;
return NT_STATUS_OK ;
failed :
talloc_free ( ctx ) ;
return ret ;
}
2006-12-13 16:39:50 +00:00
static NTSTATUS idmap_rid_id_to_sid ( TALLOC_CTX * memctx , struct idmap_rid_context * ctx , struct id_map * map )
2006-12-12 14:52:13 +00:00
{
2007-01-22 16:54:02 +00:00
struct winbindd_domain * domain ;
2006-12-12 14:52:13 +00:00
/* apply filters before checking */
if ( ( map - > xid . id < ctx - > low_id ) | | ( map - > xid . id > ctx - > high_id ) ) {
DEBUG ( 5 , ( " Requested id (%u) out of range (%u - %u). Filtered! \n " ,
map - > xid . id , ctx - > low_id , ctx - > high_id ) ) ;
return NT_STATUS_NONE_MAPPED ;
}
2007-01-22 16:54:02 +00:00
if ( ( domain = find_domain_from_name_noinit ( ctx - > domain_name ) ) = = NULL ) {
return NT_STATUS_NO_SUCH_DOMAIN ;
}
sid_compose ( map - > sid , & domain - > sid , map - > xid . id - ctx - > low_id + ctx - > base_rid ) ;
2006-12-12 14:52:13 +00:00
2007-04-19 22:26:09 +00:00
/* We **really** should have some way of validating
the SID exists and is the correct type here . But
that is a deficiency in the idmap_rid design . */
2006-12-12 14:52:13 +00:00
2007-01-14 17:58:24 +00:00
map - > status = ID_MAPPED ;
2006-12-12 14:52:13 +00:00
return NT_STATUS_OK ;
}
/**********************************
Single sid to id lookup function .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-12-13 16:39:50 +00:00
static NTSTATUS idmap_rid_sid_to_id ( TALLOC_CTX * memctx , struct idmap_rid_context * ctx , struct id_map * map )
2006-12-12 14:52:13 +00:00
{
uint32_t rid ;
sid_peek_rid ( map - > sid , & rid ) ;
map - > xid . id = rid - ctx - > base_rid + ctx - > low_id ;
/* apply filters before returning result */
2007-04-19 22:26:09 +00:00
2006-12-12 14:52:13 +00:00
if ( ( map - > xid . id < ctx - > low_id ) | | ( map - > xid . id > ctx - > high_id ) ) {
DEBUG ( 5 , ( " Requested id (%u) out of range (%u - %u). Filtered! \n " ,
map - > xid . id , ctx - > low_id , ctx - > high_id ) ) ;
2007-01-14 17:58:24 +00:00
map - > status = ID_UNMAPPED ;
2006-12-12 14:52:13 +00:00
return NT_STATUS_NONE_MAPPED ;
}
2007-04-19 22:26:09 +00:00
/* We **really** should have some way of validating
the SID exists and is the correct type here . But
that is a deficiency in the idmap_rid design . */
2007-01-14 17:58:24 +00:00
map - > status = ID_MAPPED ;
2006-12-12 14:52:13 +00:00
return NT_STATUS_OK ;
}
/**********************************
lookup a set of unix ids .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS idmap_rid_unixids_to_sids ( struct idmap_domain * dom , struct id_map * * ids )
{
2006-12-13 16:39:50 +00:00
struct idmap_rid_context * ridctx ;
TALLOC_CTX * ctx ;
2006-12-12 14:52:13 +00:00
NTSTATUS ret ;
int i ;
2007-06-16 18:59:02 +00:00
/* Initilization my have been deferred because of an error, retry or fail */
if ( ! dom - > initialized ) {
ret = idmap_rid_initialize ( dom ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
return ret ;
}
}
2006-12-13 16:39:50 +00:00
ridctx = talloc_get_type ( dom - > private_data , struct idmap_rid_context ) ;
ctx = talloc_new ( dom ) ;
if ( ! ctx ) {
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
2006-12-12 14:52:13 +00:00
for ( i = 0 ; ids [ i ] ; i + + ) {
2006-12-13 16:39:50 +00:00
ret = idmap_rid_id_to_sid ( ctx , ridctx , ids [ i ] ) ;
2006-12-12 14:52:13 +00:00
if ( ( ! NT_STATUS_IS_OK ( ret ) ) & &
( ! NT_STATUS_EQUAL ( ret , NT_STATUS_NONE_MAPPED ) ) ) {
/* some fatal error occurred, log it */
DEBUG ( 3 , ( " Unexpected error resolving an ID (%d) \n " , ids [ i ] - > xid . id ) ) ;
}
}
2006-12-13 16:39:50 +00:00
talloc_free ( ctx ) ;
2006-12-12 14:52:13 +00:00
return NT_STATUS_OK ;
}
/**********************************
lookup a set of sids .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS idmap_rid_sids_to_unixids ( struct idmap_domain * dom , struct id_map * * ids )
{
2006-12-13 16:39:50 +00:00
struct idmap_rid_context * ridctx ;
TALLOC_CTX * ctx ;
2006-12-12 14:52:13 +00:00
NTSTATUS ret ;
int i ;
2007-06-16 18:59:02 +00:00
/* Initilization my have been deferred because of an error, retry or fail */
if ( ! dom - > initialized ) {
ret = idmap_rid_initialize ( dom ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
return ret ;
}
}
2006-12-13 16:39:50 +00:00
ridctx = talloc_get_type ( dom - > private_data , struct idmap_rid_context ) ;
ctx = talloc_new ( dom ) ;
if ( ! ctx ) {
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
2006-12-12 14:52:13 +00:00
for ( i = 0 ; ids [ i ] ; i + + ) {
2006-12-13 16:39:50 +00:00
ret = idmap_rid_sid_to_id ( ctx , ridctx , ids [ i ] ) ;
2006-12-12 14:52:13 +00:00
if ( ( ! NT_STATUS_IS_OK ( ret ) ) & &
( ! NT_STATUS_EQUAL ( ret , NT_STATUS_NONE_MAPPED ) ) ) {
/* some fatal error occurred, log it */
DEBUG ( 3 , ( " Unexpected error resolving a SID (%s) \n " ,
2007-12-15 21:11:36 +01:00
sid_string_dbg ( ids [ i ] - > sid ) ) ) ;
2006-12-12 14:52:13 +00:00
}
}
2006-12-13 16:39:50 +00:00
talloc_free ( ctx ) ;
2006-12-12 14:52:13 +00:00
return NT_STATUS_OK ;
}
static NTSTATUS idmap_rid_close ( struct idmap_domain * dom )
{
if ( dom - > private_data ) {
TALLOC_FREE ( dom - > private_data ) ;
}
return NT_STATUS_OK ;
}
static struct idmap_methods rid_methods = {
. init = idmap_rid_initialize ,
. unixids_to_sids = idmap_rid_unixids_to_sids ,
. sids_to_unixids = idmap_rid_sids_to_unixids ,
. close_fn = idmap_rid_close
} ;
NTSTATUS idmap_rid_init ( void )
{
return smb_register_idmap ( SMB_IDMAP_INTERFACE_VERSION , " rid " , & rid_methods ) ;
}