| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711 |
- /** BEGIN COPYRIGHT BLOCK
- * 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; version 2 of the License.
- *
- * 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., 59 Temple
- * Place, Suite 330, Boston, MA 02111-1307 USA.
- *
- * In addition, as a special exception, Red Hat, Inc. gives You the additional
- * right to link the code of this Program with code not covered under the GNU
- * General Public License ("Non-GPL Code") and to distribute linked combinations
- * including the two, subject to the limitations in this paragraph. Non-GPL Code
- * permitted under this exception must only link to the code of this Program
- * through those well defined interfaces identified in the file named EXCEPTION
- * found in the source code files (the "Approved Interfaces"). The files of
- * Non-GPL Code may instantiate templates or use macros or inline functions from
- * the Approved Interfaces without causing the resulting work to be covered by
- * the GNU General Public License. Only Red Hat, Inc. may make changes or
- * additions to the list of Approved Interfaces. You must obey the GNU General
- * Public License in all respects for all of the Program code and other code used
- * in conjunction with the Program except the Non-GPL Code covered by this
- * exception. If you modify this file, you may extend this exception to your
- * version of the file, but you are not obligated to do so. If you do not wish to
- * provide this exception without modification, you must delete this exception
- * statement from your version and license this file solely under the GPL without
- * exception.
- *
- * Authors:
- * Pete Rowley <[email protected]>
- * Nathan Kinder <[email protected]>
- *
- * Copyright (C) 2010 Red Hat, Inc.
- * All rights reserved.
- * END COPYRIGHT BLOCK
- **/
- /* The memberof plugin updates the memberof attribute of entries
- * based on modifications performed on groupofuniquenames entries
- *
- * In addition the plugin provides a DS task that may be started
- * administrative clients and that creates the initial memberof
- * list for imported entries and/or fixes the memberof list of
- * existing entries that have inconsistent state (for example,
- * if the memberof attribute was incorrectly edited directly)
- *
- * To start the memberof task add an entry like:
- *
- * dn: cn=mytask, cn=memberof task, cn=tasks, cn=config
- * objectClass: top
- * objectClass: extensibleObject
- * cn: mytask
- * basedn: dc=example, dc=com
- * filter: (uid=test4)
- *
- * where "basedn" is required and refers to the top most node to perform the
- * task on, and where "filter" is an optional attribute that provides a filter
- * describing the entries to be worked on
- */
- #ifdef HAVE_CONFIG_H
- # include <config.h>
- #endif
- #include "slapi-plugin.h"
- #include "string.h"
- #include "nspr.h"
- #include "memberof.h"
- static Slapi_PluginDesc pdesc = { "memberof", VENDOR,
- DS_PACKAGE_VERSION, "memberof plugin" };
- static void* _PluginID = NULL;
- static Slapi_DN* _ConfigAreaDN = NULL;
- static Slapi_DN* _pluginDN = NULL;
- static PRMonitor *memberof_operation_lock = 0;
- MemberOfConfig *qsortConfig = 0;
- static int g_plugin_started = 0;
- static int usetxn = 0;
- static int premodfn = 0;
- typedef struct _memberofstringll
- {
- const char *dn;
- void *next;
- } memberofstringll;
- typedef struct _memberof_get_groups_data
- {
- MemberOfConfig *config;
- Slapi_Value *memberdn_val;
- Slapi_ValueSet **groupvals;
- Slapi_ValueSet **group_norm_vals;
- } memberof_get_groups_data;
- /*** function prototypes ***/
- /* exported functions */
- int memberof_postop_init(Slapi_PBlock *pb );
- static int memberof_internal_postop_init(Slapi_PBlock *pb);
- static int memberof_preop_init(Slapi_PBlock *pb);
- /* plugin callbacks */
- static int memberof_postop_del(Slapi_PBlock *pb );
- static int memberof_postop_modrdn(Slapi_PBlock *pb );
- static int memberof_postop_modify(Slapi_PBlock *pb );
- static int memberof_postop_add(Slapi_PBlock *pb );
- static int memberof_postop_start(Slapi_PBlock *pb);
- static int memberof_postop_close(Slapi_PBlock *pb);
- /* supporting cast */
- static int memberof_oktodo(Slapi_PBlock *pb);
- static Slapi_DN *memberof_getsdn(Slapi_PBlock *pb);
- static int memberof_modop_one(Slapi_PBlock *pb, MemberOfConfig *config, int mod_op,
- Slapi_DN *op_this_sdn, Slapi_DN *op_to_sdn);
- static int memberof_modop_one_r(Slapi_PBlock *pb, MemberOfConfig *config, int mod_op,
- Slapi_DN *group_sdn, Slapi_DN *op_this_sdn, Slapi_DN *op_to_sdn,
- memberofstringll *stack);
- static int memberof_add_one(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *addthis_sdn, Slapi_DN *addto_sdn);
- static int memberof_del_one(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *delthis_sdn, Slapi_DN *delfrom_sdn);
- static int memberof_mod_smod_list(Slapi_PBlock *pb, MemberOfConfig *config,
- int mod, Slapi_DN *group_sdn, Slapi_Mod *smod);
- static int memberof_add_smod_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn, Slapi_Mod *smod);
- static int memberof_del_smod_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn, Slapi_Mod *smod);
- static int memberof_mod_attr_list(Slapi_PBlock *pb, MemberOfConfig *config, int mod,
- Slapi_DN *group_sdn, Slapi_Attr *attr);
- static int memberof_mod_attr_list_r(Slapi_PBlock *pb, MemberOfConfig *config,
- int mod, Slapi_DN *group_sdn, Slapi_DN *op_this_sdn, Slapi_Attr *attr,
- memberofstringll *stack);
- static int memberof_add_attr_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn, Slapi_Attr *attr);
- static int memberof_del_attr_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn, Slapi_Attr *attr);
- static int memberof_moddn_attr_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *pre_sdn, Slapi_DN *post_sdn, Slapi_Attr *attr);
- static int memberof_replace_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn);
- static void memberof_set_plugin_id(void * plugin_id);
- static int memberof_compare(MemberOfConfig *config, const void *a, const void *b);
- static int memberof_qsort_compare(const void *a, const void *b);
- static void memberof_load_array(Slapi_Value **array, Slapi_Attr *attr);
- static void memberof_del_dn_from_groups(Slapi_PBlock *pb, MemberOfConfig *config, Slapi_DN *sdn);
- static int memberof_call_foreach_dn(Slapi_PBlock *pb, Slapi_DN *sdn,
- char **types, plugin_search_entry_callback callback, void *callback_data);
- static int memberof_is_direct_member(MemberOfConfig *config, Slapi_Value *groupdn,
- Slapi_Value *memberdn);
- static int memberof_is_grouping_attr(char *type, MemberOfConfig *config);
- static Slapi_ValueSet *memberof_get_groups(MemberOfConfig *config, Slapi_DN *member_sdn);
- static int memberof_get_groups_r(MemberOfConfig *config, Slapi_DN *member_sdn,
- memberof_get_groups_data *data);
- static int memberof_get_groups_callback(Slapi_Entry *e, void *callback_data);
- static int memberof_test_membership(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn);
- static int memberof_test_membership_callback(Slapi_Entry *e, void *callback_data);
- static int memberof_del_dn_type_callback(Slapi_Entry *e, void *callback_data);
- static int memberof_replace_dn_type_callback(Slapi_Entry *e, void *callback_data);
- static void memberof_replace_dn_from_groups(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *pre_sdn, Slapi_DN *post_sdn);
- static int memberof_modop_one_replace_r(Slapi_PBlock *pb, MemberOfConfig *config,
- int mod_op, Slapi_DN *group_sdn, Slapi_DN *op_this_sdn,
- Slapi_DN *replace_with_sdn, Slapi_DN *op_to_sdn, memberofstringll *stack);
- static int memberof_task_add(Slapi_PBlock *pb, Slapi_Entry *e,
- Slapi_Entry *eAfter, int *returncode, char *returntext,
- void *arg);
- static void memberof_task_destructor(Slapi_Task *task);
- static const char *fetch_attr(Slapi_Entry *e, const char *attrname,
- const char *default_val);
- static void memberof_fixup_task_thread(void *arg);
- static int memberof_fix_memberof(MemberOfConfig *config, char *dn, char *filter_str);
- static int memberof_fix_memberof_callback(Slapi_Entry *e, void *callback_data);
- /*** implementation ***/
- /*** exported functions ***/
- /*
- * memberof_postop_init()
- *
- * Register plugin call backs
- *
- */
- int
- memberof_postop_init(Slapi_PBlock *pb)
- {
- int ret = 0;
- char *memberof_plugin_identity = 0;
- Slapi_Entry *plugin_entry = NULL;
- char *plugin_type = NULL;
- char *preop_plugin_type = NULL;
- int delfn = SLAPI_PLUGIN_POST_DELETE_FN;
- int mdnfn = SLAPI_PLUGIN_POST_MODRDN_FN;
- int modfn = SLAPI_PLUGIN_POST_MODIFY_FN;
- int addfn = SLAPI_PLUGIN_POST_ADD_FN;
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "--> memberof_postop_init\n" );
- /* get args */
- if ((slapi_pblock_get(pb, SLAPI_PLUGIN_CONFIG_ENTRY, &plugin_entry) == 0) &&
- plugin_entry &&
- (plugin_type = slapi_entry_attr_get_charptr(plugin_entry, "nsslapd-plugintype")) &&
- plugin_type && strstr(plugin_type, "betxn")) {
- usetxn = 1;
- delfn = SLAPI_PLUGIN_BE_TXN_POST_DELETE_FN;
- mdnfn = SLAPI_PLUGIN_BE_TXN_POST_MODRDN_FN;
- modfn = SLAPI_PLUGIN_BE_TXN_POST_MODIFY_FN;
- addfn = SLAPI_PLUGIN_BE_TXN_POST_ADD_FN;
- }
- slapi_ch_free_string(&plugin_type);
- if(usetxn){
- preop_plugin_type = "betxnpreoperation";
- premodfn = SLAPI_PLUGIN_BE_TXN_PRE_MODIFY_FN;
- } else {
- preop_plugin_type = "preoperation";
- premodfn = SLAPI_PLUGIN_PRE_MODIFY_FN;
- }
- if(plugin_entry){
- memberof_set_plugin_area(slapi_entry_get_sdn(plugin_entry));
- }
- /*
- * Get plugin identity and stored it for later use
- * Used for internal operations
- */
- slapi_pblock_get (pb, SLAPI_PLUGIN_IDENTITY, &memberof_plugin_identity);
- PR_ASSERT (memberof_plugin_identity);
- memberof_set_plugin_id(memberof_plugin_identity);
- ret = ( slapi_pblock_set( pb, SLAPI_PLUGIN_VERSION, SLAPI_PLUGIN_VERSION_01 ) != 0 ||
- slapi_pblock_set( pb, SLAPI_PLUGIN_DESCRIPTION, (void *)&pdesc ) != 0 ||
- slapi_pblock_set( pb, delfn, (void *) memberof_postop_del ) != 0 ||
- slapi_pblock_set( pb, mdnfn, (void *) memberof_postop_modrdn ) != 0 ||
- slapi_pblock_set( pb, modfn, (void *) memberof_postop_modify ) != 0 ||
- slapi_pblock_set( pb, addfn, (void *) memberof_postop_add ) != 0 ||
- slapi_pblock_set( pb, SLAPI_PLUGIN_START_FN, (void *) memberof_postop_start ) != 0 ||
- slapi_pblock_set( pb, SLAPI_PLUGIN_CLOSE_FN, (void *) memberof_postop_close ) != 0 );
- if (!ret && !usetxn &&
- slapi_register_plugin("internalpostoperation", /* op type */
- 1, /* Enabled */
- "memberof_postop_init", /* this function desc */
- memberof_internal_postop_init, /* init func */
- MEMBEROF_INT_PREOP_DESC, /* plugin desc */
- NULL, /* ? */
- memberof_plugin_identity /* access control */))
- {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_postop_init failed\n" );
- ret = -1;
- }
- else if (ret)
- {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_postop_init failed\n" );
- ret = -1;
- }
- /*
- * Setup the preop plugin for shared config updates
- */
- if (!ret && !usetxn &&
- slapi_register_plugin(preop_plugin_type, /* op type */
- 1, /* Enabled */
- "memberof_preop_init", /* this function desc */
- memberof_preop_init, /* init func */
- MEMBEROF_PREOP_DESC, /* plugin desc */
- NULL, /* ? */
- memberof_plugin_identity /* access control */))
- {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_preop_init failed\n" );
- ret = -1;
- }
- else if (ret)
- {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_preop_init failed\n");
- ret = -1;
- }
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "<-- memberof_postop_init\n" );
- return ret;
- }
- static int
- memberof_preop_init(Slapi_PBlock *pb)
- {
- int status = 0;
- if (slapi_pblock_set(pb, SLAPI_PLUGIN_VERSION, SLAPI_PLUGIN_VERSION_01) != 0 ||
- slapi_pblock_set(pb, SLAPI_PLUGIN_DESCRIPTION, (void *) &pdesc) != 0 ||
- slapi_pblock_set(pb, premodfn, (void *)memberof_shared_config_validate) != 0)
- {
- slapi_log_error(SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_internal_postop_init: failed to register plugin\n");
- status = -1;
- }
- return status;
- }
- static int
- memberof_internal_postop_init(Slapi_PBlock *pb)
- {
- int status = 0;
- if (slapi_pblock_set(pb, SLAPI_PLUGIN_VERSION,
- SLAPI_PLUGIN_VERSION_01) != 0 ||
- slapi_pblock_set(pb, SLAPI_PLUGIN_DESCRIPTION,
- (void *) &pdesc) != 0 ||
- slapi_pblock_set(pb, SLAPI_PLUGIN_INTERNAL_POST_DELETE_FN,
- (void *) memberof_postop_del) != 0 ||
- slapi_pblock_set( pb, SLAPI_PLUGIN_INTERNAL_POST_MODRDN_FN,
- (void *) memberof_postop_modrdn ) != 0 ||
- slapi_pblock_set( pb, SLAPI_PLUGIN_INTERNAL_POST_MODIFY_FN,
- (void *) memberof_postop_modify ) != 0 ||
- slapi_pblock_set( pb, SLAPI_PLUGIN_INTERNAL_POST_ADD_FN,
- (void *) memberof_postop_add ) != 0) {
- slapi_log_error(SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_internal_postop_init: failed to register plugin\n");
- status = -1;
- }
- return status;
- }
- /*
- * memberof_postop_start()
- *
- * Do plugin start up stuff
- *
- */
- int memberof_postop_start(Slapi_PBlock *pb)
- {
- Slapi_PBlock *search_pb = NULL;
- Slapi_Entry **entries = NULL;
- Slapi_Entry *config_e = NULL; /* entry containing plugin config */
- char *config_area = NULL;
- int result = 0;
- int rc = 0;
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "--> memberof_postop_start\n" );
- /* Check if we're already started */
- if (g_plugin_started) {
- goto bail;
- }
- memberof_operation_lock = PR_NewMonitor();
- if(0 == memberof_operation_lock)
- {
- rc = -1;
- goto bail;
- }
- /* Set the alternate config area if one is defined. */
- slapi_pblock_get(pb, SLAPI_PLUGIN_CONFIG_AREA, &config_area);
- if (config_area)
- {
- search_pb = slapi_pblock_new();
- slapi_search_internal_set_pb(search_pb, config_area, LDAP_SCOPE_BASE, "objectclass=*",
- NULL, 0, NULL, NULL, memberof_get_plugin_id(), 0);
- slapi_search_internal_pb(search_pb);
- slapi_pblock_get(search_pb, SLAPI_PLUGIN_INTOP_RESULT, &result);
- if (LDAP_SUCCESS != result) {
- if (result == LDAP_NO_SUCH_OBJECT) {
- /* log an error and use the plugin entry for the config */
- slapi_log_error(SLAPI_LOG_PLUGIN, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_postop_start: Config entry \"%s\" does "
- "not exist.\n", config_area);
- rc = -1;
- goto bail;
- }
- } else {
- slapi_pblock_get(search_pb, SLAPI_PLUGIN_INTOP_SEARCH_ENTRIES, &entries);
- if(entries && entries[0]){
- config_e = entries[0];
- } else {
- slapi_log_error(SLAPI_LOG_PLUGIN, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_postop_start: Config entry \"%s\" was "
- "not located.\n", config_area);
- rc = -1;
- goto bail;
- }
- }
- } else {
- /* The plugin entry itself contains the config */
- if ( slapi_pblock_get( pb, SLAPI_ADD_ENTRY, &config_e ) != 0 ) {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "missing config entry\n" );
- rc = -1;
- goto bail;
- }
- }
- memberof_set_plugin_area(slapi_entry_get_sdn(config_e));
- memberof_set_config_area(slapi_entry_get_sdn(config_e));
- if (( rc = memberof_config( config_e )) != LDAP_SUCCESS ) {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "configuration failed (%s)\n", ldap_err2string( rc ));
- rc = -1;
- goto bail;
- }
- rc = slapi_task_register_handler("memberof task", memberof_task_add);
- if(rc)
- {
- goto bail;
- }
- g_plugin_started = 1;
- /*
- * TODO: start up operation actor thread
- * need to get to a point where server failure
- * or shutdown doesn't hose our operations
- * so we should create a task entry that contains
- * all required information to complete the operation
- * then the tasks can be restarted safely if
- * interrupted
- */
- bail:
- slapi_free_search_results_internal(search_pb);
- slapi_pblock_destroy(search_pb);
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "<-- memberof_postop_start\n" );
- return rc;
- }
- /*
- * memberof_postop_close()
- *
- * Do plugin shut down stuff
- *
- */
- int memberof_postop_close(Slapi_PBlock *pb)
- {
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "--> memberof_postop_close\n" );
- g_plugin_started = 0;
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "<-- memberof_postop_close\n" );
- return 0;
- }
- void
- memberof_set_config_area(Slapi_DN *dn)
- {
- slapi_sdn_free(&_ConfigAreaDN);
- _ConfigAreaDN = slapi_sdn_dup(dn);
- }
- Slapi_DN *
- memberof_get_config_area()
- {
- return _ConfigAreaDN;
- }
- void
- memberof_set_plugin_area(Slapi_DN *sdn)
- {
- slapi_sdn_free(&_pluginDN);
- _pluginDN = slapi_sdn_dup(sdn);
- }
- Slapi_DN *
- memberof_get_plugin_area()
- {
- return _pluginDN;
- }
- /*
- * memberof_postop_del()
- *
- * All entries with a memberOf attribute that contains the group DN get retrieved
- * and have the their memberOf attribute regenerated (it is far too complex and
- * error prone to attempt to change only those dn values involved in this case -
- * mainly because the deleted group may itself be a member of other groups which
- * may be members of other groups etc. in a big recursive mess involving dependency
- * chains that must be created and traversed in order to decide if an entry should
- * really have those groups removed too)
- */
- int memberof_postop_del(Slapi_PBlock *pb)
- {
- int ret = SLAPI_PLUGIN_SUCCESS;
- MemberOfConfig configCopy = {0, 0, 0, 0};
- Slapi_DN *sdn;
- void *caller_id = NULL;
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "--> memberof_postop_del\n" );
- /* We don't want to process internal modify
- * operations that originate from this plugin. */
- slapi_pblock_get(pb, SLAPI_PLUGIN_IDENTITY, &caller_id);
- if (caller_id == memberof_get_plugin_id()) {
- /* Just return without processing */
- return SLAPI_PLUGIN_SUCCESS;
- }
- if(memberof_oktodo(pb) && (sdn = memberof_getsdn(pb)))
- {
- struct slapi_entry *e = NULL;
- slapi_pblock_get( pb, SLAPI_ENTRY_PRE_OP, &e );
- /* We need to get the config lock first. Trying to get the
- * config lock after we already hold the op lock can cause
- * a deadlock. */
- memberof_rlock_config();
- /* copy config so it doesn't change out from under us */
- memberof_copy_config(&configCopy, memberof_get_config());
- memberof_unlock_config();
- /* get the memberOf operation lock */
- memberof_lock();
-
- /* remove this DN from the
- * membership lists of groups
- */
- memberof_del_dn_from_groups(pb, &configCopy, sdn);
- /* is the entry of interest as a group? */
- if(e && configCopy.group_filter && !slapi_filter_test_simple(e, configCopy.group_filter))
- {
- int i = 0;
- Slapi_Attr *attr = 0;
- /* Loop through to find each grouping attribute separately. */
- for (i = 0; configCopy.groupattrs[i]; i++)
- {
- if (0 == slapi_entry_attr_find(e, configCopy.groupattrs[i], &attr))
- {
- memberof_del_attr_list(pb, &configCopy, sdn, attr);
- }
- }
- }
- memberof_unlock();
- memberof_free_config(&configCopy);
- }
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "<-- memberof_postop_del\n" );
- return ret;
- }
- typedef struct _memberof_del_dn_data
- {
- char *dn;
- char *type;
- } memberof_del_dn_data;
- /* Deletes a member dn from all groups that refer to it. */
- static void
- memberof_del_dn_from_groups(Slapi_PBlock *pb, MemberOfConfig *config, Slapi_DN *sdn)
- {
- int i = 0;
- char *groupattrs[2] = {0, 0};
- /* Loop through each grouping attribute to find groups that have
- * dn as a member. For any matches, delete the dn value from the
- * same grouping attribute. */
- for (i = 0; config->groupattrs && config->groupattrs[i]; i++)
- {
- memberof_del_dn_data data = {(char *)slapi_sdn_get_dn(sdn),
- config->groupattrs[i]};
- groupattrs[0] = config->groupattrs[i];
- memberof_call_foreach_dn(pb, sdn, groupattrs,
- memberof_del_dn_type_callback, &data);
- }
- }
- int memberof_del_dn_type_callback(Slapi_Entry *e, void *callback_data)
- {
- int rc = 0;
- LDAPMod mod;
- LDAPMod *mods[2];
- char *val[2];
- Slapi_PBlock *mod_pb = 0;
- mod_pb = slapi_pblock_new();
- mods[0] = &mod;
- mods[1] = 0;
- val[0] = ((memberof_del_dn_data *)callback_data)->dn;
- val[1] = 0;
- mod.mod_op = LDAP_MOD_DELETE;
- mod.mod_type = ((memberof_del_dn_data *)callback_data)->type;
- mod.mod_values = val;
- slapi_modify_internal_set_pb_ext(
- mod_pb, slapi_entry_get_sdn(e),
- mods, 0, 0,
- memberof_get_plugin_id(), 0);
- slapi_modify_internal_pb(mod_pb);
- slapi_pblock_get(mod_pb,
- SLAPI_PLUGIN_INTOP_RESULT,
- &rc);
- slapi_pblock_destroy(mod_pb);
- return rc;
- }
- /*
- * Does a callback search of "type=dn" under the db suffix that "dn" is in,
- * unless all_backends is set, then we look at all the backends. If "dn"
- * is a user, you'd want "type" to be "member". If "dn" is a group, you
- * could want type to be either "member" or "memberOf" depending on the case.
- */
- int memberof_call_foreach_dn(Slapi_PBlock *pb, Slapi_DN *sdn,
- char **types, plugin_search_entry_callback callback, void *callback_data)
- {
- Slapi_PBlock *search_pb = NULL;
- Slapi_DN *base_sdn = NULL;
- Slapi_Backend *be = NULL;
- char *escaped_filter_val;
- char *filter_str = NULL;
- char *cookie = NULL;
- int all_backends = memberof_config_get_all_backends();
- int types_name_len = 0;
- int num_types = 0;
- int dn_len = slapi_sdn_get_ndn_len(sdn);
- int free_it = 0;
- int rc = 0;
- int i = 0;
- /* Count the number of types. */
- for (num_types = 0; types && types[num_types]; num_types++)
- {
- /* Add up the total length of all attribute names.
- * We need to know this for building the filter. */
- types_name_len += strlen(types[num_types]);
- }
- /* Escape the dn, and build the search filter. */
- escaped_filter_val = slapi_escape_filter_value((char *)slapi_sdn_get_dn(sdn), dn_len);
- if(escaped_filter_val){
- dn_len = strlen(escaped_filter_val);
- free_it = 1;
- } else {
- escaped_filter_val = (char *)slapi_sdn_get_dn(sdn);
- }
- if (num_types > 1)
- {
- int bytes_out = 0;
- int filter_str_len = types_name_len + (num_types * (3 + dn_len)) + 4;
- /* Allocate enough space for the filter */
- filter_str = slapi_ch_malloc(filter_str_len);
- /* Add beginning of filter. */
- bytes_out = snprintf(filter_str, filter_str_len - bytes_out, "(|");
- /* Add filter section for each type. */
- for (i = 0; types[i]; i++)
- {
- bytes_out += snprintf(filter_str + bytes_out, filter_str_len - bytes_out,
- "(%s=%s)", types[i], escaped_filter_val);
- }
- /* Add end of filter. */
- snprintf(filter_str + bytes_out, filter_str_len - bytes_out, ")");
- }
- else if (num_types == 1)
- {
- filter_str = slapi_ch_smprintf("(%s=%s)", types[0], escaped_filter_val);
- }
- if(free_it)
- slapi_ch_free_string(&escaped_filter_val);
- if(filter_str == NULL){
- return rc;
- }
- search_pb = slapi_pblock_new();
- be = slapi_get_first_backend(&cookie);
- while(be){
- if(!all_backends){
- be = slapi_be_select(sdn);
- if(be == NULL){
- break;
- }
- }
- if((base_sdn = (Slapi_DN *)slapi_be_getsuffix(be,0)) == NULL){
- if(!all_backends){
- break;
- } else {
- /* its ok, goto the next backend */
- be = slapi_get_next_backend(cookie);
- continue;
- }
- }
- slapi_search_internal_set_pb(search_pb, slapi_sdn_get_dn(base_sdn),
- LDAP_SCOPE_SUBTREE, filter_str, 0, 0, 0, 0, memberof_get_plugin_id(), 0);
- slapi_search_internal_callback_pb(search_pb, callback_data, 0, callback, 0);
- if(!all_backends){
- break;
- }
- slapi_pblock_init(search_pb);
- be = slapi_get_next_backend(cookie);
- }
- slapi_pblock_destroy(search_pb);
- slapi_ch_free((void **)&cookie);
- slapi_ch_free_string(&filter_str);
- return rc;
- }
- /*
- * memberof_postop_modrdn()
- *
- * All entries with a memberOf attribute that contains the old group DN get retrieved
- * and have the old group DN deleted and the new group DN added to their memberOf attribute
- */
- int memberof_postop_modrdn(Slapi_PBlock *pb)
- {
- int ret = SLAPI_PLUGIN_SUCCESS;
- void *caller_id = NULL;
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "--> memberof_postop_modrdn\n" );
- /* We don't want to process internal modify
- * operations that originate from this plugin. */
- slapi_pblock_get(pb, SLAPI_PLUGIN_IDENTITY, &caller_id);
- if (caller_id == memberof_get_plugin_id()) {
- /* Just return without processing */
- return SLAPI_PLUGIN_SUCCESS;
- }
- if(memberof_oktodo(pb))
- {
- MemberOfConfig *mainConfig = 0;
- MemberOfConfig configCopy = {0, 0, 0, 0};
- struct slapi_entry *pre_e = NULL;
- struct slapi_entry *post_e = NULL;
- Slapi_DN *pre_sdn = 0;
- Slapi_DN *post_sdn = 0;
- slapi_pblock_get( pb, SLAPI_ENTRY_PRE_OP, &pre_e );
- slapi_pblock_get( pb, SLAPI_ENTRY_POST_OP, &post_e );
-
- if(pre_e && post_e)
- {
- pre_sdn = slapi_entry_get_sdn(pre_e);
- post_sdn = slapi_entry_get_sdn(post_e);
- }
- /* copy config so it doesn't change out from under us */
- memberof_rlock_config();
- mainConfig = memberof_get_config();
- memberof_copy_config(&configCopy, mainConfig);
- memberof_unlock_config();
- memberof_lock();
- /* update any downstream members */
- if(pre_sdn && post_sdn && configCopy.group_filter &&
- !slapi_filter_test_simple(post_e, configCopy.group_filter))
- {
- int i = 0;
- Slapi_Attr *attr = 0;
- /* get a list of member attributes present in the group
- * entry that is being renamed. */
- for (i = 0; configCopy.groupattrs[i]; i++)
- {
- if(0 == slapi_entry_attr_find(post_e, configCopy.groupattrs[i], &attr))
- {
- if(memberof_moddn_attr_list(pb, &configCopy, pre_sdn,
- post_sdn, attr) != 0){
- break;
- }
- }
- }
- }
- /* It's possible that this is an entry who is a member
- * of other group entries. We need to update any member
- * attributes to refer to the new name. */
- if (pre_sdn && post_sdn) {
- memberof_replace_dn_from_groups(pb, &configCopy, pre_sdn, post_sdn);
- }
- memberof_unlock();
- memberof_free_config(&configCopy);
- }
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "<-- memberof_postop_modrdn\n" );
- return ret;
- }
- typedef struct _replace_dn_data
- {
- char *pre_dn;
- char *post_dn;
- char *type;
- } replace_dn_data;
- /* Finds any groups that have pre_dn as a member and modifies them to
- * to use post_dn instead. */
- static void
- memberof_replace_dn_from_groups(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *pre_sdn, Slapi_DN *post_sdn)
- {
- int i = 0;
- char *groupattrs[2] = {0, 0};
- /* Loop through each grouping attribute to find groups that have
- * pre_dn as a member. For any matches, replace pre_dn with post_dn
- * using the same grouping attribute. */
- for (i = 0; config->groupattrs && config->groupattrs[i]; i++)
- {
- replace_dn_data data = {(char *)slapi_sdn_get_dn(pre_sdn),
- (char *)slapi_sdn_get_dn(post_sdn),
- config->groupattrs[i]};
- groupattrs[0] = config->groupattrs[i];
- memberof_call_foreach_dn(pb, pre_sdn, groupattrs,
- memberof_replace_dn_type_callback, &data);
- }
- }
- int memberof_replace_dn_type_callback(Slapi_Entry *e, void *callback_data)
- {
- int rc = 0;
- LDAPMod delmod;
- LDAPMod addmod;
- LDAPMod *mods[3];
- char *delval[2];
- char *addval[2];
- Slapi_PBlock *mod_pb = 0;
- mod_pb = slapi_pblock_new();
- mods[0] = &delmod;
- mods[1] = &addmod;
- mods[2] = 0;
- delval[0] = ((replace_dn_data *)callback_data)->pre_dn;
- delval[1] = 0;
- delmod.mod_op = LDAP_MOD_DELETE;
- delmod.mod_type = ((replace_dn_data *)callback_data)->type;
- delmod.mod_values = delval;
- addval[0] = ((replace_dn_data *)callback_data)->post_dn;
- addval[1] = 0;
- addmod.mod_op = LDAP_MOD_ADD;
- addmod.mod_type = ((replace_dn_data *)callback_data)->type;
- addmod.mod_values = addval;
- slapi_modify_internal_set_pb_ext(
- mod_pb, slapi_entry_get_sdn(e),
- mods, 0, 0,
- memberof_get_plugin_id(), 0);
- slapi_modify_internal_pb(mod_pb);
- slapi_pblock_get(mod_pb,
- SLAPI_PLUGIN_INTOP_RESULT,
- &rc);
- slapi_pblock_destroy(mod_pb);
- return rc;
- }
- /*
- * memberof_postop_modify()
- *
- * Added members are retrieved and have the group DN added to their memberOf attribute
- * Deleted members are retrieved and have the group DN deleted from their memberOf attribute
- * On replace of the membership attribute values:
- * 1. Sort old and new values
- * 2. Iterate through both lists at same time
- * 3. Any value not in old list but in new list - add group DN to memberOf attribute
- * 4. Any value in old list but not in new list - remove group DN from memberOf attribute
- *
- * Note: this will suck for large groups but nonetheless is optimal (it's linear) given
- * current restrictions i.e. originally adding members in sorted order would allow
- * us to sort one list only (the new one) but that is under server control, not this plugin
- */
- int memberof_postop_modify(Slapi_PBlock *pb)
- {
- int ret = SLAPI_PLUGIN_SUCCESS;
- Slapi_DN *sdn = NULL;
- Slapi_Mods *smods = 0;
- Slapi_Mod *smod = 0;
- LDAPMod **mods;
- Slapi_Mod *next_mod = 0;
- void *caller_id = NULL;
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "--> memberof_postop_modify\n" );
- /* We don't want to process internal modify
- * operations that originate from this plugin. */
- slapi_pblock_get(pb, SLAPI_PLUGIN_IDENTITY, &caller_id);
- if (caller_id == memberof_get_plugin_id()) {
- /* Just return without processing */
- return SLAPI_PLUGIN_SUCCESS;
- }
- /* check if we are updating the shared config entry */
- slapi_pblock_get(pb, SLAPI_TARGET_SDN, &sdn);
- if (slapi_sdn_issuffix(sdn, memberof_get_config_area()) &&
- slapi_sdn_compare(sdn, memberof_get_config_area()) == 0)
- {
- Slapi_Entry *entry = NULL;
- char returntext[SLAPI_DSE_RETURNTEXT_SIZE];
- int result = 0;
- slapi_pblock_get(pb, SLAPI_ENTRY_POST_OP, &entry);
- if(entry){
- if( SLAPI_DSE_CALLBACK_ERROR == memberof_apply_config (pb, NULL, entry, &result, returntext, NULL)){
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM, "%s", returntext);
- ret = SLAPI_PLUGIN_FAILURE;
- goto done;
- }
- } else {
- /* this should not happen since this was validated in preop */
- ret = SLAPI_PLUGIN_FAILURE;
- goto done;
- }
- /* we're done, no need to do the other processing */
- goto done;
- }
- if(memberof_oktodo(pb) && (sdn = memberof_getsdn(pb)))
- {
- int config_copied = 0;
- MemberOfConfig *mainConfig = 0;
- MemberOfConfig configCopy = {0, 0, 0, 0};
- /* get the mod set */
- slapi_pblock_get(pb, SLAPI_MODIFY_MODS, &mods);
- smods = slapi_mods_new();
- slapi_mods_init_byref(smods, mods);
- next_mod = slapi_mod_new();
- smod = slapi_mods_get_first_smod(smods, next_mod);
- while(smod)
- {
- int interested = 0;
- char *type = (char *)slapi_mod_get_type(smod);
- /* We only want to copy the config if we encounter an
- * operation that we need to act on. We also want to
- * only copy the config the first time it's needed so
- * it remains the same for all mods in the operation,
- * despite any config changes that may be made. */
- if (!config_copied)
- {
- memberof_rlock_config();
- mainConfig = memberof_get_config();
- if (memberof_is_grouping_attr(type, mainConfig))
- {
- interested = 1;
- /* copy config so it doesn't change out from under us */
- memberof_copy_config(&configCopy, mainConfig);
- config_copied = 1;
- }
- memberof_unlock_config();
- } else {
- if (memberof_is_grouping_attr(type, &configCopy))
- {
- interested = 1;
- }
- }
- if(interested)
- {
- int op = slapi_mod_get_operation(smod);
- memberof_lock();
- /* the modify op decides the function */
- switch(op & ~LDAP_MOD_BVALUES)
- {
- case LDAP_MOD_ADD:
- {
- /* add group DN to targets */
- memberof_add_smod_list(pb, &configCopy, sdn, smod);
- break;
- }
-
- case LDAP_MOD_DELETE:
- {
- /* If there are no values in the smod, we should
- * just do a replace instead. The user is just
- * trying to delete all members from this group
- * entry, which the replace code deals with. */
- if (slapi_mod_get_num_values(smod) == 0)
- {
- memberof_replace_list(pb, &configCopy, sdn);
- }
- else
- {
- /* remove group DN from target values in smod*/
- memberof_del_smod_list(pb, &configCopy, sdn, smod);
- }
- break;
- }
- case LDAP_MOD_REPLACE:
- {
- /* replace current values */
- memberof_replace_list(pb, &configCopy, sdn);
- break;
- }
- default:
- {
- slapi_log_error(
- SLAPI_LOG_PLUGIN,
- MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_postop_modify: unknown mod type\n" );
- break;
- }
- }
- memberof_unlock();
- }
- slapi_mod_done(next_mod);
- smod = slapi_mods_get_next_smod(smods, next_mod);
- }
- if (config_copied)
- {
- memberof_free_config(&configCopy);
- }
- slapi_mod_free(&next_mod);
- slapi_mods_free(&smods);
- }
- done:
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "<-- memberof_postop_modify\n" );
- return ret;
- }
- /*
- * memberof_postop_add()
- *
- * All members in the membership attribute of the new entry get retrieved
- * and have the group DN added to their memberOf attribute
- */
- int memberof_postop_add(Slapi_PBlock *pb)
- {
- int ret = SLAPI_PLUGIN_SUCCESS;
- int interested = 0;
- Slapi_DN *sdn = 0;
- void *caller_id = NULL;
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "--> memberof_postop_add\n" );
- /* We don't want to process internal modify
- * operations that originate from this plugin. */
- slapi_pblock_get(pb, SLAPI_PLUGIN_IDENTITY, &caller_id);
- if (caller_id == memberof_get_plugin_id()) {
- /* Just return without processing */
- return SLAPI_PLUGIN_SUCCESS;
- }
- if(memberof_oktodo(pb) && (sdn = memberof_getsdn(pb)))
- {
- MemberOfConfig *mainConfig = 0;
- MemberOfConfig configCopy = {0, 0, 0, 0};
- struct slapi_entry *e = NULL;
- slapi_pblock_get( pb, SLAPI_ENTRY_POST_OP, &e );
-
- /* is the entry of interest? */
- memberof_rlock_config();
- mainConfig = memberof_get_config();
- if(e && mainConfig && mainConfig->group_filter &&
- !slapi_filter_test_simple(e, mainConfig->group_filter))
- {
- interested = 1;
- /* copy config so it doesn't change out from under us */
- memberof_copy_config(&configCopy, mainConfig);
- }
- memberof_unlock_config();
- if(interested)
- {
- int i = 0;
- Slapi_Attr *attr = 0;
- memberof_lock();
- for (i = 0; configCopy.groupattrs[i]; i++)
- {
- if(0 == slapi_entry_attr_find(e, configCopy.groupattrs[i], &attr))
- {
- memberof_add_attr_list(pb, &configCopy, sdn, attr);
- }
- }
- memberof_unlock();
- memberof_free_config(&configCopy);
- }
- }
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "<-- memberof_postop_add\n" );
- return ret;
- }
- /*** Support functions ***/
- /*
- * memberof_oktodo()
- *
- * Check that the op succeeded
- * Note: we also respond to replicated ops so we don't test for that
- * this does require that the memberOf attribute not be replicated
- * and this means that memberof is consistent with local state
- * not the network system state
- *
- */
- int memberof_oktodo(Slapi_PBlock *pb)
- {
- int ret = 1;
- int oprc = 0;
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "--> memberof_postop_oktodo\n" );
- if (!g_plugin_started) {
- ret = 0;
- goto bail;
- }
- if(slapi_pblock_get(pb, SLAPI_PLUGIN_OPRETURN, &oprc) != 0)
- {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_postop_oktodo: could not get parameters\n" );
- ret = -1;
- }
- /* this plugin should only execute if the operation succeeded
- */
- if(oprc != 0)
- {
- ret = 0;
- }
-
- slapi_log_error( SLAPI_LOG_TRACE, MEMBEROF_PLUGIN_SUBSYSTEM,
- "<-- memberof_postop_oktodo\n" );
- bail:
- return ret;
- }
- static Slapi_DN *
- memberof_getsdn(Slapi_PBlock *pb)
- {
- Slapi_DN *sdn = NULL;
- slapi_pblock_get(pb, SLAPI_TARGET_SDN, &sdn);
- return sdn;
- }
- /*
- * memberof_modop_one()
- *
- * Perform op on memberof attribute of op_to using op_this as the value
- * However, if op_to happens to be a group, we must arrange for the group
- * members to have the mod performed on them instead, and we must take
- * care to not recurse when we have visted a group before
- *
- * Also, we must not delete entries that are a member of the group
- */
- int memberof_modop_one(Slapi_PBlock *pb, MemberOfConfig *config, int mod_op,
- Slapi_DN *op_this_sdn, Slapi_DN *op_to_sdn)
- {
- return memberof_modop_one_r(pb, config, mod_op, op_this_sdn,
- op_this_sdn, op_to_sdn, 0);
- }
- /* memberof_modop_one_r()
- *
- * recursive function to perform above (most things don't need the replace arg)
- */
- int memberof_modop_one_r(Slapi_PBlock *pb, MemberOfConfig *config, int mod_op,
- Slapi_DN *group_sdn, Slapi_DN *op_this_sdn, Slapi_DN *op_to_sdn,
- memberofstringll *stack)
- {
- return memberof_modop_one_replace_r(
- pb, config, mod_op, group_sdn, op_this_sdn, 0, op_to_sdn, stack);
- }
- /* memberof_modop_one_replace_r()
- *
- * recursive function to perform above (with added replace arg)
- */
- int
- memberof_modop_one_replace_r(Slapi_PBlock *pb, MemberOfConfig *config,
- int mod_op, Slapi_DN *group_sdn, Slapi_DN *op_this_sdn,
- Slapi_DN *replace_with_sdn, Slapi_DN *op_to_sdn, memberofstringll *stack)
- {
- int rc = 0;
- LDAPMod mod;
- LDAPMod replace_mod;
- LDAPMod *mods[3];
- char *val[2];
- char *replace_val[2];
- Slapi_PBlock *mod_pb = 0;
- Slapi_Entry *e = 0;
- memberofstringll *ll = 0;
- char *op_str = 0;
- const char *op_to;
- const char *op_this;
- Slapi_Value *to_dn_val;
- Slapi_Value *this_dn_val;
- op_to = slapi_sdn_get_ndn(op_to_sdn);
- op_this = slapi_sdn_get_ndn(op_this_sdn);
- to_dn_val = slapi_value_new_string(op_to);
- this_dn_val = slapi_value_new_string(op_this);
- if(this_dn_val == NULL || to_dn_val == NULL){
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_modop_one_replace_r: failed to get DN values (NULL)\n");
- goto bail;
- }
- /* op_this and op_to are both case-normalized */
- slapi_value_set_flags(this_dn_val, SLAPI_ATTR_FLAG_NORMALIZED_CIS);
- slapi_value_set_flags(to_dn_val, SLAPI_ATTR_FLAG_NORMALIZED_CIS);
- if (config == NULL) {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_modop_one_replace_r: NULL config parameter\n");
- goto bail;
- }
- /* determine if this is a group op or single entry */
- slapi_search_internal_get_entry( op_to_sdn, config->groupattrs,
- &e, memberof_get_plugin_id());
- if(!e)
- {
- /* In the case of a delete, we need to worry about the
- * missing entry being a nested group. There's a small
- * window where another thread may have deleted a nested
- * group that our group_dn entry refers to. This has the
- * potential of us missing some indirect member entries
- * that need to be updated. */
- if(LDAP_MOD_DELETE == mod_op)
- {
- Slapi_PBlock *search_pb = slapi_pblock_new();
- Slapi_DN *base_sdn = 0;
- Slapi_Backend *be = 0;
- char *filter_str;
- char *cookie = NULL;
- int n_entries = 0;
- int all_backends = config->allBackends;
- filter_str = slapi_filter_sprintf("(%s=%s%s)", config->memberof_attr, ESC_NEXT_VAL, op_to);
- be = slapi_get_first_backend(&cookie);
- while(be){
- /*
- * We can't tell for sure if the op_to entry is a
- * user or a group since the entry doesn't exist
- * anymore. We can safely ignore the missing entry
- * if no other entries have a memberOf attribute that
- * points to the missing entry.
- */
- if(!all_backends){
- be = slapi_be_select(op_to_sdn);
- if(be == NULL){
- break;
- }
- }
- if((base_sdn = (Slapi_DN*)slapi_be_getsuffix(be,0)) == NULL){
- if(!all_backends){
- break;
- } else {
- be = slapi_get_next_backend (cookie);
- continue;
- }
- }
- if(filter_str)
- {
- slapi_search_internal_set_pb(search_pb, slapi_sdn_get_dn(base_sdn),
- LDAP_SCOPE_SUBTREE, filter_str, 0, 0, 0, 0,
- memberof_get_plugin_id(), 0);
- if (slapi_search_internal_pb(search_pb))
- {
- /* get result and log an error */
- int res = 0;
- slapi_pblock_get(search_pb, SLAPI_PLUGIN_INTOP_RESULT, &res);
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_modop_one_replace_r: error searching for members: %d\n", res);
- } else {
- slapi_pblock_get(search_pb, SLAPI_NENTRIES, &n_entries);
- if(n_entries > 0)
- {
- /* We want to fixup the membership for the
- * entries that referred to the missing group
- * entry. This will fix the references to
- * the missing group as well as the group
- * represented by op_this. */
- memberof_test_membership(pb, config, op_to_sdn);
- }
- }
- slapi_free_search_results_internal(search_pb);
- }
- slapi_pblock_init(search_pb);
- if(!all_backends){
- break;
- }
- be = slapi_get_next_backend (cookie);
- }
- slapi_pblock_destroy(search_pb);
- slapi_ch_free_string(&filter_str);
- slapi_ch_free((void **)&cookie);
- }
- goto bail;
- }
- if(LDAP_MOD_DELETE == mod_op)
- {
- op_str = "DELETE";
- }
- else if(LDAP_MOD_ADD == mod_op)
- {
- op_str = "ADD";
- }
- else if(LDAP_MOD_REPLACE == mod_op)
- {
- op_str = "REPLACE";
- }
- else
- {
- op_str = "UNKNOWN";
- }
- slapi_log_error( SLAPI_LOG_PLUGIN, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_modop_one_replace_r: %s %s in %s\n"
- ,op_str, op_this, op_to);
- if(config->group_filter && !slapi_filter_test_simple(e, config->group_filter))
- {
- /* group */
- Slapi_Value *ll_dn_val = 0;
- Slapi_Attr *members = 0;
- int i = 0;
- ll = stack;
- /* have we been here before? */
- while(ll)
- {
- ll_dn_val = slapi_value_new_string(ll->dn);
- /* ll->dn is case-normalized */
- slapi_value_set_flags(ll_dn_val, SLAPI_ATTR_FLAG_NORMALIZED_CIS);
- if(0 == memberof_compare(config, &ll_dn_val, &to_dn_val))
- {
- slapi_value_free(&ll_dn_val);
- /* someone set up infinitely
- recursive groups - bail out */
- slapi_log_error( SLAPI_LOG_PLUGIN,
- MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_modop_one_replace_r: group recursion"
- " detected in %s\n"
- ,op_to);
- goto bail;
- }
- slapi_value_free(&ll_dn_val);
- ll = ll->next;
- }
- /* do op on group */
- slapi_log_error( SLAPI_LOG_PLUGIN,
- MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_modop_one_replace_r: descending into group %s\n",
- op_to);
- /* Add the nested group's DN to the stack so we can detect loops later. */
- ll = (memberofstringll*)slapi_ch_malloc(sizeof(memberofstringll));
- ll->dn = op_to;
- ll->next = stack;
-
- /* Go through each grouping attribute one at a time. */
- for (i = 0; config->groupattrs[i]; i++)
- {
- slapi_entry_attr_find( e, config->groupattrs[i], &members );
- if(members)
- {
- if(memberof_mod_attr_list_r(pb, config, mod_op, group_sdn,
- op_this_sdn, members, ll) != 0){
- rc = -1;
- goto bail;
- }
- }
- }
- {
- /* crazyness follows:
- * strict-aliasing doesn't like the required cast
- * to void for slapi_ch_free so we are made to
- * juggle to get a normal thing done
- */
- void *pll = ll;
- slapi_ch_free(&pll);
- ll = 0;
- }
- }
- /* continue with operation */
- {
- /* We want to avoid listing a group as a memberOf itself
- * in case someone set up a circular grouping.
- */
- if (0 == memberof_compare(config, &this_dn_val, &to_dn_val))
- {
- const char *strval = "NULL";
- if (this_dn_val) {
- strval = slapi_value_get_string(this_dn_val);
- }
- slapi_log_error( SLAPI_LOG_PLUGIN,
- MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_modop_one_replace_r: not processing memberOf "
- "operations on self entry: %s\n", strval);
- goto bail;
- }
- /* For add and del modify operations, we just regenerate the
- * memberOf attribute. */
- if(LDAP_MOD_DELETE == mod_op || LDAP_MOD_ADD == mod_op)
- {
- /* find parent groups and replace our member attr */
- memberof_fix_memberof_callback(e, config);
- } else {
- /* single entry - do mod */
- mod_pb = slapi_pblock_new();
- mods[0] = &mod;
- if(LDAP_MOD_REPLACE == mod_op)
- {
- mods[1] = &replace_mod;
- mods[2] = 0;
- }
- else
- {
- mods[1] = 0;
- }
- val[0] = (char *)op_this;
- val[1] = 0;
- mod.mod_op = LDAP_MOD_REPLACE == mod_op?LDAP_MOD_DELETE:mod_op;
- mod.mod_type = config->memberof_attr;
- mod.mod_values = val;
- if(LDAP_MOD_REPLACE == mod_op)
- {
- replace_val[0] = (char *)slapi_sdn_get_dn(replace_with_sdn);
- replace_val[1] = 0;
- replace_mod.mod_op = LDAP_MOD_ADD;
- replace_mod.mod_type = config->memberof_attr;
- replace_mod.mod_values = replace_val;
- }
- slapi_modify_internal_set_pb(
- mod_pb, op_to,
- mods, 0, 0,
- memberof_get_plugin_id(), 0);
- slapi_modify_internal_pb(mod_pb);
- slapi_pblock_get(mod_pb,
- SLAPI_PLUGIN_INTOP_RESULT,
- &rc);
- slapi_pblock_destroy(mod_pb);
- }
- }
- bail:
- slapi_value_free(&to_dn_val);
- slapi_value_free(&this_dn_val);
- slapi_entry_free(e);
- return rc;
- }
- /*
- * memberof_add_one()
- *
- * Add addthis DN to the memberof attribute of addto
- *
- */
- int
- memberof_add_one(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *addthis_sdn, Slapi_DN *addto_sdn)
- {
- return memberof_modop_one(pb, config, LDAP_MOD_ADD, addthis_sdn, addto_sdn);
- }
- /*
- * memberof_del_one()
- *
- * Delete delthis DN from the memberof attribute of delfrom
- *
- */
- int
- memberof_del_one(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *delthis_sdn, Slapi_DN *delfrom_sdn)
- {
- return memberof_modop_one(pb, config, LDAP_MOD_DELETE, delthis_sdn, delfrom_sdn);
- }
- /*
- * memberof_mod_smod_list()
- *
- * Perform mod for group DN to the memberof attribute of the list of targets
- *
- */
- int memberof_mod_smod_list(Slapi_PBlock *pb, MemberOfConfig *config, int mod,
- Slapi_DN *group_sdn, Slapi_Mod *smod)
- {
- int rc = 0;
- struct berval *bv = slapi_mod_get_first_value(smod);
- int last_size = 0;
- char *last_str = 0;
- Slapi_DN *sdn = slapi_sdn_new();
- while(bv)
- {
- char *dn_str = 0;
- if(last_size > bv->bv_len)
- {
- dn_str = last_str;
- }
- else
- {
- int the_size = (bv->bv_len * 2) + 1;
- if(last_str)
- slapi_ch_free_string(&last_str);
- dn_str = (char*)slapi_ch_malloc(the_size);
- last_str = dn_str;
- last_size = the_size;
- }
- memset(dn_str, 0, last_size);
- strncpy(dn_str, bv->bv_val, (size_t)bv->bv_len);
- slapi_sdn_set_dn_byref(sdn, dn_str);
- memberof_modop_one(pb, config, mod, group_sdn, sdn);
- bv = slapi_mod_get_next_value(smod);
- }
- slapi_sdn_free(&sdn);
- if(last_str)
- slapi_ch_free_string(&last_str);
- return rc;
- }
- /*
- * memberof_add_smod_list()
- *
- * Add group DN to the memberof attribute of the list of targets
- *
- */
- int memberof_add_smod_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn, Slapi_Mod *smod)
- {
- return memberof_mod_smod_list(pb, config, LDAP_MOD_ADD, group_sdn, smod);
- }
- /*
- * memberof_del_smod_list()
- *
- * Remove group DN from the memberof attribute of the list of targets
- *
- */
- int memberof_del_smod_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn, Slapi_Mod *smod)
- {
- return memberof_mod_smod_list(pb, config, LDAP_MOD_DELETE, group_sdn, smod);
- }
- /**
- * Plugin identity mgmt
- */
- void memberof_set_plugin_id(void * plugin_id)
- {
- _PluginID=plugin_id;
- }
- void * memberof_get_plugin_id()
- {
- return _PluginID;
- }
- /*
- * memberof_mod_attr_list()
- *
- * Perform mod for group DN to the memberof attribute of the list of targets
- *
- */
- int memberof_mod_attr_list(Slapi_PBlock *pb, MemberOfConfig *config, int mod,
- Slapi_DN *group_sdn, Slapi_Attr *attr)
- {
- return memberof_mod_attr_list_r(pb, config, mod, group_sdn, group_sdn,
- attr, 0);
- }
- int memberof_mod_attr_list_r(Slapi_PBlock *pb, MemberOfConfig *config, int mod,
- Slapi_DN *group_sdn, Slapi_DN *op_this_sdn,
- Slapi_Attr *attr, memberofstringll *stack)
- {
- int rc = 0;
- Slapi_Value *val = 0;
- Slapi_Value *op_this_val = 0;
- int last_size = 0;
- char *last_str = 0;
- int hint = slapi_attr_first_value(attr, &val);
- Slapi_DN *sdn = slapi_sdn_new();
- op_this_val = slapi_value_new_string(slapi_sdn_get_ndn(op_this_sdn));
- slapi_value_set_flags(op_this_val, SLAPI_ATTR_FLAG_NORMALIZED_CIS);
- while(val)
- {
- char *dn_str = 0;
- struct berval *bv = 0;
- /* We don't want to process a memberOf operation on ourselves. */
- if(0 != memberof_compare(config, &val, &op_this_val))
- {
- bv = (struct berval *)slapi_value_get_berval(val);
- if(last_size > bv->bv_len)
- {
- dn_str = last_str;
- }
- else
- {
- int the_size = (bv->bv_len * 2) + 1;
- if(last_str)
- slapi_ch_free_string(&last_str);
- dn_str = (char*)slapi_ch_malloc(the_size);
- last_str = dn_str;
- last_size = the_size;
- }
- memset(dn_str, 0, last_size);
- strncpy(dn_str, bv->bv_val, (size_t)bv->bv_len);
- /* If we're doing a replace (as we would in the MODRDN case), we need
- * to specify the new group DN value */
- slapi_sdn_set_normdn_byref(sdn, dn_str); /* dn_str is normalized */
- if(mod == LDAP_MOD_REPLACE)
- {
- memberof_modop_one_replace_r(pb, config, mod, group_sdn,
- op_this_sdn, group_sdn,
- sdn, stack);
- }
- else
- {
- memberof_modop_one_r(pb, config, mod, group_sdn,
- op_this_sdn, sdn, stack);
- }
- }
- hint = slapi_attr_next_value(attr, hint, &val);
- }
- slapi_value_free(&op_this_val);
- slapi_sdn_free(&sdn);
- if(last_str)
- slapi_ch_free_string(&last_str);
- return rc;
- }
- /*
- * memberof_add_attr_list()
- *
- * Add group DN to the memberof attribute of the list of targets
- *
- */
- int memberof_add_attr_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn, Slapi_Attr *attr)
- {
- return memberof_mod_attr_list(pb, config, LDAP_MOD_ADD, group_sdn, attr);
- }
- /*
- * memberof_del_attr_list()
- *
- * Remove group DN from the memberof attribute of the list of targets
- *
- */
- int memberof_del_attr_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn, Slapi_Attr *attr)
- {
- return memberof_mod_attr_list(pb, config, LDAP_MOD_DELETE, group_sdn, attr);
- }
- /*
- * memberof_moddn_attr_list()
- *
- * Perform mod for group DN to the memberof attribute of the list of targets
- *
- */
- int memberof_moddn_attr_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *pre_sdn, Slapi_DN *post_sdn, Slapi_Attr *attr)
- {
- int rc = 0;
- Slapi_Value *val = 0;
- int last_size = 0;
- char *last_str = 0;
- int hint = slapi_attr_first_value(attr, &val);
- Slapi_DN *sdn = slapi_sdn_new();
- while(val)
- {
- char *dn_str = 0;
- struct berval *bv = (struct berval *)slapi_value_get_berval(val);
- if(last_size > bv->bv_len)
- {
- dn_str = last_str;
- }
- else
- {
- int the_size = (bv->bv_len * 2) + 1;
- if(last_str)
- slapi_ch_free_string(&last_str);
- dn_str = (char*)slapi_ch_malloc(the_size);
- last_str = dn_str;
- last_size = the_size;
- }
- memset(dn_str, 0, last_size);
- strncpy(dn_str, bv->bv_val, (size_t)bv->bv_len);
- slapi_sdn_set_normdn_byref(sdn, dn_str); /* dn_str is normalized */
- memberof_modop_one_replace_r(pb, config, LDAP_MOD_REPLACE,
- post_sdn, pre_sdn, post_sdn, sdn, 0);
- hint = slapi_attr_next_value(attr, hint, &val);
- }
- slapi_sdn_free(&sdn);
- if(last_str)
- slapi_ch_free_string(&last_str);
- return rc;
- }
- /* memberof_get_groups()
- *
- * Gets a list of all groups that an entry is a member of.
- * This is done by looking only at member attribute values.
- * A Slapi_ValueSet* is returned. It is up to the caller to
- * free it.
- */
- Slapi_ValueSet *
- memberof_get_groups(MemberOfConfig *config, Slapi_DN *member_sdn)
- {
- Slapi_ValueSet *groupvals = slapi_valueset_new();
- Slapi_ValueSet *group_norm_vals = slapi_valueset_new();
- Slapi_Value *memberdn_val =
- slapi_value_new_string(slapi_sdn_get_ndn(member_sdn));
- slapi_value_set_flags(memberdn_val, SLAPI_ATTR_FLAG_NORMALIZED_CIS);
- memberof_get_groups_data data = {config, memberdn_val, &groupvals, &group_norm_vals};
- memberof_get_groups_r(config, member_sdn, &data);
- slapi_value_free(&memberdn_val);
- slapi_valueset_free(group_norm_vals);
- return groupvals;
- }
- int
- memberof_get_groups_r(MemberOfConfig *config, Slapi_DN *member_sdn,
- memberof_get_groups_data *data)
- {
- /* Search for any grouping attributes that point to memberdn.
- * For each match, add it to the list, recurse and do same search */
- return memberof_call_foreach_dn(NULL, member_sdn, config->groupattrs,
- memberof_get_groups_callback, data);
- }
- /* memberof_get_groups_callback()
- *
- * Callback to perform work of memberof_get_groups()
- */
- int memberof_get_groups_callback(Slapi_Entry *e, void *callback_data)
- {
- Slapi_DN *group_sdn = slapi_entry_get_sdn(e);
- char *group_ndn = slapi_entry_get_ndn(e);
- char *group_dn = slapi_entry_get_dn(e);
- Slapi_Value *group_ndn_val = 0;
- Slapi_Value *group_dn_val = 0;
- Slapi_ValueSet *groupvals = *((memberof_get_groups_data*)callback_data)->groupvals;
- Slapi_ValueSet *group_norm_vals = *((memberof_get_groups_data*)callback_data)->group_norm_vals;
- int rc = 0;
- if(slapi_is_shutting_down()){
- rc = -1;
- goto bail;
- }
- if (!groupvals)
- {
- slapi_log_error( SLAPI_LOG_PLUGIN, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_get_groups_callback: NULL groupvals\n");
- rc = -1;
- goto bail;
- }
- /* get the DN of the group */
- group_ndn_val = slapi_value_new_string(group_ndn);
- /* group_dn is case-normalized */
- slapi_value_set_flags(group_ndn_val, SLAPI_ATTR_FLAG_NORMALIZED_CIS);
- /* check if e is the same as our original member entry */
- if (0 == memberof_compare(((memberof_get_groups_data*)callback_data)->config,
- &((memberof_get_groups_data*)callback_data)->memberdn_val, &group_ndn_val))
- {
- /* A recursive group caused us to find our original
- * entry we passed to memberof_get_groups(). We just
- * skip processing this entry. */
- slapi_log_error( SLAPI_LOG_PLUGIN, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_get_groups_callback: group recursion"
- " detected in %s\n" ,group_ndn);
- slapi_value_free(&group_ndn_val);
- goto bail;
- }
- /* Have we been here before? Note that we don't loop through all of the group_slapiattrs
- * in config. We only need this attribute for it's syntax so the comparison can be
- * performed. Since all of the grouping attributes are validated to use the Dinstinguished
- * Name syntax, we can safely just use the first group_slapiattr. */
- if (group_norm_vals && slapi_valueset_find(
- ((memberof_get_groups_data*)callback_data)->config->group_slapiattrs[0], group_norm_vals, group_ndn_val))
- {
- /* we either hit a recursive grouping, or an entry is
- * a member of a group through multiple paths. Either
- * way, we can just skip processing this entry since we've
- * already gone through this part of the grouping hierarchy. */
- slapi_log_error( SLAPI_LOG_PLUGIN, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_get_groups_callback: possible group recursion"
- " detected in %s\n" ,group_ndn);
- slapi_value_free(&group_ndn_val);
- goto bail;
- }
- /* Push group_dn_val into the valueset. This memory is now owned
- * by the valueset. */
- group_dn_val = slapi_value_new_string(group_dn);
- slapi_valueset_add_value_ext(groupvals, group_dn_val, SLAPI_VALUE_FLAG_PASSIN);
- slapi_valueset_add_value_ext(group_norm_vals, group_ndn_val, SLAPI_VALUE_FLAG_PASSIN);
-
- /* now recurse to find parent groups of e */
- memberof_get_groups_r(((memberof_get_groups_data*)callback_data)->config,
- group_sdn, callback_data);
- bail:
- return rc;
- }
- /* memberof_is_direct_member()
- *
- * tests for direct membership of memberdn in group groupdn
- * returns non-zero when true, zero otherwise
- */
- int memberof_is_direct_member(MemberOfConfig *config, Slapi_Value *groupdn,
- Slapi_Value *memberdn)
- {
- int rc = 0;
- Slapi_DN *sdn = 0;
- Slapi_Entry *group_e = 0;
- Slapi_Attr *attr = 0;
- int i = 0;
- sdn = slapi_sdn_new_normdn_byref(slapi_value_get_string(groupdn));
- slapi_search_internal_get_entry(sdn, config->groupattrs,
- &group_e, memberof_get_plugin_id());
- if(group_e)
- {
- /* See if memberdn is referred to by any of the group attributes. */
- for (i = 0; config->groupattrs[i]; i++)
- {
- slapi_entry_attr_find(group_e, config->groupattrs[i], &attr );
- if(attr && (0 == slapi_attr_value_find(attr, slapi_value_get_berval(memberdn))))
- {
- rc = 1;
- break;
- }
- }
- slapi_entry_free(group_e);
- }
- slapi_sdn_free(&sdn);
- return rc;
- }
- /* memberof_is_grouping_attr()
- *
- * Checks if a supplied attribute is one of the configured
- * grouping attributes.
- *
- * Returns non-zero when true, zero otherwise.
- */
- static int memberof_is_grouping_attr(char *type, MemberOfConfig *config)
- {
- int match = 0;
- int i = 0;
- for (i = 0; config && config->groupattrs[i]; i++)
- {
- match = slapi_attr_types_equivalent(type, config->groupattrs[i]);
- if (match)
- {
- /* If we found a match, we're done. */
- break;
- }
- }
- return match;
- }
- /* memberof_test_membership()
- *
- * Finds all entries who are a "memberOf" the group
- * represented by "group_dn". For each matching entry, we
- * call memberof_test_membership_callback().
- *
- * for each attribute in the memberof attribute
- * determine if the entry is still a member.
- *
- * test each for direct membership
- * move groups entry is memberof to member group
- * test remaining groups for membership in member groups
- * iterate until a pass fails to move a group over to member groups
- * remaining groups should be deleted
- */
- int
- memberof_test_membership(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn)
- {
- char *attrs[2] = {config->memberof_attr, 0};
- return memberof_call_foreach_dn(pb, group_sdn, attrs,
- memberof_test_membership_callback , config);
- }
- /*
- * memberof_test_membership_callback()
- *
- * A callback function to do the work of memberof_test_membership().
- * Note that this not only tests membership, but updates the memberOf
- * attributes in the entry to be correct.
- */
- int memberof_test_membership_callback(Slapi_Entry *e, void *callback_data)
- {
- int rc = 0;
- Slapi_Attr *attr = 0;
- int total = 0;
- Slapi_Value **member_array = 0;
- Slapi_Value **candidate_array = 0;
- Slapi_Value *entry_dn = 0;
- Slapi_DN *entry_sdn = 0;
- MemberOfConfig *config = (MemberOfConfig *)callback_data;
- Slapi_DN *sdn = slapi_sdn_new();
- entry_sdn = slapi_entry_get_sdn(e);
- entry_dn = slapi_value_new_string(slapi_entry_get_ndn(e));
- if(entry_dn == NULL){
- goto bail;
- }
- slapi_value_set_flags(entry_dn, SLAPI_ATTR_FLAG_NORMALIZED_CIS);
- /* divide groups into member and non-member lists */
- slapi_entry_attr_find(e, config->memberof_attr, &attr );
- if(attr)
- {
- slapi_attr_get_numvalues( attr, &total);
- if(total)
- {
- Slapi_Value *val = 0;
- int hint = 0;
- int c_index = 0;
- int m_index = 0;
- int member_found = 1;
- int outer_index = 0;
- candidate_array =
- (Slapi_Value**)
- slapi_ch_malloc(sizeof(Slapi_Value*)*total);
- memset(candidate_array, 0, sizeof(Slapi_Value*)*total);
- member_array =
- (Slapi_Value**)
- slapi_ch_malloc(sizeof(Slapi_Value*)*total);
- memset(member_array, 0, sizeof(Slapi_Value*)*total);
- hint = slapi_attr_first_value(attr, &val);
- while(val)
- {
- /* test for direct membership */
- if(memberof_is_direct_member(config, val, entry_dn))
- {
- /* it is a member */
- member_array[m_index] = val;
- m_index++;
- }
- else
- {
- /* not a member, still a candidate */
- candidate_array[c_index] = val;
- c_index++;
- }
- hint = slapi_attr_next_value(attr, hint, &val);
- }
- /* now iterate over members testing for membership
- in candidate groups and moving candidates to members
- when successful, quit when a full iteration adds no
- new members
- */
- while(member_found)
- {
- member_found = 0;
- /* For each group that this entry is a verified member of, see if
- * any of the candidate groups are members. If they are, add them
- * to the list of verified groups that this entry is a member of.
- */
- while(outer_index < m_index)
- {
- int inner_index = 0;
- while(inner_index < c_index)
- {
- /* Check for a special value in this position
- * that indicates that the candidate was moved
- * to the member array. */
- if((void*)1 == candidate_array[inner_index])
- {
- /* was moved, skip */
- inner_index++;
- continue;
- }
- if(memberof_is_direct_member(
- config,
- candidate_array[inner_index],
- member_array[outer_index]))
- {
- member_array[m_index] =
- candidate_array
- [inner_index];
- m_index++;
- candidate_array[inner_index] =
- (void*)1;
-
- member_found = 1;
- }
- inner_index++;
- }
- outer_index++;
- }
- }
- /* here we are left only with values to delete
- from the memberof attribute in the candidate list
- */
- outer_index = 0;
- while(outer_index < c_index)
- {
- /* Check for a special value in this position
- * that indicates that the candidate was moved
- * to the member array. */
- if((void*)1 == candidate_array[outer_index])
- {
- /* item moved, skip */
- outer_index++;
- continue;
- }
- slapi_sdn_set_normdn_byref(sdn,
- slapi_value_get_string(candidate_array[outer_index]));
- memberof_del_one(0, config, sdn, entry_sdn);
- outer_index++;
- }
- {
- /* crazyness follows:
- * strict-aliasing doesn't like the required cast
- * to void for slapi_ch_free so we are made to
- * juggle to get a normal thing done
- */
- void *pmember_array = member_array;
- void *pcandidate_array = candidate_array;
- slapi_ch_free(&pcandidate_array);
- slapi_ch_free(&pmember_array);
- candidate_array = 0;
- member_array = 0;
- }
- }
- }
- bail:
- slapi_sdn_free(&sdn);
- slapi_value_free(&entry_dn);
- return rc;
- }
- /*
- * memberof_replace_list()
- *
- * Perform replace the group DN list in the memberof attribute of the list of targets
- *
- */
- int
- memberof_replace_list(Slapi_PBlock *pb, MemberOfConfig *config,
- Slapi_DN *group_sdn)
- {
- struct slapi_entry *pre_e = NULL;
- struct slapi_entry *post_e = NULL;
- Slapi_Attr *pre_attr = 0;
- Slapi_Attr *post_attr = 0;
- int i = 0;
- slapi_pblock_get( pb, SLAPI_ENTRY_PRE_OP, &pre_e );
- slapi_pblock_get( pb, SLAPI_ENTRY_POST_OP, &post_e );
-
- for (i = 0; config && config->groupattrs[i]; i++)
- {
- if(pre_e && post_e)
- {
- slapi_entry_attr_find( pre_e, config->groupattrs[i], &pre_attr );
- slapi_entry_attr_find( post_e, config->groupattrs[i], &post_attr );
- }
- if(pre_attr || post_attr)
- {
- int pre_total = 0;
- int post_total = 0;
- Slapi_Value **pre_array = 0;
- Slapi_Value **post_array = 0;
- int pre_index = 0;
- int post_index = 0;
- Slapi_DN *sdn = slapi_sdn_new();
- /* create arrays of values */
- if(pre_attr)
- {
- slapi_attr_get_numvalues( pre_attr, &pre_total);
- }
- if(post_attr)
- {
- slapi_attr_get_numvalues( post_attr, &post_total);
- }
- /* Stash a plugin global pointer here and have memberof_qsort_compare
- * use it. We have to do this because we use memberof_qsort_compare
- * as the comparator function for qsort, which requires the function
- * to only take two void* args. This is thread-safe since we only
- * store and use the pointer while holding the memberOf operation
- * lock. */
- qsortConfig = config;
- if(pre_total)
- {
- pre_array =
- (Slapi_Value**)
- slapi_ch_malloc(sizeof(Slapi_Value*)*pre_total);
- memberof_load_array(pre_array, pre_attr);
- qsort(
- pre_array,
- pre_total,
- sizeof(Slapi_Value*),
- memberof_qsort_compare);
- }
- if(post_total)
- {
- post_array =
- (Slapi_Value**)
- slapi_ch_malloc(sizeof(Slapi_Value*)*post_total);
- memberof_load_array(post_array, post_attr);
- qsort(
- post_array,
- post_total,
- sizeof(Slapi_Value*),
- memberof_qsort_compare);
- }
- qsortConfig = 0;
- /* work through arrays, following these rules:
- in pre, in post, do nothing
- in pre, not in post, delete from entry
- not in pre, in post, add to entry
- */
- while(pre_index < pre_total || post_index < post_total)
- {
- if(pre_index == pre_total)
- {
- /* add the rest of post */
- slapi_sdn_set_normdn_byref(sdn,
- slapi_value_get_string(post_array[post_index]));
- memberof_add_one(pb, config, group_sdn, sdn);
- post_index++;
- }
- else if(post_index == post_total)
- {
- /* delete the rest of pre */
- slapi_sdn_set_normdn_byref(sdn,
- slapi_value_get_string(pre_array[pre_index]));
- memberof_del_one(pb, config, group_sdn, sdn);
- pre_index++;
- }
- else
- {
- /* decide what to do */
- int cmp = memberof_compare(
- config,
- &(pre_array[pre_index]),
- &(post_array[post_index]));
- if(cmp < 0)
- {
- /* delete pre array */
- slapi_sdn_set_normdn_byref(sdn,
- slapi_value_get_string(pre_array[pre_index]));
- memberof_del_one(pb, config, group_sdn, sdn);
- pre_index++;
- }
- else if(cmp > 0)
- {
- /* add post array */
- slapi_sdn_set_normdn_byref(sdn,
- slapi_value_get_string(post_array[post_index]));
- memberof_add_one(pb, config, group_sdn, sdn);
- post_index++;
- }
- else
- {
- /* do nothing, advance */
- pre_index++;
- post_index++;
- }
- }
- }
- slapi_sdn_free(&sdn);
- slapi_ch_free((void **)&pre_array);
- slapi_ch_free((void **)&post_array);
- }
- }
-
- return 0;
- }
- /* memberof_load_array()
- *
- * put attribute values in array structure
- */
- void memberof_load_array(Slapi_Value **array, Slapi_Attr *attr)
- {
- Slapi_Value *val = 0;
- int hint = slapi_attr_first_value(attr, &val);
- while(val)
- {
- *array = val;
- array++;
- hint = slapi_attr_next_value(attr, hint, &val);
- }
- }
- /* memberof_compare()
- *
- * compare two attr values
- */
- int memberof_compare(MemberOfConfig *config, const void *a, const void *b)
- {
- Slapi_Value *val1;
- Slapi_Value *val2;
- if(a == NULL && b != NULL){
- return 1;
- } else if(a != NULL && b == NULL){
- return -1;
- } else if(a == NULL && b == NULL){
- return 0;
- }
- val1 = *((Slapi_Value **)a);
- val2 = *((Slapi_Value **)b);
- /* We only need to provide a Slapi_Attr here for it's syntax. We
- * already validated all grouping attributes to use the Distinguished
- * Name syntax, so we can safely just use the first attr. */
- return slapi_attr_value_cmp_ext(config->group_slapiattrs[0], val1, val2);
- }
- /* memberof_qsort_compare()
- *
- * This is a version of memberof_compare that uses a plugin
- * global copy of the config. We'd prefer to pass in a copy
- * of config that is local to the running thread, but we can't
- * do this since qsort is using us as a comparator function.
- * We should only use this function when using qsort, and only
- * when the memberOf lock is acquired.
- */
- int memberof_qsort_compare(const void *a, const void *b)
- {
- Slapi_Value *val1 = *((Slapi_Value **)a);
- Slapi_Value *val2 = *((Slapi_Value **)b);
- /* We only need to provide a Slapi_Attr here for it's syntax. We
- * already validated all grouping attributes to use the Distinguished
- * Name syntax, so we can safely just use the first attr. */
- return slapi_attr_value_cmp_ext(qsortConfig->group_slapiattrs[0],
- val1, val2);
- }
- /* betxn: This locking mechanism is necessary to guarantee the memberof
- * consistency */
- void memberof_lock()
- {
- if (usetxn) {
- PR_EnterMonitor(memberof_operation_lock);
- }
- }
- void memberof_unlock()
- {
- if (usetxn) {
- PR_ExitMonitor(memberof_operation_lock);
- }
- }
- typedef struct _task_data
- {
- char *dn;
- char *bind_dn;
- char *filter_str;
- } task_data;
- void memberof_fixup_task_thread(void *arg)
- {
- MemberOfConfig configCopy = {0, 0, 0, 0};
- Slapi_Task *task = (Slapi_Task *)arg;
- task_data *td = NULL;
- int rc = 0;
- Slapi_PBlock *fixup_pb = NULL;
- /* Fetch our task data from the task */
- td = (task_data *)slapi_task_get_data(task);
- /* set bind DN in the thread data */
- slapi_td_set_dn(slapi_ch_strdup(td->bind_dn));
- slapi_task_begin(task, 1);
- slapi_task_log_notice(task, "Memberof task starts (arg: %s) ...\n",
- td->filter_str);
- slapi_log_error(SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "Memberof task starts (arg: %s) ...\n", td->filter_str);
- /* We need to get the config lock first. Trying to get the
- * config lock after we already hold the op lock can cause
- * a deadlock. */
- memberof_rlock_config();
- /* copy config so it doesn't change out from under us */
- memberof_copy_config(&configCopy, memberof_get_config());
- memberof_unlock_config();
- if (usetxn) {
- Slapi_DN *sdn = slapi_sdn_new_dn_byref(td->dn);
- Slapi_Backend *be = slapi_be_select(sdn);
- slapi_sdn_free(&sdn);
- if (be) {
- fixup_pb = slapi_pblock_new();
- slapi_pblock_set(fixup_pb, SLAPI_BACKEND, be);
- rc = slapi_back_transaction_begin(fixup_pb);
- if (rc) {
- slapi_log_error(SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_fixup_task_thread: failed to start transaction\n");
- }
- } else {
- slapi_log_error(SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "memberof_fixup_task_thread: failed to get be backend from %s\n",
- td->dn);
- }
- }
- /* get the memberOf operation lock */
- memberof_lock();
- /* do real work */
- rc = memberof_fix_memberof(&configCopy, td->dn, td->filter_str);
-
- /* release the memberOf operation lock */
- memberof_unlock();
- if (usetxn && fixup_pb) {
- if (rc) { /* failes */
- slapi_back_transaction_abort(fixup_pb);
- } else {
- slapi_back_transaction_commit(fixup_pb);
- }
- slapi_pblock_destroy(fixup_pb);
- }
- memberof_free_config(&configCopy);
- slapi_task_log_notice(task, "Memberof task finished.");
- slapi_task_log_status(task, "Memberof task finished.");
- slapi_task_inc_progress(task);
- slapi_log_error(SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "Memberof task finished (arg: %s) ...\n", td->filter_str);
- /* this will queue the destruction of the task */
- slapi_task_finish(task, rc);
- }
- /* extract a single value from the entry (as a string) -- if it's not in the
- * entry, the default will be returned (which can be NULL).
- * you do not need to free anything returned by this.
- */
- const char *fetch_attr(Slapi_Entry *e, const char *attrname,
- const char *default_val)
- {
- Slapi_Attr *attr;
- Slapi_Value *val = NULL;
- if (slapi_entry_attr_find(e, attrname, &attr) != 0)
- return default_val;
- slapi_attr_first_value(attr, &val);
- return slapi_value_get_string(val);
- }
- int memberof_task_add(Slapi_PBlock *pb, Slapi_Entry *e,
- Slapi_Entry *eAfter, int *returncode, char *returntext,
- void *arg)
- {
- PRThread *thread = NULL;
- int rv = SLAPI_DSE_CALLBACK_OK;
- task_data *mytaskdata = NULL;
- Slapi_Task *task = NULL;
- char *bind_dn;
- const char *filter;
- const char *dn = 0;
- *returncode = LDAP_SUCCESS;
- /* make sure the plugin was not stopped from a shutdown */
- if (!g_plugin_started)
- {
- *returncode = LDAP_OPERATIONS_ERROR;
- rv = SLAPI_DSE_CALLBACK_ERROR;
- goto out;
- }
- /* get arg(s) */
- if ((dn = fetch_attr(e, "basedn", 0)) == NULL)
- {
- *returncode = LDAP_OBJECT_CLASS_VIOLATION;
- rv = SLAPI_DSE_CALLBACK_ERROR;
- goto out;
- }
- if ((filter = fetch_attr(e, "filter", "(|(objectclass=inetuser)(objectclass=inetadmin))")) == NULL)
- {
- *returncode = LDAP_OBJECT_CLASS_VIOLATION;
- rv = SLAPI_DSE_CALLBACK_ERROR;
- goto out;
- }
- /* setup our task data */
- slapi_pblock_get(pb, SLAPI_REQUESTOR_DN, &bind_dn);
- mytaskdata = (task_data*)slapi_ch_malloc(sizeof(task_data));
- if (mytaskdata == NULL)
- {
- *returncode = LDAP_OPERATIONS_ERROR;
- rv = SLAPI_DSE_CALLBACK_ERROR;
- goto out;
- }
- mytaskdata->dn = slapi_ch_strdup(dn);
- mytaskdata->filter_str = slapi_ch_strdup(filter);
- mytaskdata->bind_dn = slapi_ch_strdup(bind_dn);
- /* allocate new task now */
- task = slapi_new_task(slapi_entry_get_ndn(e));
- /* register our destructor for cleaning up our private data */
- slapi_task_set_destructor_fn(task, memberof_task_destructor);
- /* Stash a pointer to our data in the task */
- slapi_task_set_data(task, mytaskdata);
- /* start the sample task as a separate thread */
- thread = PR_CreateThread(PR_USER_THREAD, memberof_fixup_task_thread,
- (void *)task, PR_PRIORITY_NORMAL, PR_GLOBAL_THREAD,
- PR_UNJOINABLE_THREAD, SLAPD_DEFAULT_THREAD_STACKSIZE);
- if (thread == NULL)
- {
- slapi_log_error( SLAPI_LOG_FATAL, MEMBEROF_PLUGIN_SUBSYSTEM,
- "unable to create task thread!\n");
- *returncode = LDAP_OPERATIONS_ERROR;
- rv = SLAPI_DSE_CALLBACK_ERROR;
- slapi_task_finish(task, *returncode);
- } else {
- rv = SLAPI_DSE_CALLBACK_OK;
- }
- out:
- return rv;
- }
- void
- memberof_task_destructor(Slapi_Task *task)
- {
- if (task) {
- task_data *mydata = (task_data *)slapi_task_get_data(task);
- if (mydata) {
- slapi_ch_free_string(&mydata->dn);
- slapi_ch_free_string(&mydata->bind_dn);
- slapi_ch_free_string(&mydata->filter_str);
- /* Need to cast to avoid a compiler warning */
- slapi_ch_free((void **)&mydata);
- }
- }
- }
- int memberof_fix_memberof(MemberOfConfig *config, char *dn, char *filter_str)
- {
- int rc = 0;
- Slapi_PBlock *search_pb = slapi_pblock_new();
- slapi_search_internal_set_pb(search_pb, dn,
- LDAP_SCOPE_SUBTREE, filter_str, 0, 0,
- 0, 0,
- memberof_get_plugin_id(),
- 0);
- rc = slapi_search_internal_callback_pb(search_pb,
- config,
- 0, memberof_fix_memberof_callback,
- 0);
- slapi_pblock_destroy(search_pb);
- return rc;
- }
- /* memberof_fix_memberof_callback()
- * Add initial and/or fix up broken group list in entry
- *
- * 1. Remove all present memberOf values
- * 2. Add direct group membership memberOf values
- * 3. Add indirect group membership memberOf values
- */
- int memberof_fix_memberof_callback(Slapi_Entry *e, void *callback_data)
- {
- int rc = 0;
- Slapi_DN *sdn = slapi_entry_get_sdn(e);
- MemberOfConfig *config = (MemberOfConfig *)callback_data;
- memberof_del_dn_data del_data = {0, config->memberof_attr};
- Slapi_ValueSet *groups = 0;
- /* get a list of all of the groups this user belongs to */
- groups = memberof_get_groups(config, sdn);
- /* If we found some groups, replace the existing memberOf attribute
- * with the found values. */
- if (groups && slapi_valueset_count(groups))
- {
- Slapi_PBlock *mod_pb = slapi_pblock_new();
- Slapi_Value *val = 0;
- Slapi_Mod *smod;
- LDAPMod **mods = (LDAPMod **) slapi_ch_malloc(2 * sizeof(LDAPMod *));
- int hint = 0;
- smod = slapi_mod_new();
- slapi_mod_init(smod, 0);
- slapi_mod_set_operation(smod, LDAP_MOD_REPLACE | LDAP_MOD_BVALUES);
- slapi_mod_set_type(smod, config->memberof_attr);
- /* Loop through all of our values and add them to smod */
- hint = slapi_valueset_first_value(groups, &val);
- while (val)
- {
- /* this makes a copy of the berval */
- slapi_mod_add_value(smod, slapi_value_get_berval(val));
- hint = slapi_valueset_next_value(groups, hint, &val);
- }
-
- mods[0] = slapi_mod_get_ldapmod_passout(smod);
- mods[1] = 0;
- slapi_modify_internal_set_pb_ext(
- mod_pb, sdn, mods, 0, 0,
- memberof_get_plugin_id(), 0);
- slapi_modify_internal_pb(mod_pb);
- slapi_pblock_get(mod_pb, SLAPI_PLUGIN_INTOP_RESULT, &rc);
- ldap_mods_free(mods, 1);
- slapi_mod_free(&smod);
- slapi_pblock_destroy(mod_pb);
- } else {
- /* No groups were found, so remove the memberOf attribute
- * from this entry. */
- memberof_del_dn_type_callback(e, &del_data);
- }
- slapi_valueset_free(groups);
-
- return rc;
- }
|