2 Unix SMB/CIFS implementation.
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 "../libcli/security/security.h"
25 struct wb_getpwsid_state
{
26 struct tevent_context
*ev
;
28 struct wbint_userinfo
*userinfo
;
29 struct winbindd_pw
*pw
;
32 static void wb_getpwsid_queryuser_done(struct tevent_req
*subreq
);
33 static void wb_getpwsid_lookupsid_done(struct tevent_req
*subreq
);
34 static void wb_getpwsid_done(struct tevent_req
*subreq
);
36 struct tevent_req
*wb_getpwsid_send(TALLOC_CTX
*mem_ctx
,
37 struct tevent_context
*ev
,
38 const struct dom_sid
*user_sid
,
39 struct winbindd_pw
*pw
)
41 struct tevent_req
*req
, *subreq
;
42 struct wb_getpwsid_state
*state
;
44 req
= tevent_req_create(mem_ctx
, &state
, struct wb_getpwsid_state
);
48 sid_copy(&state
->sid
, user_sid
);
52 subreq
= wb_queryuser_send(state
, ev
, &state
->sid
);
53 if (tevent_req_nomem(subreq
, req
)) {
54 return tevent_req_post(req
, ev
);
56 tevent_req_set_callback(subreq
, wb_getpwsid_queryuser_done
, req
);
60 static void wb_getpwsid_queryuser_done(struct tevent_req
*subreq
)
62 struct tevent_req
*req
= tevent_req_callback_data(
63 subreq
, struct tevent_req
);
64 struct wb_getpwsid_state
*state
= tevent_req_data(
65 req
, struct wb_getpwsid_state
);
68 status
= wb_queryuser_recv(subreq
, state
, &state
->userinfo
);
70 if (NT_STATUS_IS_OK(status
)
71 && (state
->userinfo
->acct_name
!= NULL
)
72 && (state
->userinfo
->acct_name
[0] != '\0'))
75 * QueryUser got us a name, let's go directly to the
78 subreq
= wb_fill_pwent_send(state
, state
->ev
, state
->userinfo
,
80 if (tevent_req_nomem(subreq
, req
)) {
83 tevent_req_set_callback(subreq
, wb_getpwsid_done
, req
);
88 * Either query_user did not succeed, or it
89 * succeeded but did not return an acct_name.
90 * (TODO: Can this happen at all???)
91 * ==> Try lsa_lookupsids.
93 if (state
->userinfo
== NULL
) {
94 state
->userinfo
= talloc_zero(state
, struct wbint_userinfo
);
95 if (tevent_req_nomem(state
->userinfo
, req
)) {
99 /* a successful query_user call would have filled these */
100 sid_copy(&state
->userinfo
->user_sid
, &state
->sid
);
101 state
->userinfo
->homedir
= NULL
;
102 state
->userinfo
->shell
= NULL
;
103 state
->userinfo
->primary_gid
= (gid_t
)-1;
106 subreq
= wb_lookupsid_send(state
, state
->ev
, &state
->sid
);
107 if (tevent_req_nomem(subreq
, req
)) {
110 tevent_req_set_callback(subreq
, wb_getpwsid_lookupsid_done
, req
);
113 static void wb_getpwsid_lookupsid_done(struct tevent_req
*subreq
)
115 struct tevent_req
*req
= tevent_req_callback_data(
116 subreq
, struct tevent_req
);
117 struct wb_getpwsid_state
*state
= tevent_req_data(
118 req
, struct wb_getpwsid_state
);
120 enum lsa_SidType type
;
123 status
= wb_lookupsid_recv(subreq
, state
->userinfo
, &type
, &domain
,
124 &state
->userinfo
->acct_name
);
126 if (tevent_req_nterror(req
, status
)) {
132 case SID_NAME_COMPUTER
:
134 * user case: we only need the account name from lookup_sids
137 case SID_NAME_DOM_GRP
:
139 case SID_NAME_WKN_GRP
:
141 * also treat group-type SIDs (they might map to ID_TYPE_BOTH)
143 sid_copy(&state
->userinfo
->group_sid
, &state
->sid
);
146 tevent_req_nterror(req
, NT_STATUS_NO_SUCH_USER
);
150 subreq
= wb_fill_pwent_send(state
, state
->ev
, state
->userinfo
,
152 if (tevent_req_nomem(subreq
, req
)) {
155 tevent_req_set_callback(subreq
, wb_getpwsid_done
, req
);
158 static void wb_getpwsid_done(struct tevent_req
*subreq
)
160 struct tevent_req
*req
= tevent_req_callback_data(
161 subreq
, struct tevent_req
);
164 status
= wb_fill_pwent_recv(subreq
);
165 if (tevent_req_nterror(req
, status
)) {
168 tevent_req_done(req
);
171 NTSTATUS
wb_getpwsid_recv(struct tevent_req
*req
)
173 return tevent_req_simple_recv_ntstatus(req
);