2020-05-07 15:09:32 +01:00
/*
* Unix SMB / CIFS implementation .
*
* Window Search Service
*
* Copyright ( c ) Noel Power
*
* 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"
# include "librpc/wsp/wsp_util.h"
# include "librpc/gen_ndr/wsp.h"
2023-08-29 15:01:03 +01:00
# include "librpc/gen_ndr/ndr_wsp.h"
# include "lib/util/strv_util.h"
# include "lib/util/strv.h"
# include "lib/util/util_str_hex.h"
# include "source3/param/param_proto.h"
# include "lib/util/dlinklist.h"
# define BUFFER_SIZE 1024000
struct guidtopropmap_holder
{
struct guidtopropmap * guidtopropmaploc ;
} ;
struct full_propset_info_list {
struct full_propset_info_list * prev , * next ;
struct full_propset_info info ;
} ;
struct guidtopropmap {
struct guidtopropmap * prev , * next ;
struct GUID guid ;
struct full_propset_info_list * propset ;
} ;
static struct guidtopropmap * find_guid_props (
struct guidtopropmap_holder * holder ,
const struct GUID * guid )
{
struct guidtopropmap * mapitem ;
for ( mapitem = holder - > guidtopropmaploc ; mapitem ; mapitem = mapitem - > next ) {
if ( GUID_equal ( guid , & mapitem - > guid ) ) {
return mapitem ;
}
}
return NULL ;
}
static bool getbool ( char * str )
{
char * cpy = talloc_strdup ( NULL , str ) ;
bool result ;
trim_string ( cpy , " " , " " ) ;
if ( strequal ( " TRUE " , cpy ) ) {
result = true ;
} else {
result = false ;
}
TALLOC_FREE ( cpy ) ;
return result ;
}
struct {
const char * typename ;
uint16_t type ;
} vtype_map [ ] = {
{ " GUID " , VT_CLSID } ,
{ " String " , VT_LPWSTR } ,
{ " BString " , VT_BSTR } ,
{ " Double " , VT_R8 } ,
{ " Buffer " , VT_BLOB_OBJECT } ,
{ " Byte " , VT_UI1 } ,
{ " UInt64 " , VT_UI8 } ,
{ " Int64 " , VT_I8 } ,
{ " UInt32 " , VT_UI4 } ,
{ " Int32 " , VT_I4 } ,
{ " UInt16 " , VT_UI2 } ,
{ " Int16 " , VT_I2 } ,
{ " DateTime " , VT_FILETIME } ,
{ " Boolean " , VT_BOOL }
} ;
static uint16_t getvtype ( char * str , bool isvec )
{
uint16_t result = UINT16_MAX ;
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( vtype_map ) ; i + + ) {
if ( strequal ( vtype_map [ i ] . typename , str ) ) {
result = vtype_map [ i ] . type ;
if ( isvec ) {
result | = VT_VECTOR ;
}
break ;
}
}
return result ;
}
static bool parse_csv_line ( TALLOC_CTX * ctx ,
char * * csvs , size_t num_values ,
struct guidtopropmap_holder * propmap_holder )
{
struct guidtopropmap * mapitem = NULL ;
struct full_propset_info_list * item = NULL ;
char * guid_str = NULL ;
struct GUID guid ;
bool ok ;
item = talloc_zero ( ctx ,
struct full_propset_info_list ) ;
if ( ! item ) {
return false ;
}
item - > info . in_inverted_index = false ;
item - > info . is_column = true ;
item - > info . can_col_be_indexed = true ;
if ( strlen ( csvs [ 1 ] ) ) {
guid_str = talloc_strdup ( ctx , csvs [ 1 ] ) ;
}
if ( ! guid_str ) {
DBG_ERR ( " out of memory \n " ) ;
return false ;
}
if ( ! trim_string ( guid_str , " { " , " } " ) ) {
return false ;
}
if ( strlen ( csvs [ 0 ] ) ) {
char * tmp = talloc_strdup ( item , csvs [ 0 ] ) ;
trim_string ( tmp , " " , " " ) ;
item - > info . name = tmp ;
}
if ( strlen ( csvs [ 2 ] ) ) {
item - > info . id = atoi ( csvs [ 2 ] ) ;
}
if ( strlen ( csvs [ 3 ] ) ) {
item - > info . in_inverted_index = getbool ( csvs [ 3 ] ) ;
}
if ( strlen ( csvs [ 4 ] ) ) {
item - > info . is_column = getbool ( csvs [ 4 ] ) ;
}
if ( strlen ( csvs [ 5 ] ) ) {
item - > info . can_col_be_indexed = getbool ( csvs [ 5 ] ) ;
}
if ( strlen ( csvs [ 6 ] ) ) {
bool isvec = false ;
uint16_t type ;
if ( strlen ( csvs [ 0 ] ) ) {
isvec = getbool ( csvs [ 8 ] ) ;
}
type = getvtype ( csvs [ 6 ] , isvec ) ;
if ( type = = UINT16_MAX ) {
DBG_ERR ( " failed to parse type \n " ) ;
return false ;
}
item - > info . vtype = type ;
}
ok = parse_guid_string ( guid_str , & guid ) ;
if ( ! ok ) {
return false ;
}
mapitem = find_guid_props ( propmap_holder , & guid ) ;
if ( ! mapitem ) {
mapitem = talloc_zero ( propmap_holder ,
struct guidtopropmap ) ;
if ( ! mapitem ) {
return false ;
}
mapitem - > guid = guid ;
DLIST_ADD_END ( propmap_holder - > guidtopropmaploc , mapitem ) ;
}
talloc_steal ( mapitem , item ) ;
DLIST_ADD_END ( mapitem - > propset , item ) ;
return true ;
}
static bool parse_properties_line ( TALLOC_CTX * ctx ,
const char * line ,
struct guidtopropmap_holder * propmap_holder )
{
int ret ;
int pos ;
char * strv = NULL ;
char * * csv_line = NULL ;
char * t = NULL ;
size_t len ;
ret = strv_split ( ctx ,
& strv ,
line ,
" , " ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " failed to split line \n " ) ;
return false ;
}
len = strv_count ( strv ) ;
if ( len < 9 ) {
DBG_WARNING ( " skipping line as it doesn't have "
" enough fields \n " ) ;
return true ;
}
csv_line = talloc_zero_array ( ctx ,
char * ,
len ) ;
if ( ! csv_line ) {
DBG_ERR ( " out of memory \n " ) ;
return false ;
}
for ( pos = 0 ; pos < talloc_array_length ( csv_line ) ; pos + + ) {
t = strv_next ( strv , t ) ;
/* the scraped property file can have a non ascii char */
if ( strlen ( t ) = = 1 & & * t = = 0xa0 ) {
csv_line [ pos ] = talloc_strdup ( csv_line ,
" " ) ;
} else {
csv_line [ pos ] = talloc_strdup ( csv_line ,
t ) ;
}
trim_string ( csv_line [ pos ] , " " , " " ) ;
}
if ( ! parse_csv_line ( csv_line , csv_line , len , propmap_holder ) ) {
DBG_ERR ( " failed to parse line \n " ) ;
TALLOC_FREE ( csv_line ) ;
return false ;
}
TALLOC_FREE ( csv_line ) ;
return true ;
}
static bool parse_properties_csvfile ( TALLOC_CTX * ctx ,
struct guidtopropmap_holder * propmap_holder ,
const char * filename )
{
char * * lines = NULL ;
int numlines ;
int i ;
if ( filename = = NULL | | strlen ( filename ) = = 0 ) {
return false ;
}
lines = file_lines_load ( filename ,
& numlines ,
BUFFER_SIZE ,
ctx ) ;
if ( ! lines ) {
DBG_ERR ( " Failed to load %s \n " , filename ) ;
return false ;
}
DBG_ERR ( " parsed %d lines \n " , numlines ) ;
for ( i = 0 ; i < numlines ; i + + ) {
TALLOC_CTX * line_ctx = talloc_init ( " line context " ) ;
if ( ! line_ctx ) {
DBG_ERR ( " out of memory \n " ) ;
return false ;
}
trim_string ( lines [ i ] , " " , " " ) ;
if ( lines [ i ] [ 0 ] = = ' # ' ) {
DBG_WARNING ( " skipping comment at line %d. \n ) " , i ) ;
TALLOC_FREE ( line_ctx ) ;
continue ;
}
if ( ! parse_properties_line ( line_ctx ,
lines [ i ] ,
propmap_holder ) ) {
DBG_ERR ( " Failed to parse line %d \n " , i ) ;
}
TALLOC_FREE ( line_ctx ) ;
}
return true ;
}
static bool populate_map ( struct guidtopropmap_holder * propmap_holder )
{
const char * path = NULL ;
path = lp_wsp_property_file ( ) ;
/* first populate the map from property file */
if ( path ) {
parse_properties_csvfile ( propmap_holder , propmap_holder , path ) ;
}
return true ;
}
static struct guidtopropmap_holder * propmap ( void )
{
static struct guidtopropmap_holder * holder = NULL ;
if ( ! holder ) {
holder = talloc_zero ( NULL , struct guidtopropmap_holder ) ;
if ( holder ) {
populate_map ( holder ) ;
}
}
return holder ;
}
2020-05-07 15:09:32 +01:00
const struct full_propset_info * get_propset_info_with_guid (
const char * prop_name ,
struct GUID * propset_guid )
{
2023-08-29 15:01:03 +01:00
const struct full_propset_info * result = NULL ;
struct guidtopropmap_holder * holder = NULL ;
struct guidtopropmap * mapitem = NULL ;
2020-05-07 15:09:32 +01:00
size_t i ;
const struct full_guid_propset * guid_propset = NULL ;
2023-08-29 15:01:03 +01:00
/* search builtin props first */
2020-05-07 15:09:32 +01:00
for ( i = 0 ; full_propertyset [ i ] . prop_info ! = NULL ; i + + ) {
const struct full_propset_info * item = NULL ;
guid_propset = & full_propertyset [ i ] ;
item = guid_propset - > prop_info ;
while ( item - > id ) {
if ( strequal ( prop_name , item - > name ) ) {
* propset_guid = guid_propset - > guid ;
result = item ;
break ;
}
item + + ;
}
if ( result ) {
break ;
}
}
2023-08-29 15:01:03 +01:00
if ( result ) {
return result ;
}
/* if we didn't find a match in builtin props try the extra props */
holder = propmap ( ) ;
for ( mapitem = holder - > guidtopropmaploc ; mapitem ;
mapitem = mapitem - > next ) {
struct full_propset_info_list * propitem ;
for ( propitem = mapitem - > propset ; propitem ;
propitem = propitem - > next ) {
if ( strequal ( prop_name , propitem - > info . name ) ) {
* propset_guid = mapitem - > guid ;
result = & propitem - > info ;
break ;
}
}
}
2020-05-07 15:09:32 +01:00
return result ;
}
const struct full_propset_info * get_prop_info ( const char * prop_name )
{
const struct full_propset_info * result = NULL ;
struct GUID guid ;
result = get_propset_info_with_guid ( prop_name , & guid ) ;
return result ;
}
char * prop_from_fullprop ( TALLOC_CTX * ctx , struct wsp_cfullpropspec * fullprop )
{
size_t i ;
char * result = NULL ;
const struct full_propset_info * item = NULL ;
2023-08-29 15:01:03 +01:00
const struct full_propset_info_list * prop_item = NULL ;
2020-05-07 15:09:32 +01:00
bool search_by_id = ( fullprop - > ulkind = = PRSPEC_PROPID ) ;
2023-08-29 15:01:03 +01:00
struct guidtopropmap_holder * holder = NULL ;
struct guidtopropmap * mapitem = NULL ;
2020-05-07 15:09:32 +01:00
2023-08-29 15:01:03 +01:00
/* check builtin properties */
2020-05-07 15:09:32 +01:00
for ( i = 0 ; full_propertyset [ i ] . prop_info ! = NULL ; i + + ) {
/* find propset */
if ( GUID_equal ( & fullprop - > guidpropset ,
& full_propertyset [ i ] . guid ) ) {
item = full_propertyset [ i ] . prop_info ;
break ;
}
}
if ( item ) {
while ( item - > id ) {
if ( search_by_id ) {
if ( fullprop - > name_or_id . prspec = = item - > id ) {
result = talloc_strdup ( ctx , item - > name ) ;
break ;
}
} else if ( strcmp ( item - > name ,
fullprop - > name_or_id . propname . vstring )
= = 0 ) {
result = talloc_strdup ( ctx , item - > name ) ;
break ;
}
item + + ;
}
}
2023-08-29 15:01:03 +01:00
/* not found, search the extra props */
if ( ! result ) {
holder = propmap ( ) ;
for ( mapitem = holder - > guidtopropmaploc ; mapitem ;
mapitem = mapitem - > next ) {
if ( GUID_equal ( & fullprop - > guidpropset ,
& mapitem - > guid ) ) {
prop_item = mapitem - > propset ;
break ;
}
}
for ( ; prop_item ; prop_item = prop_item - > next ) {
if ( search_by_id ) {
if ( fullprop - > name_or_id . prspec = =
prop_item - > info . id ) {
result = talloc_strdup ( ctx ,
prop_item - > info . name ) ;
break ;
}
} else if ( strcmp ( prop_item - > info . name ,
fullprop - > name_or_id . propname . vstring ) = = 0 ) {
result = talloc_strdup ( ctx ,
prop_item - > info . name ) ;
break ;
}
}
}
2020-05-07 15:09:32 +01:00
if ( ! result ) {
result = GUID_string ( ctx , & fullprop - > guidpropset ) ;
if ( search_by_id ) {
result = talloc_asprintf ( result , " %s/%d " , result ,
fullprop - > name_or_id . prspec ) ;
} else {
result = talloc_asprintf ( result , " %s/%s " , result ,
fullprop - > name_or_id . propname . vstring ) ;
}
}
return result ;
}
2022-10-17 19:14:35 +01:00
const char * genmeth_to_string ( uint32_t genmethod )
{
const char * result = NULL ;
switch ( genmethod ) {
case 0 :
result = " equals " ;
break ;
case 1 :
result = " starts with " ;
break ;
case 2 :
result = " matches inflection " ;
break ;
default :
result = NULL ;
break ;
}
return result ;
}
bool is_operator ( struct wsp_crestriction * restriction ) {
bool result ;
switch ( restriction - > ultype ) {
case RTAND :
case RTOR :
case RTNOT :
result = true ;
break ;
default :
result = false ;
break ;
}
return result ;
}
const char * op_as_string ( struct wsp_crestriction * restriction )
{
const char * op = NULL ;
if ( is_operator ( restriction ) ) {
switch ( restriction - > ultype ) {
case RTAND :
op = " && " ;
break ;
case RTOR :
op = " || " ;
break ;
case RTNOT :
op = " ! " ;
break ;
}
} else if ( restriction - > ultype = = RTPROPERTY ) {
struct wsp_cpropertyrestriction * prop_restr =
& restriction - > restriction . cpropertyrestriction ;
switch ( prop_restr - > relop & 0 XF ) {
case PREQ :
op = " = " ;
break ;
case PRNE :
op = " != " ;
break ;
case PRGE :
op = " >= " ;
break ;
case PRLE :
op = " <= " ;
break ;
case PRLT :
op = " < " ;
break ;
case PRGT :
op = " > " ;
break ;
default :
break ;
}
} else if ( restriction - > ultype = = RTCONTENT ) {
struct wsp_ccontentrestriction * content = NULL ;
content = & restriction - > restriction . ccontentrestriction ;
op = genmeth_to_string ( content - > ulgeneratemethod ) ;
} else if ( restriction - > ultype = = RTNATLANGUAGE ) {
op = " = " ;
}
return op ;
}
2020-05-07 15:09:32 +01:00
struct wsp_cfullpropspec * get_full_prop ( struct wsp_crestriction * restriction )
{
struct wsp_cfullpropspec * result ;
switch ( restriction - > ultype ) {
case RTPROPERTY :
result = & restriction - > restriction . cpropertyrestriction . property ;
break ;
case RTCONTENT :
result = & restriction - > restriction . ccontentrestriction . property ;
break ;
case RTNATLANGUAGE :
result = & restriction - > restriction . cnatlanguagerestriction . property ;
break ;
default :
result = NULL ;
break ;
}
return result ;
}
2022-10-17 14:20:49 +01:00
2022-10-17 19:14:35 +01:00
const char * variant_as_string ( TALLOC_CTX * ctx ,
struct wsp_cbasestoragevariant * value , bool quote )
{
const char * result = NULL ;
switch ( value - > vtype ) {
case VT_UI1 :
result = talloc_asprintf ( ctx , " %u " ,
value - > vvalue . vt_ui1 ) ;
break ;
case VT_INT :
case VT_I4 :
result = talloc_asprintf ( ctx , " %d " ,
value - > vvalue . vt_i4 ) ;
break ;
case VT_ERROR :
case VT_UINT :
case VT_UI4 :
result = talloc_asprintf ( ctx , " %u " ,
value - > vvalue . vt_ui4 ) ;
break ;
case VT_UI2 :
case VT_I2 :
result = talloc_asprintf ( ctx , " %u " ,
value - > vvalue . vt_ui2 ) ;
break ;
case VT_BOOL :
result = talloc_asprintf ( ctx , " %s " ,
value - > vvalue . vt_ui2 = = 0xFFFF ?
" true " : " false " ) ;
break ;
case VT_DATE :
case VT_FILETIME : {
NTTIME filetime = value - > vvalue . vt_ui8 ;
time_t unixtime ;
struct tm * tm = NULL ;
char datestring [ 256 ] ;
unixtime = nt_time_to_unix ( filetime ) ;
tm = gmtime ( & unixtime ) ;
strftime ( datestring , sizeof ( datestring ) , " %FT%TZ " , tm ) ;
result = talloc_strdup ( ctx , datestring ) ;
break ;
}
case VT_R4 : {
float f ;
if ( sizeof ( f ) ! = sizeof ( value - > vvalue . vt_ui4 ) ) {
DBG_ERR ( " can't convert float \n " ) ;
break ;
}
memcpy ( ( void * ) & f ,
( void * ) & value - > vvalue . vt_ui4 ,
sizeof ( value - > vvalue . vt_ui4 ) ) ;
result = talloc_asprintf ( ctx , " %f " ,
f ) ;
break ;
}
case VT_R8 : {
/* should this really be unsigned ? */
double dval ;
if ( sizeof ( dval ) ! = sizeof ( value - > vvalue . vt_i8 ) ) {
DBG_ERR ( " can't convert double \n " ) ;
break ;
}
memcpy ( ( void * ) & dval ,
( void * ) & value - > vvalue . vt_i8 ,
sizeof ( dval ) ) ;
result = talloc_asprintf ( ctx , " %f " ,
dval ) ;
break ;
}
case VT_I8 : {
result = talloc_asprintf ( ctx , " % " PRIi64 ,
value - > vvalue . vt_i8 ) ;
break ;
}
case VT_UI8 : {
result = talloc_asprintf ( ctx , " % " PRIu64 ,
value - > vvalue . vt_ui8 ) ;
break ;
}
case VT_LPWSTR :
result = talloc_asprintf ( ctx , " %s%s%s " ,
quote ? " \' " : " " ,
value - > vvalue . vt_lpwstr . value ,
quote ? " \' " : " " ) ;
break ;
case VT_LPWSTR | VT_VECTOR : {
int num_elems =
value - > vvalue . vt_lpwstr_v . vvector_elements ;
int i ;
for ( i = 0 ; i < num_elems ; i + + ) {
struct vt_lpwstr_vec * vec ;
const char * val ;
vec = & value - > vvalue . vt_lpwstr_v ;
val = vec - > vvector_data [ i ] . value ;
result =
talloc_asprintf ( ctx ,
" %s%s%s%s%s " ,
result ? result : " " ,
i ? " , " : " " ,
quote ? " \' " : " " ,
val ,
quote ? " \' " : " " ) ;
}
break ;
}
default :
DBG_INFO ( " can't represent unsupported vtype 0x%x as string \n " ,
value - > vtype ) ;
break ;
}
return result ;
}
2022-10-17 14:20:49 +01:00
void set_variant_lpwstr ( TALLOC_CTX * ctx ,
struct wsp_cbasestoragevariant * vvalue ,
const char * string_val )
{
vvalue - > vtype = VT_LPWSTR ;
vvalue - > vvalue . vt_lpwstr . value = talloc_strdup ( ctx , string_val ) ;
}
static void fill_string_vec ( TALLOC_CTX * ctx ,
struct wsp_cbasestoragevariant * variant ,
const char * * strings , uint16_t elems )
{
int i ;
variant - > vvalue . vt_lpwstr_v . vvector_elements = elems ;
variant - > vvalue . vt_lpwstr_v . vvector_data = talloc_zero_array ( ctx ,
struct vt_lpwstr ,
elems ) ;
for ( i = 0 ; i < elems ; i + + ) {
variant - > vvalue . vt_lpwstr_v . vvector_data [ i ] . value = talloc_strdup ( ctx , strings [ i ] ) ;
}
}
void set_variant_lpwstr_vector ( TALLOC_CTX * ctx ,
struct wsp_cbasestoragevariant * variant ,
const char * * string_vals , uint32_t elems )
{
variant - > vtype = VT_LPWSTR | VT_VECTOR ;
fill_string_vec ( ctx , variant , string_vals , elems ) ;
}