2 Unix SMB/CIFS implementation.
3 async lookupgroupmembers
4 Copyright (C) Volker Lendecke 2009
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 3 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program. If not, see <http://www.gnu.org/licenses/>.
22 #include "librpc/gen_ndr/ndr_winbind_c.h"
23 #include "../librpc/gen_ndr/ndr_security.h"
24 #include "../libcli/security/security.h"
25 #include "lib/util/util_tdb.h"
26 #include "lib/dbwrap/dbwrap.h"
27 #include "lib/dbwrap/dbwrap_rbt.h"
30 * We have 3 sets of routines here:
32 * wb_lookupgroupmem is the low-level one-group routine
34 * wb_groups_members walks a list of groups
36 * wb_group_members finally is the high-level routine expanding groups
41 * TODO: fill_grent_mem_domusers must be re-added
45 * Look up members of a single group. Essentially a wrapper around the
46 * lookup_groupmem winbindd_methods routine.
49 struct wb_lookupgroupmem_state
{
51 struct wbint_Principals members
;
54 static void wb_lookupgroupmem_done(struct tevent_req
*subreq
);
56 static struct tevent_req
*wb_lookupgroupmem_send(TALLOC_CTX
*mem_ctx
,
57 struct tevent_context
*ev
,
58 const struct dom_sid
*group_sid
,
59 enum lsa_SidType type
)
61 struct tevent_req
*req
, *subreq
;
62 struct wb_lookupgroupmem_state
*state
;
63 struct winbindd_domain
*domain
;
65 req
= tevent_req_create(mem_ctx
, &state
,
66 struct wb_lookupgroupmem_state
);
70 sid_copy(&state
->sid
, group_sid
);
72 domain
= find_domain_from_sid_noinit(group_sid
);
74 tevent_req_nterror(req
, NT_STATUS_NO_SUCH_GROUP
);
75 return tevent_req_post(req
, ev
);
78 subreq
= dcerpc_wbint_LookupGroupMembers_send(
79 state
, ev
, dom_child_handle(domain
), &state
->sid
, type
,
81 if (tevent_req_nomem(subreq
, req
)) {
82 return tevent_req_post(req
, ev
);
84 tevent_req_set_callback(subreq
, wb_lookupgroupmem_done
, req
);
88 static void wb_lookupgroupmem_done(struct tevent_req
*subreq
)
90 struct tevent_req
*req
= tevent_req_callback_data(
91 subreq
, struct tevent_req
);
92 struct wb_lookupgroupmem_state
*state
= tevent_req_data(
93 req
, struct wb_lookupgroupmem_state
);
94 NTSTATUS status
, result
;
96 status
= dcerpc_wbint_LookupGroupMembers_recv(subreq
, state
, &result
);
98 if (any_nt_status_not_ok(status
, result
, &status
)) {
99 tevent_req_nterror(req
, status
);
102 tevent_req_done(req
);
105 static NTSTATUS
wb_lookupgroupmem_recv(struct tevent_req
*req
,
107 uint32_t *num_members
,
108 struct wbint_Principal
**members
)
110 struct wb_lookupgroupmem_state
*state
= tevent_req_data(
111 req
, struct wb_lookupgroupmem_state
);
114 if (tevent_req_is_nterror(req
, &status
)) {
118 *num_members
= state
->members
.num_principals
;
119 *members
= talloc_move(mem_ctx
, &state
->members
.principals
);
124 * Same as wb_lookupgroupmem for a list of groups
127 struct wb_groups_members_state
{
128 struct tevent_context
*ev
;
129 struct wbint_Principal
*groups
;
132 struct wbint_Principal
*all_members
;
135 static NTSTATUS
wb_groups_members_next_subreq(
136 struct wb_groups_members_state
*state
,
137 TALLOC_CTX
*mem_ctx
, struct tevent_req
**psubreq
);
138 static void wb_groups_members_done(struct tevent_req
*subreq
);
140 static struct tevent_req
*wb_groups_members_send(TALLOC_CTX
*mem_ctx
,
141 struct tevent_context
*ev
,
143 struct wbint_Principal
*groups
)
145 struct tevent_req
*req
, *subreq
= NULL
;
146 struct wb_groups_members_state
*state
;
149 req
= tevent_req_create(mem_ctx
, &state
,
150 struct wb_groups_members_state
);
155 state
->groups
= groups
;
156 state
->num_groups
= num_groups
;
157 state
->next_group
= 0;
158 state
->all_members
= NULL
;
160 D_DEBUG("Looking up %"PRIu32
" group(s).\n", num_groups
);
161 status
= wb_groups_members_next_subreq(state
, state
, &subreq
);
162 if (tevent_req_nterror(req
, status
)) {
163 return tevent_req_post(req
, ev
);
165 if (subreq
== NULL
) {
166 tevent_req_done(req
);
167 return tevent_req_post(req
, ev
);
169 tevent_req_set_callback(subreq
, wb_groups_members_done
, req
);
173 static NTSTATUS
wb_groups_members_next_subreq(
174 struct wb_groups_members_state
*state
,
175 TALLOC_CTX
*mem_ctx
, struct tevent_req
**psubreq
)
177 struct tevent_req
*subreq
;
178 struct wbint_Principal
*g
;
180 if (state
->next_group
>= state
->num_groups
) {
185 g
= &state
->groups
[state
->next_group
];
186 state
->next_group
+= 1;
188 subreq
= wb_lookupgroupmem_send(mem_ctx
, state
->ev
, &g
->sid
, g
->type
);
189 if (subreq
== NULL
) {
190 return NT_STATUS_NO_MEMORY
;
196 static void wb_groups_members_done(struct tevent_req
*subreq
)
198 struct tevent_req
*req
= tevent_req_callback_data(
199 subreq
, struct tevent_req
);
200 struct wb_groups_members_state
*state
= tevent_req_data(
201 req
, struct wb_groups_members_state
);
202 uint32_t i
, num_all_members
;
203 uint32_t num_members
= 0;
204 struct wbint_Principal
*members
= NULL
;
207 status
= wb_lookupgroupmem_recv(subreq
, state
, &num_members
, &members
);
211 * In this error handling here we might have to be a bit more generous
212 * and just continue if an error occurred.
215 if (!NT_STATUS_IS_OK(status
)) {
216 if (!NT_STATUS_EQUAL(
217 status
, NT_STATUS_TRUSTED_DOMAIN_FAILURE
)) {
218 tevent_req_nterror(req
, status
);
224 num_all_members
= talloc_array_length(state
->all_members
);
226 D_DEBUG("Adding %"PRIu32
" new member(s) to existing %"PRIu32
" member(s)\n",
230 state
->all_members
= talloc_realloc(
231 state
, state
->all_members
, struct wbint_Principal
,
232 num_all_members
+ num_members
);
233 if ((num_all_members
+ num_members
!= 0)
234 && tevent_req_nomem(state
->all_members
, req
)) {
237 for (i
=0; i
<num_members
; i
++) {
238 struct wbint_Principal
*src
, *dst
;
240 dst
= &state
->all_members
[num_all_members
+ i
];
241 sid_copy(&dst
->sid
, &src
->sid
);
242 dst
->name
= talloc_move(state
->all_members
, &src
->name
);
243 dst
->type
= src
->type
;
245 TALLOC_FREE(members
);
247 status
= wb_groups_members_next_subreq(state
, state
, &subreq
);
248 if (tevent_req_nterror(req
, status
)) {
251 if (subreq
== NULL
) {
252 tevent_req_done(req
);
255 tevent_req_set_callback(subreq
, wb_groups_members_done
, req
);
258 static NTSTATUS
wb_groups_members_recv(struct tevent_req
*req
,
260 uint32_t *num_members
,
261 struct wbint_Principal
**members
)
263 struct wb_groups_members_state
*state
= tevent_req_data(
264 req
, struct wb_groups_members_state
);
267 if (tevent_req_is_nterror(req
, &status
)) {
270 *num_members
= talloc_array_length(state
->all_members
);
271 *members
= talloc_move(mem_ctx
, &state
->all_members
);
277 * This is the routine expanding a list of groups up to a certain level. We
278 * collect the users in a rbt database: We have to add them without duplicates,
279 * and the db is indexed by SID.
282 struct wb_group_members_state
{
283 struct tevent_context
*ev
;
285 struct db_context
*users
;
286 struct wbint_Principal
*groups
;
289 static NTSTATUS
wb_group_members_next_subreq(
290 struct wb_group_members_state
*state
,
291 TALLOC_CTX
*mem_ctx
, struct tevent_req
**psubreq
);
292 static void wb_group_members_done(struct tevent_req
*subreq
);
294 struct tevent_req
*wb_group_members_send(TALLOC_CTX
*mem_ctx
,
295 struct tevent_context
*ev
,
296 const struct dom_sid
*sid
,
298 enum lsa_SidType
*type
,
301 struct tevent_req
*req
, *subreq
= NULL
;
302 struct wb_group_members_state
*state
;
304 struct dom_sid_buf buf
;
307 req
= tevent_req_create(mem_ctx
, &state
,
308 struct wb_group_members_state
);
312 D_INFO("WB command group_members start (max_depth=%d).\n", max_depth
);
313 for (i
= 0; i
< num_sids
; i
++) {
314 D_INFO("Looking up members of group SID %s with SID type %d\n",
315 dom_sid_str_buf(&sid
[i
], &buf
),
320 state
->depth
= max_depth
;
321 state
->users
= db_open_rbt(state
);
322 if (tevent_req_nomem(state
->users
, req
)) {
323 return tevent_req_post(req
, ev
);
326 state
->groups
= talloc_array(state
, struct wbint_Principal
, num_sids
);
327 if (tevent_req_nomem(state
->groups
, req
)) {
328 return tevent_req_post(req
, ev
);
331 for (i
= 0; i
< num_sids
; i
++) {
332 state
->groups
[i
].name
= NULL
;
333 sid_copy(&state
->groups
[i
].sid
, &sid
[i
]);
334 state
->groups
[i
].type
= type
[i
];
337 status
= wb_group_members_next_subreq(state
, state
, &subreq
);
338 if (tevent_req_nterror(req
, status
)) {
339 return tevent_req_post(req
, ev
);
341 if (subreq
== NULL
) {
342 tevent_req_done(req
);
343 return tevent_req_post(req
, ev
);
345 tevent_req_set_callback(subreq
, wb_group_members_done
, req
);
349 static NTSTATUS
wb_group_members_next_subreq(
350 struct wb_group_members_state
*state
,
351 TALLOC_CTX
*mem_ctx
, struct tevent_req
**psubreq
)
353 struct tevent_req
*subreq
;
355 if ((talloc_array_length(state
->groups
) == 0)
356 || (state
->depth
<= 0)) {
358 D_DEBUG("Finished. The depth is %d.\n", state
->depth
);
363 D_DEBUG("The depth is decremented to %d.\n", state
->depth
);
364 subreq
= wb_groups_members_send(
365 mem_ctx
, state
->ev
, talloc_array_length(state
->groups
),
367 if (subreq
== NULL
) {
368 return NT_STATUS_NO_MEMORY
;
374 NTSTATUS
add_member_to_db(struct db_context
*db
, struct dom_sid
*sid
,
377 size_t len
= ndr_size_dom_sid(sid
, 0);
379 TDB_DATA key
= { .dptr
= sidbuf
, .dsize
= sizeof(sidbuf
) };
382 sid_linearize(sidbuf
, sizeof(sidbuf
), sid
);
384 status
= dbwrap_store(db
, key
, string_term_tdb_data(name
), 0);
388 static void wb_group_members_done(struct tevent_req
*subreq
)
390 struct tevent_req
*req
= tevent_req_callback_data(
391 subreq
, struct tevent_req
);
392 struct wb_group_members_state
*state
= tevent_req_data(
393 req
, struct wb_group_members_state
);
394 uint32_t i
, num_groups
, new_groups
;
395 uint32_t num_members
= 0;
396 struct wbint_Principal
*members
= NULL
;
399 status
= wb_groups_members_recv(subreq
, state
, &num_members
, &members
);
401 if (tevent_req_nterror(req
, status
)) {
406 for (i
=0; i
<num_members
; i
++) {
407 switch (members
[i
].type
) {
408 case SID_NAME_DOM_GRP
:
410 case SID_NAME_WKN_GRP
:
414 /* Ignore everything else */
420 TALLOC_FREE(state
->groups
);
421 state
->groups
= talloc_array(state
, struct wbint_Principal
,
425 * Collect the users into state->users and the groups into
426 * state->groups for the next iteration.
429 for (i
=0; i
<num_members
; i
++) {
430 switch (members
[i
].type
) {
432 case SID_NAME_COMPUTER
: {
434 * Add a copy of members[i] to state->users
436 status
= add_member_to_db(state
->users
, &members
[i
].sid
,
438 if (tevent_req_nterror(req
, status
)) {
444 case SID_NAME_DOM_GRP
:
446 case SID_NAME_WKN_GRP
: {
447 struct wbint_Principal
*g
;
449 * Save members[i] for the next round
451 g
= &state
->groups
[num_groups
];
452 sid_copy(&g
->sid
, &members
[i
].sid
);
453 g
->name
= talloc_move(state
->groups
, &members
[i
].name
);
454 g
->type
= members
[i
].type
;
459 /* Ignore everything else */
464 status
= wb_group_members_next_subreq(state
, state
, &subreq
);
465 if (tevent_req_nterror(req
, status
)) {
468 if (subreq
== NULL
) {
469 tevent_req_done(req
);
472 tevent_req_set_callback(subreq
, wb_group_members_done
, req
);
475 NTSTATUS
wb_group_members_recv(struct tevent_req
*req
, TALLOC_CTX
*mem_ctx
,
476 struct db_context
**members
)
478 struct wb_group_members_state
*state
= tevent_req_data(
479 req
, struct wb_group_members_state
);
482 D_INFO("WB command group_members end.\n");
483 if (tevent_req_is_nterror(req
, &status
)) {
484 D_WARNING("Failed with %s.\n", nt_errstr(status
));
487 *members
= talloc_move(mem_ctx
, &state
->users
);