2014-12-24 04:33:34 +03:00
/*
Unix SMB / CIFS implementation .
idmap script backend , used for Samba setups where you need to map SIDs to
specific UIDs / GIDs .
Copyright ( C ) Richard Sharpe 2014.
This is heavily based upon idmap_tdb2 . c , which is :
Copyright ( C ) Tim Potter 2000
Copyright ( C ) Jim McDonough < jmcd @ us . ibm . com > 2003
Copyright ( C ) Jeremy Allison 2006
Copyright ( C ) Simo Sorce 2003 - 2006
Copyright ( C ) Michael Adam 2009 - 2010
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 2 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 , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
# include "system/filesys.h"
# include "winbindd.h"
# include "idmap.h"
# include "idmap_rw.h"
# include "../libcli/security/dom_sid.h"
2016-02-16 19:09:43 +03:00
# include "lib/util_file.h"
2016-02-17 16:01:47 +03:00
# include "lib/util/tevent_unix.h"
2014-12-24 04:33:34 +03:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_IDMAP
struct idmap_script_context {
const char * script ; /* script to provide idmaps */
} ;
/*
run a script to perform a mapping
The script should accept the following command lines :
SIDTOID S - 1 - xxxx - > XID : < id > | ERR : < str >
SIDTOID S - 1 - xxxx - > UID : < id > | ERR : < str >
SIDTOID S - 1 - xxxx - > GID : < id > | ERR : < str >
IDTOSID XID xxxx - > SID : < sid > | ERR : < str >
IDTOSID UID xxxx - > SID : < sid > | ERR : < str >
IDTOSID GID xxxx - > SID : < sid > | ERR : < str >
where XID means both a UID and a GID . This is the case for ID_TYPE_BOTH .
TODO : Needs more validation . . . like that we got a UID when we asked for one .
*/
2016-02-17 16:01:47 +03:00
struct idmap_script_xid2sid_state {
const char * syscmd ;
size_t idx ;
uint8_t * out ;
} ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
static void idmap_script_xid2sid_done ( struct tevent_req * subreq ) ;
static struct tevent_req * idmap_script_xid2sid_send (
TALLOC_CTX * mem_ctx , struct tevent_context * ev ,
struct unixid xid , const char * script , size_t idx )
{
struct tevent_req * req , * subreq ;
struct idmap_script_xid2sid_state * state ;
char key ;
req = tevent_req_create ( mem_ctx , & state ,
struct idmap_script_xid2sid_state ) ;
if ( req = = NULL ) {
return NULL ;
}
state - > idx = idx ;
switch ( xid . type ) {
case ID_TYPE_UID :
key = ' U ' ;
break ;
case ID_TYPE_GID :
key = ' G ' ;
break ;
case ID_TYPE_BOTH :
key = ' X ' ;
break ;
default :
DBG_WARNING ( " INVALID unix ID type: 0x02%x \n " , xid . type ) ;
tevent_req_error ( req , EINVAL ) ;
return tevent_req_post ( req , ev ) ;
2014-12-24 04:33:34 +03:00
}
2016-08-26 18:16:51 +03:00
state - > syscmd = talloc_asprintf ( state , " %s IDTOSID %cID %lu " , script , key ,
2016-02-17 16:01:47 +03:00
( unsigned long ) xid . id ) ;
if ( tevent_req_nomem ( state - > syscmd , req ) ) {
return tevent_req_post ( req , ev ) ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
subreq = file_pload_send ( state , ev , state - > syscmd , 1024 ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
tevent_req_set_callback ( subreq , idmap_script_xid2sid_done , req ) ;
return req ;
}
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
static void idmap_script_xid2sid_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct idmap_script_xid2sid_state * state = tevent_req_data (
req , struct idmap_script_xid2sid_state ) ;
int ret ;
ret = file_pload_recv ( subreq , state , & state - > out ) ;
TALLOC_FREE ( subreq ) ;
if ( tevent_req_error ( req , ret ) ) {
return ;
}
tevent_req_done ( req ) ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
static int idmap_script_xid2sid_recv ( struct tevent_req * req , size_t * idx ,
enum id_mapping * status ,
struct dom_sid * sid )
2014-12-24 04:33:34 +03:00
{
2016-02-17 16:01:47 +03:00
struct idmap_script_xid2sid_state * state = tevent_req_data (
req , struct idmap_script_xid2sid_state ) ;
char * out = ( char * ) state - > out ;
size_t out_size = talloc_get_size ( out ) ;
int err ;
if ( tevent_req_is_unix_error ( req , & err ) ) {
return err ;
}
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
if ( out_size = = 0 ) {
goto unmapped ;
}
if ( state - > out [ out_size - 1 ] ! = ' \0 ' ) {
goto unmapped ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
* idx = state - > idx ;
if ( ( strncmp ( out , " SID:S- " , 6 ) ! = 0 ) | |
! dom_sid_parse ( out + 4 , sid ) ) {
DBG_WARNING ( " Bad sid from script: %s \n " , out ) ;
goto unmapped ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
* status = ID_MAPPED ;
return 0 ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
unmapped :
* sid = ( struct dom_sid ) { 0 } ;
* status = ID_UNMAPPED ;
return 0 ;
}
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
struct idmap_script_xids2sids_state {
struct id_map * * ids ;
size_t num_ids ;
size_t num_done ;
} ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
static void idmap_script_xids2sids_done ( struct tevent_req * subreq ) ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
static struct tevent_req * idmap_script_xids2sids_send (
TALLOC_CTX * mem_ctx , struct tevent_context * ev ,
struct id_map * * ids , size_t num_ids , const char * script )
{
struct tevent_req * req ;
struct idmap_script_xids2sids_state * state ;
size_t i ;
req = tevent_req_create ( mem_ctx , & state ,
struct idmap_script_xids2sids_state ) ;
if ( req = = NULL ) {
return NULL ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
state - > ids = ids ;
state - > num_ids = num_ids ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
if ( state - > num_ids = = 0 ) {
tevent_req_done ( req ) ;
return tevent_req_post ( req , ev ) ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
for ( i = 0 ; i < num_ids ; i + + ) {
struct tevent_req * subreq ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
subreq = idmap_script_xid2sid_send (
state , ev , ids [ i ] - > xid , script , i ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , idmap_script_xids2sids_done ,
req ) ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
return req ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
static void idmap_script_xids2sids_done ( struct tevent_req * subreq )
2014-12-24 04:33:34 +03:00
{
2016-02-17 16:01:47 +03:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct idmap_script_xids2sids_state * state = tevent_req_data (
req , struct idmap_script_xids2sids_state ) ;
size_t idx = 0 ;
enum id_mapping status = ID_UNKNOWN ;
struct dom_sid sid = { 0 } ;
int ret ;
ret = idmap_script_xid2sid_recv ( subreq , & idx , & status , & sid ) ;
TALLOC_FREE ( subreq ) ;
if ( tevent_req_error ( req , ret ) ) {
return ;
}
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
if ( idx > = state - > num_ids ) {
tevent_req_error ( req , EINVAL ) ;
return ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
state - > ids [ idx ] - > status = status ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
state - > ids [ idx ] - > sid = dom_sid_dup ( state - > ids , & sid ) ;
if ( tevent_req_nomem ( state - > ids [ idx ] - > sid , req ) ) {
return ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
state - > num_done + = 1 ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
if ( state - > num_done > = state - > num_ids ) {
tevent_req_done ( req ) ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
}
static int idmap_script_xids2sids_recv ( struct tevent_req * req )
{
return tevent_req_simple_recv_unix ( req ) ;
}
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
static int idmap_script_xids2sids ( struct id_map * * ids , size_t num_ids ,
const char * script )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct tevent_context * ev ;
struct tevent_req * req ;
int ret = ENOMEM ;
ev = samba_tevent_context_init ( frame ) ;
if ( ev = = NULL ) {
goto fail ;
}
req = idmap_script_xids2sids_send ( frame , ev , ids , num_ids , script ) ;
if ( req = = NULL ) {
goto fail ;
}
if ( ! tevent_req_poll ( req , ev ) ) {
ret = errno ;
goto fail ;
}
ret = idmap_script_xids2sids_recv ( req ) ;
fail :
TALLOC_FREE ( frame ) ;
2014-12-24 04:33:34 +03:00
return ret ;
}
static NTSTATUS idmap_script_unixids_to_sids ( struct idmap_domain * dom ,
2016-02-17 16:01:47 +03:00
struct id_map * * ids )
2014-12-24 04:33:34 +03:00
{
2016-02-17 16:01:47 +03:00
struct idmap_script_context * ctx = talloc_get_type_abort (
dom - > private_data , struct idmap_script_context ) ;
int ret ;
size_t i , num_ids , num_mapped ;
2014-12-24 04:33:34 +03:00
DEBUG ( 10 , ( " %s called ... \n " , __func__ ) ) ;
/* Init status to avoid surprise ... */
for ( i = 0 ; ids [ i ] ; i + + ) {
ids [ i ] - > status = ID_UNKNOWN ;
}
2016-02-17 16:01:47 +03:00
num_ids = i ;
ret = idmap_script_xids2sids ( ids , num_ids , ctx - > script ) ;
if ( ret ! = 0 ) {
DBG_DEBUG ( " idmap_script_xids2sids returned %s \n " ,
strerror ( ret ) ) ;
return map_nt_error_from_unix ( ret ) ;
}
num_mapped = 0 ;
2014-12-24 04:33:34 +03:00
for ( i = 0 ; ids [ i ] ; i + + ) {
2016-02-17 16:01:47 +03:00
if ( ids [ i ] - > status = = ID_MAPPED ) {
num_mapped + = 1 ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
}
if ( num_mapped = = 0 ) {
return NT_STATUS_NONE_MAPPED ;
}
if ( num_mapped < num_ids ) {
return STATUS_SOME_UNMAPPED ;
}
return NT_STATUS_OK ;
}
struct idmap_script_sid2xid_state {
const char * syscmd ;
size_t idx ;
uint8_t * out ;
} ;
static void idmap_script_sid2xid_done ( struct tevent_req * subreq ) ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
static struct tevent_req * idmap_script_sid2xid_send (
TALLOC_CTX * mem_ctx , struct tevent_context * ev ,
const struct dom_sid * sid , const char * script , size_t idx )
{
struct tevent_req * req , * subreq ;
struct idmap_script_sid2xid_state * state ;
char sidbuf [ DOM_SID_STR_BUFLEN ] ;
req = tevent_req_create ( mem_ctx , & state ,
struct idmap_script_sid2xid_state ) ;
if ( req = = NULL ) {
return NULL ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
state - > idx = idx ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
dom_sid_string_buf ( sid , sidbuf , sizeof ( sidbuf ) ) ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
state - > syscmd = talloc_asprintf ( state , " %s SIDTOID %s " ,
script , sidbuf ) ;
if ( tevent_req_nomem ( state - > syscmd , req ) ) {
return tevent_req_post ( req , ev ) ;
}
subreq = file_pload_send ( state , ev , state - > syscmd , 1024 ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , idmap_script_sid2xid_done , req ) ;
return req ;
}
static void idmap_script_sid2xid_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct idmap_script_sid2xid_state * state = tevent_req_data (
req , struct idmap_script_sid2xid_state ) ;
int ret ;
ret = file_pload_recv ( subreq , state , & state - > out ) ;
TALLOC_FREE ( subreq ) ;
if ( tevent_req_error ( req , ret ) ) {
return ;
}
tevent_req_done ( req ) ;
}
static int idmap_script_sid2xid_recv ( struct tevent_req * req ,
size_t * idx , enum id_mapping * status ,
struct unixid * xid )
{
struct idmap_script_sid2xid_state * state = tevent_req_data (
req , struct idmap_script_sid2xid_state ) ;
char * out = ( char * ) state - > out ;
size_t out_size = talloc_get_size ( out ) ;
unsigned long v ;
int err ;
if ( tevent_req_is_unix_error ( req , & err ) ) {
return err ;
}
if ( out_size = = 0 ) {
goto unmapped ;
}
if ( state - > out [ out_size - 1 ] ! = ' \0 ' ) {
goto unmapped ;
}
* idx = state - > idx ;
if ( sscanf ( out , " XID:%lu \n " , & v ) = = 1 ) {
* xid = ( struct unixid ) { . id = v , . type = ID_TYPE_BOTH } ;
} else if ( sscanf ( out , " UID:%lu \n " , & v ) = = 1 ) {
* xid = ( struct unixid ) { . id = v , . type = ID_TYPE_UID } ;
} else if ( sscanf ( out , " GID:%lu \n " , & v ) = = 1 ) {
* xid = ( struct unixid ) { . id = v , . type = ID_TYPE_GID } ;
} else {
goto unmapped ;
}
* status = ID_MAPPED ;
return 0 ;
unmapped :
* xid = ( struct unixid ) { . id = UINT32_MAX ,
. type = ID_TYPE_NOT_SPECIFIED } ;
* status = ID_UNMAPPED ;
return 0 ;
}
struct idmap_script_sids2xids_state {
struct id_map * * ids ;
size_t num_ids ;
size_t num_done ;
} ;
static void idmap_script_sids2xids_done ( struct tevent_req * subreq ) ;
static struct tevent_req * idmap_script_sids2xids_send (
TALLOC_CTX * mem_ctx , struct tevent_context * ev ,
struct id_map * * ids , size_t num_ids , const char * script )
{
struct tevent_req * req ;
struct idmap_script_sids2xids_state * state ;
size_t i ;
req = tevent_req_create ( mem_ctx , & state ,
struct idmap_script_sids2xids_state ) ;
if ( req = = NULL ) {
return NULL ;
}
state - > ids = ids ;
state - > num_ids = num_ids ;
if ( state - > num_ids = = 0 ) {
tevent_req_done ( req ) ;
return tevent_req_post ( req , ev ) ;
}
for ( i = 0 ; i < num_ids ; i + + ) {
struct tevent_req * subreq ;
subreq = idmap_script_sid2xid_send (
state , ev , ids [ i ] - > sid , script , i ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
tevent_req_set_callback ( subreq , idmap_script_sids2xids_done ,
req ) ;
}
return req ;
}
static void idmap_script_sids2xids_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct idmap_script_sids2xids_state * state = tevent_req_data (
req , struct idmap_script_sids2xids_state ) ;
size_t idx = 0 ;
enum id_mapping status = ID_UNKNOWN ;
struct unixid xid = { . id = UINT32_MAX ,
. type = ID_TYPE_NOT_SPECIFIED } ;
int ret ;
ret = idmap_script_sid2xid_recv ( subreq , & idx , & status , & xid ) ;
TALLOC_FREE ( subreq ) ;
if ( tevent_req_error ( req , ret ) ) {
return ;
}
if ( idx > = state - > num_ids ) {
tevent_req_error ( req , EINVAL ) ;
return ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
state - > ids [ idx ] - > status = status ;
state - > ids [ idx ] - > xid = xid ;
state - > num_done + = 1 ;
if ( state - > num_done > = state - > num_ids ) {
tevent_req_done ( req ) ;
}
}
static int idmap_script_sids2xids_recv ( struct tevent_req * req )
{
return tevent_req_simple_recv_unix ( req ) ;
}
static int idmap_script_sids2xids ( struct id_map * * ids , size_t num_ids ,
const char * script )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct tevent_context * ev ;
struct tevent_req * req ;
int ret = ENOMEM ;
ev = samba_tevent_context_init ( frame ) ;
if ( ev = = NULL ) {
goto fail ;
}
req = idmap_script_sids2xids_send ( frame , ev , ids , num_ids , script ) ;
if ( req = = NULL ) {
goto fail ;
}
if ( ! tevent_req_poll ( req , ev ) ) {
ret = errno ;
goto fail ;
}
ret = idmap_script_sids2xids_recv ( req ) ;
fail :
TALLOC_FREE ( frame ) ;
2014-12-24 04:33:34 +03:00
return ret ;
}
static NTSTATUS idmap_script_sids_to_unixids ( struct idmap_domain * dom ,
2016-02-17 16:01:47 +03:00
struct id_map * * ids )
2014-12-24 04:33:34 +03:00
{
2016-02-17 16:01:47 +03:00
struct idmap_script_context * ctx = talloc_get_type_abort (
dom - > private_data , struct idmap_script_context ) ;
int ret ;
size_t i , num_ids , num_mapped ;
2014-12-24 04:33:34 +03:00
DEBUG ( 10 , ( " %s called ... \n " , __func__ ) ) ;
/* Init status to avoid surprise ... */
for ( i = 0 ; ids [ i ] ; i + + ) {
ids [ i ] - > status = ID_UNKNOWN ;
}
2016-02-17 16:01:47 +03:00
num_ids = i ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
ret = idmap_script_sids2xids ( ids , num_ids , ctx - > script ) ;
if ( ret ! = 0 ) {
DBG_DEBUG ( " idmap_script_sids2xids returned %s \n " ,
strerror ( ret ) ) ;
return map_nt_error_from_unix ( ret ) ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
num_mapped = 0 ;
2014-12-24 04:33:34 +03:00
2016-02-17 16:01:47 +03:00
for ( i = 0 ; i < num_ids ; i + + ) {
struct id_map * map = ids [ i ] ;
if ( ( map - > status = = ID_MAPPED ) & &
! idmap_unix_id_is_in_range ( map - > xid . id , dom ) ) {
DBG_INFO ( " Script returned id (%u) out of range "
" (%u - %u). Filtered! \n " ,
map - > xid . id , dom - > low_id , dom - > high_id ) ;
map - > status = ID_UNMAPPED ;
2014-12-24 04:33:34 +03:00
}
2016-02-17 16:01:47 +03:00
if ( map - > status = = ID_MAPPED ) {
num_mapped + = 1 ;
2014-12-24 04:33:34 +03:00
}
}
2016-02-17 16:01:47 +03:00
if ( num_mapped = = 0 ) {
return NT_STATUS_NONE_MAPPED ;
}
if ( num_mapped < num_ids ) {
return STATUS_SOME_UNMAPPED ;
}
return NT_STATUS_OK ;
2014-12-24 04:33:34 +03:00
}
/*
* Initialise idmap_script database .
*/
static NTSTATUS idmap_script_db_init ( struct idmap_domain * dom )
{
NTSTATUS ret ;
struct idmap_script_context * ctx ;
const char * idmap_script = NULL ;
DEBUG ( 10 , ( " %s called ... \n " , __func__ ) ) ;
ctx = talloc_zero ( dom , struct idmap_script_context ) ;
if ( ! ctx ) {
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
ret = NT_STATUS_NO_MEMORY ;
goto failed ;
}
2017-01-07 17:55:41 +03:00
ctx - > script = idmap_config_const_string ( dom - > name , " script " , NULL ) ;
2014-12-24 04:33:34 +03:00
/* Do we even need to handle this? */
idmap_script = lp_parm_const_string ( - 1 , " idmap " , " script " , NULL ) ;
if ( idmap_script ! = NULL ) {
DEBUG ( 0 , ( " Warning: 'idmap:script' is deprecated. "
" Please use 'idmap config * : script' instead! \n " ) ) ;
}
if ( strequal ( dom - > name , " * " ) & & ctx - > script = = NULL ) {
/* fall back to idmap:script for backwards compatibility */
ctx - > script = idmap_script ;
}
if ( ctx - > script ) {
DEBUG ( 1 , ( " using idmap script '%s' \n " , ctx - > script ) ) ;
}
dom - > private_data = ctx ;
dom - > read_only = true ; /* We do not allocate!*/
return NT_STATUS_OK ;
failed :
talloc_free ( ctx ) ;
return ret ;
}
static struct idmap_methods db_methods = {
. init = idmap_script_db_init ,
. unixids_to_sids = idmap_script_unixids_to_sids ,
. sids_to_unixids = idmap_script_sids_to_unixids ,
} ;
2015-08-13 19:16:20 +03:00
static_decl_idmap ;
2017-04-20 22:24:43 +03:00
NTSTATUS idmap_script_init ( TALLOC_CTX * ctx )
2014-12-24 04:33:34 +03:00
{
return smb_register_idmap ( SMB_IDMAP_INTERFACE_VERSION , " script " , & db_methods ) ;
}