From: Damjan Jovanovic Subject: [PATCH 4/6] adsldp: implement the ADS_SEARCHPREF_CHASE_REFERRALS search preference Message-Id: Date: Sun, 7 Feb 2021 17:55:28 +0200 Signed-off-by: Damjan Jovanovic --- dlls/adsldp/adsldp.c | 46 +++++++++++++++++++++++++++++++++++++++++++ dlls/wldap32/option.c | 14 ++++++++++++- include/iads.idl | 7 +++++++ include/winldap.h | 3 +++ 4 files changed, 69 insertions(+), 1 deletion(-) diff --git a/dlls/adsldp/adsldp.c b/dlls/adsldp/adsldp.c index 905018fe59a..1dacb0d6624 100644 --- a/dlls/adsldp/adsldp.c +++ b/dlls/adsldp/adsldp.c @@ -1319,6 +1319,52 @@ static HRESULT WINAPI search_SetSearchPreference(IDirectorySearch *iface, PADS_S prefs[i].dwStatus = ADS_STATUS_S_OK; break; + case ADS_SEARCHPREF_CHASE_REFERRALS: + { + ULONG referral = (ULONG)LDAP_OPT_OFF; + + if (prefs[i].vValue.dwType != ADSTYPE_INTEGER) + { + FIXME("ADS_SEARCHPREF_CHASE_REFERRALS: unsupported dwType %d\n", prefs[i].vValue.dwType); + prefs[i].dwStatus = ADS_STATUS_INVALID_SEARCHPREFVALUE; + break; + } + + TRACE("CHASE_REFERRALS: %d\n", prefs[i].vValue.u.Integer); + switch (prefs[i].vValue.u.Integer) + { + case ADS_CHASE_REFERRALS_NEVER: + referral = (ULONG)LDAP_OPT_OFF; + break; + case ADS_CHASE_REFERRALS_SUBORDINATE: + referral = LDAP_CHASE_SUBORDINATE_REFERRALS; + break; + case ADS_CHASE_REFERRALS_EXTERNAL: + referral = LDAP_CHASE_EXTERNAL_REFERRALS; + break; + case ADS_CHASE_REFERRALS_ALWAYS: + referral = (ULONG)LDAP_OPT_ON; + break; + default: + ERR("unknown/unsupported referral 0x%x\n", prefs[i].vValue.u.Integer); + prefs[i].dwStatus = ADS_STATUS_INVALID_SEARCHPREFVALUE; + hr = S_ADS_ERRORSOCCURRED; + } + if (SUCCEEDED(hr)) + { + err = ldap_set_optionW(ldap->ld, LDAP_OPT_REFERRALS, &referral); + if (err != LDAP_SUCCESS) + { + TRACE("ldap_set_option error %#x\n", err); + prefs[i].dwStatus = ADS_STATUS_INVALID_SEARCHPREF; + hr = S_ADS_ERRORSOCCURRED; + } + else + prefs[i].dwStatus = ADS_STATUS_S_OK; + } + break; + } + default: FIXME("pref %d, type %u: stub\n", prefs[i].dwSearchPref, prefs[i].vValue.dwType); prefs[i].dwStatus = ADS_STATUS_INVALID_SEARCHPREF; diff --git a/dlls/wldap32/option.c b/dlls/wldap32/option.c index cf948bf037c..013c121255e 100644 --- a/dlls/wldap32/option.c +++ b/dlls/wldap32/option.c @@ -497,11 +497,23 @@ ULONG CDECL ldap_set_optionW( WLDAP32_LDAP *ld, int option, void *value ) controlarrayfreeU( ctrlsU ); return ret; } + case WLDAP32_LDAP_OPT_REFERRALS: + { + ULONG win32_referral; + void *openldap_referral; + win32_referral = *(ULONG*)value; + openldap_referral = LDAP_OPT_ON; + if (win32_referral == (ULONG)LDAP_OPT_OFF) + openldap_referral = LDAP_OPT_OFF; + else + FIXME("upgrading referral value 0x%x to LDAP_OPT_ON (OpenLDAP lacks sufficient granularity)\n", win32_referral); + return map_error( ldap_set_option( ld->ld, option, openldap_referral )); + break; + } case WLDAP32_LDAP_OPT_DEREF: case WLDAP32_LDAP_OPT_DESC: case WLDAP32_LDAP_OPT_ERROR_NUMBER: case WLDAP32_LDAP_OPT_PROTOCOL_VERSION: - case WLDAP32_LDAP_OPT_REFERRALS: case WLDAP32_LDAP_OPT_SIZELIMIT: case WLDAP32_LDAP_OPT_TIMELIMIT: return map_error( ldap_set_option( ld->ld, option, value )); diff --git a/include/iads.idl b/include/iads.idl index 6dfafce3a87..b443633e39d 100644 --- a/include/iads.idl +++ b/include/iads.idl @@ -152,6 +152,13 @@ typedef enum ADS_PASSWORD_ENCODE_CLEAR } ADS_PASSWORD_ENCODING_ENUM; +typedef enum { + ADS_CHASE_REFERRALS_NEVER = 0, + ADS_CHASE_REFERRALS_SUBORDINATE = 0x20, + ADS_CHASE_REFERRALS_EXTERNAL = 0x40, + ADS_CHASE_REFERRALS_ALWAYS = ADS_CHASE_REFERRALS_SUBORDINATE | ADS_CHASE_REFERRALS_EXTERNAL +} ADS_CHASE_REFERRALS_ENUM; + typedef struct _ADS_CASEIGNORE_LIST { struct _ADS_CASEIGNORE_LIST *Next; diff --git a/include/winldap.h b/include/winldap.h index 2d41eaa3fa2..5295f683ee1 100644 --- a/include/winldap.h +++ b/include/winldap.h @@ -164,6 +164,9 @@ typedef struct berelement #define LDAP_OPT_ON ((void *)1) #define LDAP_OPT_OFF ((void *)0) +#define LDAP_CHASE_SUBORDINATE_REFERRALS 0x00000020 +#define LDAP_CHASE_EXTERNAL_REFERRALS 0x00000040 + #define LDAP_VERSION1 1 #define LDAP_VERSION2 2 #define LDAP_VERSION3 3