Line data Source code
1 : /*
2 : SSSD
3 :
4 : Async LDAP Helper routines for netgroups
5 :
6 : Authors:
7 : Sumit Bose <sbose@redhat.com>
8 :
9 : Copyright (C) 2010 Red Hat
10 :
11 : This program is free software; you can redistribute it and/or modify
12 : it under the terms of the GNU General Public License as published by
13 : the Free Software Foundation; either version 3 of the License, or
14 : (at your option) any later version.
15 :
16 : This program is distributed in the hope that it will be useful,
17 : but WITHOUT ANY WARRANTY; without even the implied warranty of
18 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 : GNU General Public License for more details.
20 :
21 : You should have received a copy of the GNU General Public License
22 : along with this program. If not, see <http://www.gnu.org/licenses/>.
23 : */
24 :
25 : #include "util/util.h"
26 : #include "db/sysdb.h"
27 : #include "providers/ldap/sdap_async_private.h"
28 : #include "providers/ldap/ldap_common.h"
29 :
30 0 : bool is_dn(const char *str)
31 : {
32 : int ret;
33 : LDAPDN dn;
34 :
35 0 : ret = ldap_str2dn(str, &dn, LDAP_DN_FORMAT_LDAPV3);
36 0 : ldap_dnfree(dn);
37 :
38 0 : return (ret == LDAP_SUCCESS ? true : false);
39 : }
40 :
41 0 : static errno_t sdap_save_netgroup(TALLOC_CTX *memctx,
42 : struct sss_domain_info *dom,
43 : struct sdap_options *opts,
44 : struct sysdb_attrs *attrs,
45 : char **_timestamp,
46 : time_t now)
47 : {
48 : struct ldb_message_element *el;
49 : struct sysdb_attrs *netgroup_attrs;
50 0 : const char *name = NULL;
51 : int ret;
52 0 : char *timestamp = NULL;
53 0 : char **missing = NULL;
54 :
55 0 : ret = sdap_get_netgroup_primary_name(memctx, opts, attrs, dom, &name);
56 0 : if (ret != EOK) {
57 0 : DEBUG(SSSDBG_OP_FAILURE, "Failed to get netgroup name\n");
58 0 : goto fail;
59 : }
60 :
61 0 : DEBUG(SSSDBG_TRACE_FUNC, "Processing netgroup %s\n", name);
62 :
63 0 : netgroup_attrs = sysdb_new_attrs(memctx);
64 0 : if (!netgroup_attrs) {
65 0 : ret = ENOMEM;
66 0 : goto fail;
67 : }
68 :
69 0 : ret = sdap_attrs_add_string(attrs, SYSDB_ORIG_DN,
70 : "original DN",
71 : name, netgroup_attrs);
72 0 : if (ret != EOK) {
73 0 : goto fail;
74 : }
75 :
76 0 : ret = sysdb_attrs_get_el(attrs,
77 0 : opts->netgroup_map[SDAP_AT_NETGROUP_MODSTAMP].sys_name,
78 : &el);
79 0 : if (ret) {
80 0 : goto fail;
81 : }
82 0 : if (el->num_values == 0) {
83 0 : DEBUG(SSSDBG_TRACE_LIBS,
84 : "Original mod-Timestamp is not available for [%s].\n",
85 : name);
86 : } else {
87 0 : ret = sysdb_attrs_add_string(netgroup_attrs,
88 0 : opts->netgroup_map[SDAP_AT_NETGROUP_MODSTAMP].sys_name,
89 0 : (const char*)el->values[0].data);
90 0 : if (ret) {
91 0 : goto fail;
92 : }
93 0 : timestamp = talloc_strdup(memctx, (const char*)el->values[0].data);
94 0 : if (!timestamp) {
95 0 : ret = ENOMEM;
96 0 : goto fail;
97 : }
98 : }
99 :
100 0 : ret = sdap_attrs_add_list(attrs,
101 : opts->netgroup_map[SDAP_AT_NETGROUP_TRIPLE].sys_name,
102 : "netgroup triple",
103 : name, netgroup_attrs);
104 0 : if (ret != EOK) {
105 0 : goto fail;
106 : }
107 :
108 0 : ret = sdap_attrs_add_list(attrs,
109 : opts->netgroup_map[SDAP_AT_NETGROUP_MEMBER].sys_name,
110 : "original members",
111 : name, netgroup_attrs);
112 0 : if (ret != EOK) {
113 0 : goto fail;
114 : }
115 :
116 0 : ret = sdap_attrs_add_list(attrs, SYSDB_NETGROUP_MEMBER,
117 : "members", name, netgroup_attrs);
118 0 : if (ret != EOK) {
119 0 : goto fail;
120 : }
121 :
122 0 : DEBUG(SSSDBG_TRACE_FUNC, "Storing info for netgroup %s\n", name);
123 :
124 0 : ret = sdap_save_all_names(name, attrs, dom,
125 : netgroup_attrs);
126 0 : if (ret != EOK) {
127 0 : DEBUG(SSSDBG_CRIT_FAILURE, "Failed to save netgroup names\n");
128 0 : goto fail;
129 : }
130 :
131 : /* Make sure that any attributes we requested from LDAP that we
132 : * did not receive are also removed from the sysdb
133 : */
134 0 : ret = list_missing_attrs(attrs, opts->netgroup_map, SDAP_OPTS_NETGROUP,
135 : attrs, &missing);
136 0 : if (ret != EOK) {
137 0 : DEBUG(SSSDBG_CRIT_FAILURE, "Failed to list missing attributes\n");
138 0 : goto fail;
139 : }
140 :
141 0 : ret = sysdb_add_netgroup(dom, name, NULL, netgroup_attrs, missing,
142 0 : dom->netgroup_timeout, now);
143 0 : if (ret) goto fail;
144 :
145 0 : if (_timestamp) {
146 0 : *_timestamp = timestamp;
147 : }
148 :
149 0 : return EOK;
150 :
151 : fail:
152 0 : DEBUG(SSSDBG_OP_FAILURE, "Failed to save netgroup %s\n", name);
153 0 : return ret;
154 : }
155 :
156 0 : errno_t update_dn_list(struct dn_item *dn_list, const size_t count,
157 : struct ldb_message **res, bool *all_resolved)
158 : {
159 : struct dn_item *dn_item;
160 : size_t c;
161 : const char *dn;
162 : const char *cn;
163 0 : bool not_resolved = false;
164 :
165 0 : *all_resolved = false;
166 :
167 0 : DLIST_FOR_EACH(dn_item, dn_list) {
168 0 : if (dn_item->cn != NULL) {
169 0 : continue;
170 : }
171 :
172 0 : for(c = 0; c < count; c++) {
173 0 : dn = ldb_msg_find_attr_as_string(res[c], SYSDB_ORIG_DN, NULL);
174 0 : if (dn == NULL) {
175 0 : DEBUG(SSSDBG_CRIT_FAILURE, "Missing original DN.\n");
176 0 : return EINVAL;
177 : }
178 0 : if (strcmp(dn, dn_item->dn) == 0) {
179 0 : DEBUG(SSSDBG_TRACE_ALL,
180 : "Found matching entry for [%s].\n", dn_item->dn);
181 0 : cn = ldb_msg_find_attr_as_string(res[c], SYSDB_NAME, NULL);
182 0 : if (cn == NULL) {
183 0 : DEBUG(SSSDBG_CRIT_FAILURE, "Missing name.\n");
184 0 : return EINVAL;
185 : }
186 0 : dn_item->cn = talloc_strdup(dn_item, cn);
187 0 : break;
188 : }
189 : }
190 :
191 0 : if (dn_item->cn == NULL) {
192 0 : not_resolved = true;
193 : }
194 : }
195 :
196 0 : *all_resolved = !not_resolved;
197 :
198 0 : return EOK;
199 : }
200 :
201 : struct netgr_translate_members_state {
202 : struct tevent_context *ev;
203 : struct sdap_options *opts;
204 : struct sdap_handle *sh;
205 :
206 : struct sysdb_attrs **netgroups;
207 : size_t count;
208 : struct dn_item *dn_list;
209 : struct dn_item *dn_item;
210 : struct dn_item *dn_idx;
211 : };
212 :
213 : static errno_t netgr_translate_members_ldap_step(struct tevent_req *req);
214 : static void netgr_translate_members_ldap_done(struct tevent_req *subreq);
215 :
216 0 : struct tevent_req *netgr_translate_members_send(TALLOC_CTX *memctx,
217 : struct tevent_context *ev,
218 : struct sdap_options *opts,
219 : struct sdap_handle *sh,
220 : struct sss_domain_info *dom,
221 : struct sysdb_ctx *sysdb,
222 : const size_t count,
223 : struct sysdb_attrs **netgroups)
224 : {
225 : struct tevent_req *req;
226 : struct netgr_translate_members_state *state;
227 : size_t c;
228 : size_t mc;
229 : const char **member_list;
230 : size_t sysdb_count;
231 : int ret;
232 : struct ldb_message **sysdb_res;
233 : struct dn_item *dn_item;
234 : char *dn_filter;
235 : char *sysdb_filter;
236 : struct ldb_dn *netgr_basedn;
237 : bool all_resolved;
238 0 : const char *cn_attr[] = { SYSDB_NAME, SYSDB_ORIG_DN, NULL };
239 :
240 0 : req = tevent_req_create(memctx, &state,
241 : struct netgr_translate_members_state);
242 0 : if (req == NULL) {
243 0 : return NULL;
244 : }
245 :
246 0 : state->ev = ev;
247 0 : state->opts = opts;
248 0 : state->sh = sh;
249 0 : state->netgroups = netgroups;
250 0 : state->count = count;
251 0 : state->dn_list = NULL;
252 0 : state->dn_item = NULL;
253 0 : state->dn_idx = NULL;
254 :
255 0 : for (c = 0; c < count; c++) {
256 0 : ret = sysdb_attrs_get_string_array(netgroups[c],
257 : SYSDB_ORIG_NETGROUP_MEMBER, state,
258 : &member_list);
259 0 : if (ret != EOK) {
260 0 : DEBUG(SSSDBG_TRACE_LIBS, "Missing netgroup members.\n");
261 0 : continue;
262 : }
263 :
264 0 : for (mc = 0; member_list[mc] != NULL; mc++) {
265 0 : if (is_dn(member_list[mc])) {
266 0 : dn_item = talloc_zero(state, struct dn_item);
267 0 : if (dn_item == NULL) {
268 0 : DEBUG(SSSDBG_CRIT_FAILURE, "talloc failed.\n");
269 0 : ret = ENOMEM;
270 0 : goto fail;
271 : }
272 :
273 0 : DEBUG(SSSDBG_TRACE_ALL,
274 : "Adding [%s] to DN list.\n", member_list[mc]);
275 0 : dn_item->netgroup = netgroups[c];
276 0 : dn_item->dn = member_list[mc];
277 0 : DLIST_ADD(state->dn_list, dn_item);
278 : } else {
279 0 : ret = sysdb_attrs_add_string(netgroups[c], SYSDB_NETGROUP_MEMBER,
280 0 : member_list[mc]);
281 0 : if (ret != EOK) {
282 0 : DEBUG(SSSDBG_CRIT_FAILURE,
283 : "sysdb_attrs_add_string failed.\n");
284 0 : goto fail;
285 : }
286 : }
287 : }
288 : }
289 :
290 0 : if (state->dn_list == NULL) {
291 0 : DEBUG(SSSDBG_TRACE_ALL, "No DNs found among netgroup members.\n");
292 0 : tevent_req_done(req);
293 0 : tevent_req_post(req, ev);
294 0 : return req;
295 : }
296 :
297 0 : dn_filter = talloc_strdup(state, "(|");
298 0 : if (dn_filter == NULL) {
299 0 : DEBUG(SSSDBG_CRIT_FAILURE, "talloc_strdup failed.\n");
300 0 : ret = ENOMEM;;
301 0 : goto fail;
302 : }
303 :
304 0 : DLIST_FOR_EACH(dn_item, state->dn_list) {
305 0 : dn_filter = talloc_asprintf_append(dn_filter, "(%s=%s)",
306 : SYSDB_ORIG_DN, dn_item->dn);
307 0 : if (dn_filter == NULL) {
308 0 : DEBUG(SSSDBG_CRIT_FAILURE, "talloc_asprintf_append failed.\n");
309 0 : ret = ENOMEM;
310 0 : goto fail;
311 : }
312 : }
313 :
314 0 : dn_filter = talloc_asprintf_append(dn_filter, ")");
315 0 : if (dn_filter == NULL) {
316 0 : DEBUG(SSSDBG_CRIT_FAILURE, "talloc_asprintf_append failed.\n");
317 0 : ret = ENOMEM;
318 0 : goto fail;
319 : }
320 :
321 0 : sysdb_filter = talloc_asprintf(state, "(&(%s)%s)", SYSDB_NC, dn_filter);
322 0 : if (sysdb_filter == NULL) {
323 0 : DEBUG(SSSDBG_CRIT_FAILURE, "talloc_asprintf failed.\n");
324 0 : ret = ENOMEM;
325 0 : goto fail;
326 : }
327 :
328 0 : netgr_basedn = sysdb_netgroup_base_dn(state, dom);
329 0 : if (netgr_basedn == NULL) {
330 0 : ret = ENOMEM;
331 0 : goto fail;
332 : }
333 :
334 0 : ret = sysdb_search_entry(state, sysdb, netgr_basedn, LDB_SCOPE_BASE,
335 : sysdb_filter, cn_attr, &sysdb_count, &sysdb_res);
336 0 : talloc_zfree(netgr_basedn);
337 0 : talloc_zfree(sysdb_filter);
338 0 : if (ret != EOK && ret != ENOENT) {
339 0 : DEBUG(SSSDBG_CRIT_FAILURE, "sysdb_search_entry failed.\n");
340 0 : goto fail;
341 : }
342 :
343 0 : if (ret == EOK) {
344 0 : ret = update_dn_list(state->dn_list, sysdb_count, sysdb_res,
345 : &all_resolved);
346 0 : if (ret != EOK) {
347 0 : DEBUG(SSSDBG_CRIT_FAILURE, "update_dn_list failed.\n");
348 0 : goto fail;
349 : }
350 :
351 0 : if (all_resolved) {
352 0 : DLIST_FOR_EACH(dn_item, state->dn_list) {
353 0 : ret = sysdb_attrs_add_string(dn_item->netgroup,
354 : SYSDB_NETGROUP_MEMBER,
355 0 : dn_item->cn);
356 0 : if (ret != EOK) {
357 0 : DEBUG(SSSDBG_CRIT_FAILURE,
358 : "sysdb_attrs_add_string failed.\n");
359 0 : goto fail;
360 : }
361 : }
362 :
363 0 : tevent_req_done(req);
364 0 : tevent_req_post(req, ev);
365 0 : return req;
366 : }
367 : }
368 :
369 0 : state->dn_idx = state->dn_list;
370 0 : ret = netgr_translate_members_ldap_step(req);
371 0 : if (ret != EOK && ret != EAGAIN) {
372 0 : DEBUG(SSSDBG_CRIT_FAILURE,
373 : "netgr_translate_members_ldap_step failed.\n");
374 0 : goto fail;
375 : }
376 :
377 0 : if (ret == EOK) {
378 0 : tevent_req_done(req);
379 0 : tevent_req_post(req, ev);
380 : }
381 0 : return req;
382 :
383 : fail:
384 0 : tevent_req_error(req, ret);
385 0 : tevent_req_post(req, ev);
386 0 : return req;
387 : }
388 :
389 : /* netgr_translate_members_ldap_step() returns
390 : * EOK: if everthing is translated, the caller can call tevent_req_done
391 : * EAGAIN: if there are still members waiting to be translated, the caller
392 : * should return to the mainloop
393 : * Exyz: every other return code indicates an error and tevent_req_error
394 : * should be called
395 : */
396 0 : static errno_t netgr_translate_members_ldap_step(struct tevent_req *req)
397 : {
398 0 : struct netgr_translate_members_state *state = tevent_req_data(req,
399 : struct netgr_translate_members_state);
400 : const char **cn_attr;
401 0 : char *filter = NULL;
402 : struct tevent_req *subreq;
403 : int ret;
404 :
405 0 : DLIST_FOR_EACH(state->dn_item, state->dn_idx) {
406 0 : if (state->dn_item->cn == NULL) {
407 0 : break;
408 : }
409 : }
410 0 : if (state->dn_item == NULL) {
411 0 : DLIST_FOR_EACH(state->dn_item, state->dn_list) {
412 0 : ret = sysdb_attrs_add_string(state->dn_item->netgroup,
413 : SYSDB_NETGROUP_MEMBER,
414 0 : state->dn_item->cn);
415 0 : if (ret != EOK) {
416 0 : DEBUG(SSSDBG_CRIT_FAILURE,
417 : "sysdb_attrs_add_string failed.\n");
418 0 : tevent_req_error(req, ret);
419 0 : return ret;
420 : }
421 : }
422 :
423 0 : return EOK;
424 : }
425 :
426 0 : if (!sss_ldap_dn_in_search_bases(state, state->dn_item->dn,
427 0 : state->opts->sdom->netgroup_search_bases,
428 : &filter)) {
429 : /* not in search base, skip it */
430 0 : state->dn_idx = state->dn_item->next;
431 0 : DLIST_REMOVE(state->dn_list, state->dn_item);
432 0 : return netgr_translate_members_ldap_step(req);
433 : }
434 :
435 0 : cn_attr = talloc_array(state, const char *, 3);
436 0 : if (cn_attr == NULL) {
437 0 : DEBUG(SSSDBG_CRIT_FAILURE, "talloc_array failed.\n");
438 0 : return ENOMEM;
439 : }
440 0 : cn_attr[0] = state->opts->netgroup_map[SDAP_AT_NETGROUP_NAME].name;
441 0 : cn_attr[1] = "objectclass";
442 0 : cn_attr[2] = NULL;
443 :
444 0 : DEBUG(SSSDBG_TRACE_ALL, "LDAP base search for [%s].\n", state->dn_item->dn);
445 0 : subreq = sdap_get_generic_send(state, state->ev, state->opts, state->sh,
446 0 : state->dn_item->dn, LDAP_SCOPE_BASE, filter,
447 0 : cn_attr, state->opts->netgroup_map,
448 : SDAP_OPTS_NETGROUP,
449 0 : dp_opt_get_int(state->opts->basic,
450 : SDAP_SEARCH_TIMEOUT),
451 : false);
452 0 : if (!subreq) {
453 0 : DEBUG(SSSDBG_CRIT_FAILURE, "sdap_get_generic_send failed.\n");
454 0 : return ENOMEM;
455 : }
456 0 : talloc_steal(subreq, cn_attr);
457 :
458 0 : tevent_req_set_callback(subreq, netgr_translate_members_ldap_done, req);
459 0 : return EAGAIN;
460 : }
461 :
462 0 : static void netgr_translate_members_ldap_done(struct tevent_req *subreq)
463 : {
464 0 : struct tevent_req *req = tevent_req_callback_data(subreq,
465 : struct tevent_req);
466 0 : struct netgr_translate_members_state *state = tevent_req_data(req,
467 : struct netgr_translate_members_state);
468 : int ret;
469 : size_t count;
470 : struct sysdb_attrs **netgroups;
471 : const char *str;
472 :
473 0 : ret = sdap_get_generic_recv(subreq, state, &count, &netgroups);
474 0 : talloc_zfree(subreq);
475 0 : if (ret != EOK) {
476 0 : DEBUG(SSSDBG_CRIT_FAILURE, "sdap_get_generic request failed.\n");
477 0 : goto fail;
478 : }
479 :
480 0 : switch (count) {
481 : case 0:
482 0 : DEBUG(SSSDBG_FATAL_FAILURE,
483 : "sdap_get_generic_recv found no entry for [%s].\n",
484 : state->dn_item->dn);
485 0 : break;
486 : case 1:
487 0 : ret = sysdb_attrs_get_string(netgroups[0], SYSDB_NAME, &str);
488 0 : if (ret != EOK) {
489 0 : DEBUG(SSSDBG_CRIT_FAILURE, "sysdb_attrs_add_string failed.\n");
490 0 : break;
491 : }
492 0 : state->dn_item->cn = talloc_strdup(state->dn_item, str);
493 0 : if (state->dn_item->cn == NULL) {
494 0 : DEBUG(SSSDBG_CRIT_FAILURE, "talloc_strdup failed.\n");
495 : }
496 0 : break;
497 : default:
498 0 : DEBUG(SSSDBG_CRIT_FAILURE,
499 : "Unexpected number of results [%zu] for base search.\n",
500 : count);
501 : }
502 :
503 0 : if (state->dn_item->cn == NULL) {
504 0 : DEBUG(SSSDBG_CRIT_FAILURE,
505 : "Failed to resolve netgroup name for DN [%s], using DN.\n",
506 : state->dn_item->dn);
507 0 : state->dn_item->cn = talloc_strdup(state->dn_item, state->dn_item->dn);
508 : }
509 :
510 0 : state->dn_idx = state->dn_item->next;
511 0 : ret = netgr_translate_members_ldap_step(req);
512 0 : if (ret != EOK && ret != EAGAIN) {
513 0 : DEBUG(SSSDBG_CRIT_FAILURE,
514 : "netgr_translate_members_ldap_step failed.\n");
515 0 : goto fail;
516 : }
517 :
518 0 : if (ret == EOK) {
519 0 : tevent_req_done(req);
520 : }
521 0 : return;
522 :
523 : fail:
524 0 : tevent_req_error(req, ret);
525 0 : return;
526 : }
527 :
528 0 : static errno_t netgroup_translate_ldap_members_recv(struct tevent_req *req,
529 : TALLOC_CTX *mem_ctx,
530 : size_t *count,
531 : struct sysdb_attrs ***netgroups)
532 : {
533 0 : struct netgr_translate_members_state *state = tevent_req_data(req,
534 : struct netgr_translate_members_state);
535 :
536 0 : TEVENT_REQ_RETURN_ON_ERROR(req);
537 :
538 0 : *count = state->count;
539 0 : *netgroups = talloc_steal(mem_ctx, state->netgroups);
540 :
541 0 : return EOK;
542 : }
543 :
544 : /* ==Search-Netgroups-with-filter============================================ */
545 :
546 : struct sdap_get_netgroups_state {
547 : struct tevent_context *ev;
548 : struct sdap_options *opts;
549 : struct sdap_handle *sh;
550 : struct sss_domain_info *dom;
551 : struct sysdb_ctx *sysdb;
552 : const char **attrs;
553 : const char *base_filter;
554 : char *filter;
555 : int timeout;
556 :
557 : char *higher_timestamp;
558 : struct sysdb_attrs **netgroups;
559 : size_t count;
560 :
561 : size_t base_iter;
562 : struct sdap_search_base **search_bases;
563 : };
564 :
565 : static errno_t sdap_get_netgroups_next_base(struct tevent_req *req);
566 : static void sdap_get_netgroups_process(struct tevent_req *subreq);
567 : static void netgr_translate_members_done(struct tevent_req *subreq);
568 :
569 0 : struct tevent_req *sdap_get_netgroups_send(TALLOC_CTX *memctx,
570 : struct tevent_context *ev,
571 : struct sss_domain_info *dom,
572 : struct sysdb_ctx *sysdb,
573 : struct sdap_options *opts,
574 : struct sdap_search_base **search_bases,
575 : struct sdap_handle *sh,
576 : const char **attrs,
577 : const char *filter,
578 : int timeout)
579 : {
580 : errno_t ret;
581 : struct tevent_req *req;
582 : struct sdap_get_netgroups_state *state;
583 :
584 0 : req = tevent_req_create(memctx, &state, struct sdap_get_netgroups_state);
585 0 : if (!req) return NULL;
586 :
587 0 : state->ev = ev;
588 0 : state->opts = opts;
589 0 : state->dom = dom;
590 0 : state->sh = sh;
591 0 : state->sysdb = sysdb;
592 0 : state->attrs = attrs;
593 0 : state->higher_timestamp = NULL;
594 0 : state->netgroups = NULL;
595 0 : state->count = 0;
596 0 : state->timeout = timeout;
597 0 : state->base_filter = filter;
598 0 : state->base_iter = 0;
599 0 : state->search_bases = search_bases;
600 :
601 0 : if (!state->search_bases) {
602 0 : DEBUG(SSSDBG_CRIT_FAILURE,
603 : "Netgroup lookup request without a netgroup search base\n");
604 0 : ret = EINVAL;
605 0 : goto done;
606 : }
607 :
608 :
609 0 : ret = sdap_get_netgroups_next_base(req);
610 :
611 : done:
612 0 : if (ret != EOK) {
613 0 : tevent_req_error(req, ret);
614 0 : tevent_req_post(req, state->ev);
615 : }
616 0 : return req;
617 : }
618 :
619 0 : static errno_t sdap_get_netgroups_next_base(struct tevent_req *req)
620 : {
621 : struct tevent_req *subreq;
622 : struct sdap_get_netgroups_state *state;
623 :
624 0 : state = tevent_req_data(req, struct sdap_get_netgroups_state);
625 :
626 0 : talloc_zfree(state->filter);
627 0 : state->filter = sdap_get_id_specific_filter(state,
628 : state->base_filter,
629 0 : state->search_bases[state->base_iter]->filter);
630 0 : if (!state->filter) {
631 0 : return ENOMEM;
632 : }
633 :
634 0 : DEBUG(SSSDBG_TRACE_FUNC,
635 : "Searching for netgroups with base [%s]\n",
636 : state->search_bases[state->base_iter]->basedn);
637 :
638 0 : subreq = sdap_get_generic_send(
639 : state, state->ev, state->opts, state->sh,
640 0 : state->search_bases[state->base_iter]->basedn,
641 0 : state->search_bases[state->base_iter]->scope,
642 0 : state->filter, state->attrs,
643 0 : state->opts->netgroup_map, SDAP_OPTS_NETGROUP,
644 : state->timeout,
645 : false);
646 0 : if (!subreq) {
647 0 : return ENOMEM;
648 : }
649 0 : tevent_req_set_callback(subreq, sdap_get_netgroups_process, req);
650 :
651 0 : return EOK;
652 : }
653 :
654 0 : static void sdap_get_netgroups_process(struct tevent_req *subreq)
655 : {
656 0 : struct tevent_req *req = tevent_req_callback_data(subreq,
657 : struct tevent_req);
658 0 : struct sdap_get_netgroups_state *state = tevent_req_data(req,
659 : struct sdap_get_netgroups_state);
660 : int ret;
661 :
662 0 : ret = sdap_get_generic_recv(subreq, state,
663 : &state->count, &state->netgroups);
664 0 : talloc_zfree(subreq);
665 0 : if (ret) {
666 0 : tevent_req_error(req, ret);
667 0 : return;
668 : }
669 :
670 0 : DEBUG(SSSDBG_TRACE_FUNC,
671 : "Search for netgroups, returned %zu results.\n", state->count);
672 :
673 0 : if (state->count == 0) {
674 : /* No netgroups found in this search */
675 0 : state->base_iter++;
676 0 : if (state->search_bases[state->base_iter]) {
677 : /* There are more search bases to try */
678 0 : ret = sdap_get_netgroups_next_base(req);
679 0 : if (ret != EOK) {
680 0 : tevent_req_error(req, ENOENT);
681 : }
682 0 : return;
683 : }
684 :
685 0 : tevent_req_error(req, ENOENT);
686 0 : return;
687 : }
688 :
689 0 : subreq = netgr_translate_members_send(state, state->ev, state->opts,
690 : state->sh, state->dom, state->sysdb,
691 : state->count, state->netgroups);
692 0 : if (!subreq) {
693 0 : tevent_req_error(req, ENOMEM);
694 0 : return;
695 : }
696 0 : tevent_req_set_callback(subreq, netgr_translate_members_done, req);
697 :
698 0 : return;
699 :
700 : }
701 :
702 0 : static void netgr_translate_members_done(struct tevent_req *subreq)
703 : {
704 0 : struct tevent_req *req = tevent_req_callback_data(subreq,
705 : struct tevent_req);
706 0 : struct sdap_get_netgroups_state *state = tevent_req_data(req,
707 : struct sdap_get_netgroups_state);
708 : int ret;
709 : size_t c;
710 : time_t now;
711 :
712 0 : ret = netgroup_translate_ldap_members_recv(subreq, state, &state->count,
713 : &state->netgroups);
714 0 : talloc_zfree(subreq);
715 0 : if (ret) {
716 0 : tevent_req_error(req, ret);
717 0 : return;
718 : }
719 :
720 0 : now = time(NULL);
721 0 : for (c = 0; c < state->count; c++) {
722 0 : ret = sdap_save_netgroup(state,
723 : state->dom,
724 : state->opts,
725 0 : state->netgroups[c],
726 : &state->higher_timestamp,
727 : now);
728 0 : if (ret) {
729 0 : DEBUG(SSSDBG_OP_FAILURE, "Failed to store netgroups.\n");
730 0 : tevent_req_error(req, ret);
731 0 : return;
732 : }
733 : }
734 :
735 0 : DEBUG(SSSDBG_TRACE_ALL, "Saving %zu Netgroups - Done\n", state->count);
736 :
737 0 : tevent_req_done(req);
738 : }
739 :
740 0 : int sdap_get_netgroups_recv(struct tevent_req *req,
741 : TALLOC_CTX *mem_ctx, char **timestamp,
742 : size_t *reply_count,
743 : struct sysdb_attrs ***reply)
744 : {
745 0 : struct sdap_get_netgroups_state *state = tevent_req_data(req,
746 : struct sdap_get_netgroups_state);
747 :
748 0 : TEVENT_REQ_RETURN_ON_ERROR(req);
749 :
750 0 : if (timestamp) {
751 0 : *timestamp = talloc_steal(mem_ctx, state->higher_timestamp);
752 : }
753 :
754 0 : if (reply_count) {
755 0 : *reply_count = state->count;
756 : }
757 :
758 0 : if (reply) {
759 0 : *reply = talloc_steal(mem_ctx, state->netgroups);
760 : }
761 :
762 0 : return EOK;
763 : }
|