2 Unix SMB/CIFS implementation.
3 Net_sam_logon info3 helpers
4 Copyright (C) Alexander Bokovoy 2002.
5 Copyright (C) Andrew Bartlett 2002.
6 Copyright (C) Gerald Carter 2003.
7 Copyright (C) Tim Potter 2003.
8 Copyright (C) Guenther Deschner 2008.
10 This program is free software; you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation; either version 3 of the License, or
13 (at your option) any later version.
15 This program is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
20 You should have received a copy of the GNU General Public License
21 along with this program. If not, see <http://www.gnu.org/licenses/>.
25 #include "librpc/gen_ndr/ndr_krb5pac.h"
27 #define NETSAMLOGON_TDB "netsamlogon_cache.tdb"
29 static TDB_CONTEXT
*netsamlogon_tdb
= NULL
;
31 /***********************************************************************
33 ***********************************************************************/
35 bool netsamlogon_cache_init(void)
37 bool first_try
= true;
38 const char *path
= NULL
;
40 struct tdb_context
*tdb
;
42 if (netsamlogon_tdb
) {
46 path
= cache_path(NETSAMLOGON_TDB
);
48 tdb
= tdb_open_log(path
, 0, TDB_DEFAULT
,
49 O_RDWR
| O_CREAT
, 0600);
51 DEBUG(0,("tdb_open_log('%s') - failed\n", path
));
55 ret
= tdb_check(tdb
, NULL
, NULL
);
58 DEBUG(0,("tdb_check('%s') - failed\n", path
));
62 netsamlogon_tdb
= tdb
;
71 DEBUG(0,("retry after CLEAR_IF_FIRST for '%s'\n", path
));
72 tdb
= tdb_open_log(path
, 0, TDB_CLEAR_IF_FIRST
,
73 O_RDWR
| O_CREAT
, 0600);
78 DEBUG(0,("tdb_open_log(%s) with CLEAR_IF_FIRST - failed\n", path
));
84 /***********************************************************************
85 Shutdown samlogon_cache database
86 ***********************************************************************/
88 bool netsamlogon_cache_shutdown(void)
90 if (netsamlogon_tdb
) {
91 return (tdb_close(netsamlogon_tdb
) == 0);
97 /***********************************************************************
98 Clear cache getpwnam and getgroups entries from the winbindd cache
99 ***********************************************************************/
101 void netsamlogon_clear_cached_user(struct netr_SamInfo3
*info3
)
103 struct dom_sid user_sid
;
110 if (!netsamlogon_cache_init()) {
111 DEBUG(0,("netsamlogon_clear_cached_user: cannot open "
116 sid_compose(&user_sid
, info3
->base
.domain_sid
, info3
->base
.rid
);
118 /* Prepare key as DOMAIN-SID/USER-RID string */
119 slprintf(keystr
, sizeof(keystr
), "%s", sid_to_fstring(tmp
, &user_sid
));
121 DEBUG(10,("netsamlogon_clear_cached_user: SID [%s]\n", keystr
));
123 tdb_delete_bystring(netsamlogon_tdb
, keystr
);
126 /***********************************************************************
127 Store a netr_SamInfo3 structure in a tdb for later user
128 username should be in UTF-8 format
129 ***********************************************************************/
131 bool netsamlogon_cache_store(const char *username
, struct netr_SamInfo3
*info3
)
136 struct dom_sid user_sid
;
137 time_t t
= time(NULL
);
140 enum ndr_err_code ndr_err
;
141 struct netsamlogoncache_entry r
;
147 if (!netsamlogon_cache_init()) {
148 DEBUG(0,("netsamlogon_cache_store: cannot open %s for write!\n",
153 sid_compose(&user_sid
, info3
->base
.domain_sid
, info3
->base
.rid
);
155 /* Prepare key as DOMAIN-SID/USER-RID string */
156 slprintf(keystr
, sizeof(keystr
), "%s", sid_to_fstring(tmp
, &user_sid
));
158 DEBUG(10,("netsamlogon_cache_store: SID [%s]\n", keystr
));
162 if (!(mem_ctx
= TALLOC_P( NULL
, int))) {
163 DEBUG(0,("netsamlogon_cache_store: talloc() failed!\n"));
167 /* only Samba fills in the username, not sure why NT doesn't */
168 /* so we fill it in since winbindd_getpwnam() makes use of it */
170 if (!info3
->base
.account_name
.string
) {
171 info3
->base
.account_name
.string
= talloc_strdup(info3
, username
);
177 if (DEBUGLEVEL
>= 10) {
178 NDR_PRINT_DEBUG(netsamlogoncache_entry
, &r
);
181 ndr_err
= ndr_push_struct_blob(&blob
, mem_ctx
, &r
,
182 (ndr_push_flags_fn_t
)ndr_push_netsamlogoncache_entry
);
183 if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err
)) {
184 DEBUG(0,("netsamlogon_cache_store: failed to push entry to cache\n"));
185 TALLOC_FREE(mem_ctx
);
189 data
.dsize
= blob
.length
;
190 data
.dptr
= blob
.data
;
192 if (tdb_store_bystring(netsamlogon_tdb
, keystr
, data
, TDB_REPLACE
) != -1) {
196 TALLOC_FREE(mem_ctx
);
201 /***********************************************************************
202 Retrieves a netr_SamInfo3 structure from a tdb. Caller must
203 free the user_info struct (malloc()'d memory)
204 ***********************************************************************/
206 struct netr_SamInfo3
*netsamlogon_cache_get(TALLOC_CTX
*mem_ctx
, const struct dom_sid
*user_sid
)
208 struct netr_SamInfo3
*info3
= NULL
;
211 enum ndr_err_code ndr_err
;
213 struct netsamlogoncache_entry r
;
215 if (!netsamlogon_cache_init()) {
216 DEBUG(0,("netsamlogon_cache_get: cannot open %s for write!\n",
221 /* Prepare key as DOMAIN-SID/USER-RID string */
222 slprintf(keystr
, sizeof(keystr
), "%s", sid_to_fstring(tmp
, user_sid
));
223 DEBUG(10,("netsamlogon_cache_get: SID [%s]\n", keystr
));
224 data
= tdb_fetch_bystring( netsamlogon_tdb
, keystr
);
230 info3
= TALLOC_ZERO_P(mem_ctx
, struct netr_SamInfo3
);
235 blob
= data_blob_const(data
.dptr
, data
.dsize
);
237 ndr_err
= ndr_pull_struct_blob(&blob
, mem_ctx
, &r
,
238 (ndr_pull_flags_fn_t
)ndr_pull_netsamlogoncache_entry
);
240 if (DEBUGLEVEL
>= 10) {
241 NDR_PRINT_DEBUG(netsamlogoncache_entry
, &r
);
244 if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err
)) {
245 DEBUG(0,("netsamlogon_cache_get: failed to pull entry from cache\n"));
246 tdb_delete(netsamlogon_tdb
, data
);
251 info3
= (struct netr_SamInfo3
*)talloc_memdup(mem_ctx
, &r
.info3
,
255 SAFE_FREE(data
.dptr
);
259 #if 0 /* The netsamlogon cache needs to hang around. Something about
260 this feels wrong, but it is the only way we can get all of the
261 groups. The old universal groups cache didn't expire either.
264 time_t now
= time(NULL
);
267 /* is the entry expired? */
270 if ( (time_diff
< 0 ) || (time_diff
> lp_winbind_cache_time()) ) {
271 DEBUG(10,("netsamlogon_cache_get: cache entry expired \n"));
272 tdb_delete( netsamlogon_tdb
, key
);
279 bool netsamlogon_cache_have(const struct dom_sid
*user_sid
)
281 TALLOC_CTX
*mem_ctx
= talloc_init("netsamlogon_cache_have");
282 struct netr_SamInfo3
*info3
= NULL
;
288 info3
= netsamlogon_cache_get(mem_ctx
, user_sid
);
290 result
= (info3
!= NULL
);
292 talloc_destroy(mem_ctx
);