2 Unix SMB/CIFS implementation.
3 async implementation of WINBINDD_GETGRNAM
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/>.
23 struct winbindd_getgrnam_state
{
24 struct tevent_context
*ev
;
25 fstring name_domain
, name_group
;
30 struct talloc_dict
*members
;
33 static void winbindd_getgrnam_lookupsid_done(struct tevent_req
*subreq
);
34 static void winbindd_getgrnam_done(struct tevent_req
*subreq
);
36 struct tevent_req
*winbindd_getgrnam_send(TALLOC_CTX
*mem_ctx
,
37 struct tevent_context
*ev
,
38 struct winbindd_cli_state
*cli
,
39 struct winbindd_request
*request
)
41 struct tevent_req
*req
, *subreq
;
42 struct winbindd_getgrnam_state
*state
;
43 struct winbindd_domain
*domain
;
47 req
= tevent_req_create(mem_ctx
, &state
,
48 struct winbindd_getgrnam_state
);
54 /* Ensure null termination */
55 request
->data
.groupname
[sizeof(request
->data
.groupname
)-1]='\0';
57 DEBUG(3, ("getgrnam %s\n", request
->data
.groupname
));
59 nt_status
= normalize_name_unmap(state
, request
->data
.groupname
, &tmp
);
60 /* If we didn't map anything in the above call, just reset the
61 tmp pointer to the original string */
62 if (!NT_STATUS_IS_OK(nt_status
) &&
63 !NT_STATUS_EQUAL(nt_status
, NT_STATUS_FILE_RENAMED
))
65 tmp
= request
->data
.groupname
;
68 /* Parse domain and groupname */
70 parse_domain_user(tmp
, state
->name_domain
, state
->name_group
);
72 /* if no domain or our local domain and no local tdb group, default to
73 * our local domain for aliases */
75 if ( !*(state
->name_domain
) || strequal(state
->name_domain
,
76 get_global_sam_name()) ) {
77 fstrcpy(state
->name_domain
, get_global_sam_name());
80 /* Get info for the domain */
82 domain
= find_domain_from_name_noinit(state
->name_domain
);
84 DEBUG(3, ("could not get domain sid for domain %s\n",
86 tevent_req_nterror(req
, NT_STATUS_NO_SUCH_GROUP
);
87 return tevent_req_post(req
, ev
);
90 /* should we deal with users for our domain? */
92 if ( lp_winbind_trusted_domains_only() && domain
->primary
) {
93 DEBUG(7,("winbindd_getgrnam: My domain -- rejecting "
94 "getgrnam() for %s\\%s.\n", state
->name_domain
,
96 tevent_req_nterror(req
, NT_STATUS_NO_SUCH_GROUP
);
97 return tevent_req_post(req
, ev
);
100 subreq
= wb_lookupname_send(state
, ev
, domain
->name
, state
->name_group
,
102 if (tevent_req_nomem(subreq
, req
)) {
103 return tevent_req_post(req
, ev
);
105 tevent_req_set_callback(subreq
, winbindd_getgrnam_lookupsid_done
,
110 static void winbindd_getgrnam_lookupsid_done(struct tevent_req
*subreq
)
112 struct tevent_req
*req
= tevent_req_callback_data(
113 subreq
, struct tevent_req
);
114 struct winbindd_getgrnam_state
*state
= tevent_req_data(
115 req
, struct winbindd_getgrnam_state
);
116 enum lsa_SidType type
;
119 status
= wb_lookupname_recv(subreq
, &state
->sid
, &type
);
121 if (!NT_STATUS_IS_OK(status
)) {
122 tevent_req_nterror(req
, status
);
126 if ( (type
!= SID_NAME_DOM_GRP
) && (type
!= SID_NAME_ALIAS
) ) {
127 DEBUG(5,("getgrnam_recv: not a group!\n"));
128 tevent_req_nterror(req
, NT_STATUS_NO_SUCH_GROUP
);
132 subreq
= wb_getgrsid_send(state
, state
->ev
, &state
->sid
,
133 lp_winbind_expand_groups());
134 if (tevent_req_nomem(subreq
, req
)) {
137 tevent_req_set_callback(subreq
, winbindd_getgrnam_done
, req
);
140 static void winbindd_getgrnam_done(struct tevent_req
*subreq
)
142 struct tevent_req
*req
= tevent_req_callback_data(
143 subreq
, struct tevent_req
);
144 struct winbindd_getgrnam_state
*state
= tevent_req_data(
145 req
, struct winbindd_getgrnam_state
);
148 status
= wb_getgrsid_recv(subreq
, state
, &state
->domname
, &state
->name
,
149 &state
->gid
, &state
->members
);
151 if (!NT_STATUS_IS_OK(status
)) {
152 tevent_req_nterror(req
, status
);
155 tevent_req_done(req
);
158 NTSTATUS
winbindd_getgrnam_recv(struct tevent_req
*req
,
159 struct winbindd_response
*response
)
161 struct winbindd_getgrnam_state
*state
= tevent_req_data(
162 req
, struct winbindd_getgrnam_state
);
167 if (tevent_req_is_nterror(req
, &status
)) {
168 DEBUG(5, ("Could not convert sid %s: %s\n",
169 sid_string_dbg(&state
->sid
), nt_errstr(status
)));
173 if (!fill_grent(talloc_tos(), &response
->data
.gr
, state
->domname
,
174 state
->name
, state
->gid
)) {
175 DEBUG(5, ("fill_grent failed\n"));
176 return NT_STATUS_NO_MEMORY
;
179 status
= winbindd_print_groupmembers(state
->members
, response
,
181 if (!NT_STATUS_IS_OK(status
)) {
185 response
->data
.gr
.num_gr_mem
= (uint32
)num_members
;
187 /* Group membership lives at start of extra data */
189 response
->data
.gr
.gr_mem_ofs
= 0;
190 response
->extra_data
.data
= buf
;
191 response
->length
+= talloc_get_size(response
->extra_data
.data
);