s4:auth: Enforce machine authentication policy for NTLM authentication
[Samba.git] / source3 / libads / kerberos.c
blob9a5ca567ca97ce431348cb64b9f375f91df7ea46
1 /*
2 Unix SMB/CIFS implementation.
3 kerberos utility library
4 Copyright (C) Andrew Tridgell 2001
5 Copyright (C) Remus Koos 2001
6 Copyright (C) Nalin Dahyabhai <nalin@redhat.com> 2004.
7 Copyright (C) Jeremy Allison 2004.
8 Copyright (C) Gerald Carter 2006.
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/>.
24 #include "includes.h"
25 #include "libsmb/namequery.h"
26 #include "system/filesys.h"
27 #include "smb_krb5.h"
28 #include "../librpc/gen_ndr/ndr_misc.h"
29 #include "libads/kerberos_proto.h"
30 #include "libads/cldap.h"
31 #include "secrets.h"
32 #include "../lib/tsocket/tsocket.h"
33 #include "lib/util/asn1.h"
35 #ifdef HAVE_KRB5
37 #define LIBADS_CCACHE_NAME "MEMORY:libads"
40 we use a prompter to avoid a crash bug in the kerberos libs when
41 dealing with empty passwords
42 this prompter is just a string copy ...
44 static krb5_error_code
45 kerb_prompter(krb5_context ctx, void *data,
46 const char *name,
47 const char *banner,
48 int num_prompts,
49 krb5_prompt prompts[])
51 if (num_prompts == 0) return 0;
52 if (num_prompts == 2) {
54 * only heimdal has a prompt type and we need to deal with it here to
55 * avoid loops.
57 * removing the prompter completely is not an option as at least these
58 * versions would crash: heimdal-1.0.2 and heimdal-1.1. Later heimdal
59 * version have looping detection and return with a proper error code.
62 #if defined(HAVE_KRB5_PROMPT_TYPE) /* Heimdal */
63 if (prompts[0].type == KRB5_PROMPT_TYPE_NEW_PASSWORD &&
64 prompts[1].type == KRB5_PROMPT_TYPE_NEW_PASSWORD_AGAIN) {
66 * We don't want to change passwords here. We're
67 * called from heimal when the KDC returns
68 * KRB5KDC_ERR_KEY_EXPIRED, but at this point we don't
69 * have the chance to ask the user for a new
70 * password. If we return 0 (i.e. success), we will be
71 * spinning in the endless for-loop in
72 * change_password() in
73 * source4/heimdal/lib/krb5/init_creds_pw.c:526ff
75 return KRB5KDC_ERR_KEY_EXPIRED;
77 #elif defined(HAVE_KRB5_GET_PROMPT_TYPES) /* MIT */
78 krb5_prompt_type *prompt_types = NULL;
80 prompt_types = krb5_get_prompt_types(ctx);
81 if (prompt_types != NULL) {
82 if (prompt_types[0] == KRB5_PROMPT_TYPE_NEW_PASSWORD &&
83 prompt_types[1] == KRB5_PROMPT_TYPE_NEW_PASSWORD_AGAIN) {
84 return KRB5KDC_ERR_KEY_EXP;
87 #endif
90 memset(prompts[0].reply->data, '\0', prompts[0].reply->length);
91 if (prompts[0].reply->length > 0) {
92 if (data) {
93 strncpy((char *)prompts[0].reply->data, (const char *)data,
94 prompts[0].reply->length-1);
95 prompts[0].reply->length = strlen((const char *)prompts[0].reply->data);
96 } else {
97 prompts[0].reply->length = 0;
100 return 0;
104 simulate a kinit, putting the tgt in the given cache location. If cache_name == NULL
105 place in default cache location.
106 remus@snapserver.com
108 int kerberos_kinit_password_ext(const char *given_principal,
109 const char *password,
110 int time_offset,
111 time_t *expire_time,
112 time_t *renew_till_time,
113 const char *cache_name,
114 bool request_pac,
115 bool add_netbios_addr,
116 time_t renewable_time,
117 TALLOC_CTX *mem_ctx,
118 char **_canon_principal,
119 char **_canon_realm,
120 NTSTATUS *ntstatus)
122 TALLOC_CTX *frame = talloc_stackframe();
123 krb5_context ctx = NULL;
124 krb5_error_code code = 0;
125 krb5_ccache cc = NULL;
126 krb5_principal me = NULL;
127 krb5_principal canon_princ = NULL;
128 krb5_creds my_creds;
129 krb5_get_init_creds_opt *opt = NULL;
130 smb_krb5_addresses *addr = NULL;
131 char *canon_principal = NULL;
132 char *canon_realm = NULL;
134 ZERO_STRUCT(my_creds);
136 code = smb_krb5_init_context_common(&ctx);
137 if (code != 0) {
138 DBG_ERR("kerberos init context failed (%s)\n",
139 error_message(code));
140 TALLOC_FREE(frame);
141 return code;
144 if (time_offset != 0) {
145 krb5_set_real_time(ctx, time(NULL) + time_offset, 0);
148 DBG_DEBUG("as %s using [%s] as ccache and config [%s]\n",
149 given_principal,
150 cache_name ? cache_name: krb5_cc_default_name(ctx),
151 getenv("KRB5_CONFIG"));
153 if ((code = krb5_cc_resolve(ctx, cache_name ? cache_name : krb5_cc_default_name(ctx), &cc))) {
154 goto out;
157 if ((code = smb_krb5_parse_name(ctx, given_principal, &me))) {
158 goto out;
161 if ((code = krb5_get_init_creds_opt_alloc(ctx, &opt))) {
162 goto out;
165 krb5_get_init_creds_opt_set_renew_life(opt, renewable_time);
166 krb5_get_init_creds_opt_set_forwardable(opt, True);
168 /* Turn on canonicalization for lower case realm support */
169 #ifdef SAMBA4_USES_HEIMDAL
170 krb5_get_init_creds_opt_set_win2k(ctx, opt, true);
171 krb5_get_init_creds_opt_set_canonicalize(ctx, opt, true);
172 #else /* MIT */
173 krb5_get_init_creds_opt_set_canonicalize(opt, true);
174 #endif /* MIT */
175 #if 0
176 /* insane testing */
177 krb5_get_init_creds_opt_set_tkt_life(opt, 60);
178 #endif
180 #ifdef HAVE_KRB5_GET_INIT_CREDS_OPT_SET_PAC_REQUEST
181 if (request_pac) {
182 if ((code = krb5_get_init_creds_opt_set_pac_request(ctx, opt, (krb5_boolean)request_pac))) {
183 goto out;
186 #endif
187 if (add_netbios_addr) {
188 if ((code = smb_krb5_gen_netbios_krb5_address(&addr,
189 lp_netbios_name()))) {
190 goto out;
192 krb5_get_init_creds_opt_set_address_list(opt, addr->addrs);
195 if ((code = krb5_get_init_creds_password(ctx, &my_creds, me, discard_const_p(char,password),
196 kerb_prompter, discard_const_p(char, password),
197 0, NULL, opt))) {
198 goto out;
201 canon_princ = my_creds.client;
203 code = smb_krb5_unparse_name(frame,
204 ctx,
205 canon_princ,
206 &canon_principal);
207 if (code != 0) {
208 goto out;
211 DBG_DEBUG("%s mapped to %s\n", given_principal, canon_principal);
213 canon_realm = smb_krb5_principal_get_realm(frame, ctx, canon_princ);
214 if (canon_realm == NULL) {
215 code = ENOMEM;
216 goto out;
219 if ((code = krb5_cc_initialize(ctx, cc, canon_princ))) {
220 goto out;
223 if ((code = krb5_cc_store_cred(ctx, cc, &my_creds))) {
224 goto out;
227 if (expire_time) {
228 *expire_time = (time_t) my_creds.times.endtime;
231 if (renew_till_time) {
232 *renew_till_time = (time_t) my_creds.times.renew_till;
235 if (_canon_principal != NULL) {
236 *_canon_principal = talloc_move(mem_ctx, &canon_principal);
238 if (_canon_realm != NULL) {
239 *_canon_realm = talloc_move(mem_ctx, &canon_realm);
241 out:
242 if (ntstatus) {
243 /* fast path */
244 if (code == 0) {
245 *ntstatus = NT_STATUS_OK;
246 goto cleanup;
249 /* fall back to self-made-mapping */
250 *ntstatus = krb5_to_nt_status(code);
253 cleanup:
254 krb5_free_cred_contents(ctx, &my_creds);
255 if (me) {
256 krb5_free_principal(ctx, me);
258 if (addr) {
259 smb_krb5_free_addresses(ctx, addr);
261 if (opt) {
262 krb5_get_init_creds_opt_free(ctx, opt);
264 if (cc) {
265 krb5_cc_close(ctx, cc);
267 if (ctx) {
268 krb5_free_context(ctx);
270 TALLOC_FREE(frame);
271 return code;
274 int ads_kdestroy(const char *cc_name)
276 krb5_error_code code;
277 krb5_context ctx = NULL;
278 krb5_ccache cc = NULL;
280 code = smb_krb5_init_context_common(&ctx);
281 if (code != 0) {
282 DBG_ERR("kerberos init context failed (%s)\n",
283 error_message(code));
284 return code;
287 if (!cc_name) {
288 if ((code = krb5_cc_default(ctx, &cc))) {
289 krb5_free_context(ctx);
290 return code;
292 } else {
293 if ((code = krb5_cc_resolve(ctx, cc_name, &cc))) {
294 DEBUG(3, ("ads_kdestroy: krb5_cc_resolve failed: %s\n",
295 error_message(code)));
296 krb5_free_context(ctx);
297 return code;
301 if ((code = krb5_cc_destroy (ctx, cc))) {
302 DEBUG(3, ("ads_kdestroy: krb5_cc_destroy failed: %s\n",
303 error_message(code)));
306 krb5_free_context (ctx);
307 return code;
310 int create_kerberos_key_from_string(krb5_context context,
311 krb5_principal host_princ,
312 krb5_principal salt_princ,
313 krb5_data *password,
314 krb5_keyblock *key,
315 krb5_enctype enctype,
316 bool no_salt)
318 int ret;
320 * Check if we've determined that the KDC is salting keys for this
321 * principal/enctype in a non-obvious way. If it is, try to match
322 * its behavior.
324 if (no_salt) {
325 KRB5_KEY_DATA(key) = (KRB5_KEY_DATA_CAST *)SMB_MALLOC(password->length);
326 if (!KRB5_KEY_DATA(key)) {
327 return ENOMEM;
329 memcpy(KRB5_KEY_DATA(key), password->data, password->length);
330 KRB5_KEY_LENGTH(key) = password->length;
331 KRB5_KEY_TYPE(key) = enctype;
332 return 0;
334 ret = smb_krb5_create_key_from_string(context,
335 salt_princ ? salt_princ : host_princ,
336 NULL,
337 password,
338 enctype,
339 key);
340 return ret;
343 /************************************************************************
344 ************************************************************************/
346 int kerberos_kinit_password(const char *principal,
347 const char *password,
348 int time_offset,
349 const char *cache_name)
351 return kerberos_kinit_password_ext(principal,
352 password,
353 time_offset,
356 cache_name,
357 False,
358 False,
360 NULL,
361 NULL,
362 NULL,
363 NULL);
366 /************************************************************************
367 ************************************************************************/
369 /************************************************************************
370 Create a string list of available kdc's, possibly searching by sitename.
371 Does DNS queries.
373 If "sitename" is given, the DC's in that site are listed first.
375 ************************************************************************/
377 static void add_sockaddr_unique(struct sockaddr_storage *addrs, size_t *num_addrs,
378 const struct sockaddr_storage *addr)
380 size_t i;
382 for (i=0; i<*num_addrs; i++) {
383 if (sockaddr_equal((const struct sockaddr *)&addrs[i],
384 (const struct sockaddr *)addr)) {
385 return;
388 addrs[i] = *addr;
389 *num_addrs += 1;
392 /* print_canonical_sockaddr prints an ipv6 addr in the form of
393 * [ipv6.addr]. This string, when put in a generated krb5.conf file is not
394 * always properly dealt with by some older krb5 libraries. Adding the hard-coded
395 * portnumber workarounds the issue. - gd */
397 static char *print_canonical_sockaddr_with_port(TALLOC_CTX *mem_ctx,
398 const struct sockaddr_storage *pss)
400 char *str = NULL;
402 str = print_canonical_sockaddr(mem_ctx, pss);
403 if (str == NULL) {
404 return NULL;
407 if (pss->ss_family != AF_INET6) {
408 return str;
411 #if defined(HAVE_IPV6)
412 str = talloc_asprintf_append(str, ":88");
413 #endif
414 return str;
417 static char *get_kdc_ip_string(char *mem_ctx,
418 const char *realm,
419 const char *sitename,
420 const struct sockaddr_storage *pss)
422 TALLOC_CTX *frame = talloc_stackframe();
423 size_t i;
424 struct samba_sockaddr *ip_sa_site = NULL;
425 struct samba_sockaddr *ip_sa_nonsite = NULL;
426 struct samba_sockaddr sa = {0};
427 size_t count_site = 0;
428 size_t count_nonsite;
429 size_t num_dcs;
430 struct sockaddr_storage *dc_addrs = NULL;
431 struct tsocket_address **dc_addrs2 = NULL;
432 const struct tsocket_address * const *dc_addrs3 = NULL;
433 char *result = NULL;
434 struct netlogon_samlogon_response **responses = NULL;
435 NTSTATUS status;
436 bool ok;
437 char *kdc_str = NULL;
438 char *canon_sockaddr = NULL;
440 SMB_ASSERT(pss != NULL);
442 canon_sockaddr = print_canonical_sockaddr_with_port(frame, pss);
443 if (canon_sockaddr == NULL) {
444 goto out;
447 kdc_str = talloc_asprintf(frame,
448 "\t\tkdc = %s\n",
449 canon_sockaddr);
450 if (kdc_str == NULL) {
451 goto out;
454 ok = sockaddr_storage_to_samba_sockaddr(&sa, pss);
455 if (!ok) {
456 goto out;
460 * First get the KDC's only in this site, the rest will be
461 * appended later
464 if (sitename) {
465 status = get_kdc_list(frame,
466 realm,
467 sitename,
468 &ip_sa_site,
469 &count_site);
470 if (!NT_STATUS_IS_OK(status)) {
471 DBG_ERR("get_kdc_list fail %s\n",
472 nt_errstr(status));
473 goto out;
475 DBG_DEBUG("got %zu addresses from site %s search\n",
476 count_site,
477 sitename);
480 /* Get all KDC's. */
482 status = get_kdc_list(frame,
483 realm,
484 NULL,
485 &ip_sa_nonsite,
486 &count_nonsite);
487 if (!NT_STATUS_IS_OK(status)) {
488 DBG_ERR("get_kdc_list (site-less) fail %s\n",
489 nt_errstr(status));
490 goto out;
492 DBG_DEBUG("got %zu addresses from site-less search\n", count_nonsite);
494 if (count_site + count_nonsite < count_site) {
495 /* Wrap check. */
496 DBG_ERR("get_kdc_list_talloc (site-less) fail wrap error\n");
497 goto out;
501 dc_addrs = talloc_array(talloc_tos(), struct sockaddr_storage,
502 count_site + count_nonsite);
503 if (dc_addrs == NULL) {
504 goto out;
507 num_dcs = 0;
509 for (i = 0; i < count_site; i++) {
510 if (!sockaddr_equal(&sa.u.sa, &ip_sa_site[i].u.sa)) {
511 add_sockaddr_unique(dc_addrs, &num_dcs,
512 &ip_sa_site[i].u.ss);
516 for (i = 0; i < count_nonsite; i++) {
517 if (!sockaddr_equal(&sa.u.sa, &ip_sa_nonsite[i].u.sa)) {
518 add_sockaddr_unique(dc_addrs, &num_dcs,
519 &ip_sa_nonsite[i].u.ss);
523 DBG_DEBUG("%zu additional KDCs to test\n", num_dcs);
524 if (num_dcs == 0) {
526 * We do not have additional KDCs, but we have the one passed
527 * in via `pss`. So just use that one and leave.
529 result = talloc_move(mem_ctx, &kdc_str);
530 goto out;
533 dc_addrs2 = talloc_zero_array(talloc_tos(),
534 struct tsocket_address *,
535 num_dcs);
536 if (dc_addrs2 == NULL) {
537 goto out;
540 for (i=0; i<num_dcs; i++) {
541 char addr[INET6_ADDRSTRLEN];
542 int ret;
544 print_sockaddr(addr, sizeof(addr), &dc_addrs[i]);
546 ret = tsocket_address_inet_from_strings(dc_addrs2, "ip",
547 addr, LDAP_PORT,
548 &dc_addrs2[i]);
549 if (ret != 0) {
550 status = map_nt_error_from_unix(errno);
551 DEBUG(2,("Failed to create tsocket_address for %s - %s\n",
552 addr, nt_errstr(status)));
553 goto out;
557 dc_addrs3 = (const struct tsocket_address * const *)dc_addrs2;
559 status = cldap_multi_netlogon(talloc_tos(),
560 dc_addrs3, num_dcs,
561 realm, lp_netbios_name(),
562 NETLOGON_NT_VERSION_5 | NETLOGON_NT_VERSION_5EX,
563 MIN(num_dcs, 3), timeval_current_ofs(3, 0), &responses);
564 TALLOC_FREE(dc_addrs2);
565 dc_addrs3 = NULL;
567 if (!NT_STATUS_IS_OK(status)) {
568 DEBUG(10,("get_kdc_ip_string: cldap_multi_netlogon failed: "
569 "%s\n", nt_errstr(status)));
570 goto out;
573 for (i=0; i<num_dcs; i++) {
574 char *new_kdc_str;
576 if (responses[i] == NULL) {
577 continue;
580 /* Append to the string - inefficient but not done often. */
581 new_kdc_str = talloc_asprintf_append(
582 kdc_str,
583 "\t\tkdc = %s\n",
584 print_canonical_sockaddr_with_port(
585 mem_ctx, &dc_addrs[i]));
586 if (new_kdc_str == NULL) {
587 goto out;
589 kdc_str = new_kdc_str;
592 result = talloc_move(mem_ctx, &kdc_str);
593 out:
594 if (result != NULL) {
595 DBG_DEBUG("Returning\n%s\n", result);
596 } else {
597 DBG_NOTICE("Failed to get KDC ip address\n");
600 TALLOC_FREE(frame);
601 return result;
604 /************************************************************************
605 Create a specific krb5.conf file in the private directory pointing
606 at a specific kdc for a realm. Keyed off domain name. Sets
607 KRB5_CONFIG environment variable to point to this file. Must be
608 run as root or will fail (which is a good thing :-).
609 ************************************************************************/
611 #if !defined(SAMBA4_USES_HEIMDAL) /* MIT version */
612 static char *get_enctypes(TALLOC_CTX *mem_ctx)
614 char *aes_enctypes = NULL;
615 const char *legacy_enctypes = "";
616 char *enctypes = NULL;
618 aes_enctypes = talloc_strdup(mem_ctx, "");
619 if (aes_enctypes == NULL) {
620 goto done;
623 if (lp_kerberos_encryption_types() == KERBEROS_ETYPES_ALL ||
624 lp_kerberos_encryption_types() == KERBEROS_ETYPES_STRONG) {
625 aes_enctypes = talloc_asprintf_append(
626 aes_enctypes, "%s", "aes256-cts-hmac-sha1-96 ");
627 if (aes_enctypes == NULL) {
628 goto done;
630 aes_enctypes = talloc_asprintf_append(
631 aes_enctypes, "%s", "aes128-cts-hmac-sha1-96");
632 if (aes_enctypes == NULL) {
633 goto done;
637 if (lp_weak_crypto() == SAMBA_WEAK_CRYPTO_ALLOWED &&
638 (lp_kerberos_encryption_types() == KERBEROS_ETYPES_ALL ||
639 lp_kerberos_encryption_types() == KERBEROS_ETYPES_LEGACY)) {
640 legacy_enctypes = "RC4-HMAC";
643 enctypes =
644 talloc_asprintf(mem_ctx, "\tdefault_tgs_enctypes = %s %s\n"
645 "\tdefault_tkt_enctypes = %s %s\n"
646 "\tpreferred_enctypes = %s %s\n",
647 aes_enctypes, legacy_enctypes, aes_enctypes,
648 legacy_enctypes, aes_enctypes, legacy_enctypes);
649 done:
650 TALLOC_FREE(aes_enctypes);
651 return enctypes;
653 #else /* Heimdal version */
654 static char *get_enctypes(TALLOC_CTX *mem_ctx)
656 const char *aes_enctypes = "";
657 const char *legacy_enctypes = "";
658 char *enctypes = NULL;
660 if (lp_kerberos_encryption_types() == KERBEROS_ETYPES_ALL ||
661 lp_kerberos_encryption_types() == KERBEROS_ETYPES_STRONG) {
662 aes_enctypes =
663 "aes256-cts-hmac-sha1-96 aes128-cts-hmac-sha1-96";
666 if (lp_kerberos_encryption_types() == KERBEROS_ETYPES_ALL ||
667 lp_kerberos_encryption_types() == KERBEROS_ETYPES_LEGACY) {
668 legacy_enctypes = "arcfour-hmac-md5";
671 enctypes = talloc_asprintf(mem_ctx, "\tdefault_etypes = %s %s\n",
672 aes_enctypes, legacy_enctypes);
674 return enctypes;
676 #endif
678 bool create_local_private_krb5_conf_for_domain(const char *realm,
679 const char *domain,
680 const char *sitename,
681 const struct sockaddr_storage *pss)
683 char *dname;
684 char *tmpname = NULL;
685 char *fname = NULL;
686 char *file_contents = NULL;
687 char *kdc_ip_string = NULL;
688 size_t flen = 0;
689 ssize_t ret;
690 int fd;
691 char *realm_upper = NULL;
692 bool result = false;
693 char *enctypes = NULL;
694 const char *include_system_krb5 = "";
695 mode_t mask;
697 if (!lp_create_krb5_conf()) {
698 return false;
701 if (realm == NULL) {
702 DEBUG(0, ("No realm has been specified! Do you really want to "
703 "join an Active Directory server?\n"));
704 return false;
707 if (domain == NULL || pss == NULL) {
708 return false;
711 dname = lock_path(talloc_tos(), "smb_krb5");
712 if (!dname) {
713 return false;
715 if ((mkdir(dname, 0755)==-1) && (errno != EEXIST)) {
716 DEBUG(0,("create_local_private_krb5_conf_for_domain: "
717 "failed to create directory %s. Error was %s\n",
718 dname, strerror(errno) ));
719 goto done;
722 tmpname = lock_path(talloc_tos(), "smb_tmp_krb5.XXXXXX");
723 if (!tmpname) {
724 goto done;
727 fname = talloc_asprintf(dname, "%s/krb5.conf.%s", dname, domain);
728 if (!fname) {
729 goto done;
732 DEBUG(10,("create_local_private_krb5_conf_for_domain: fname = %s, realm = %s, domain = %s\n",
733 fname, realm, domain ));
735 realm_upper = talloc_strdup(fname, realm);
736 if (!strupper_m(realm_upper)) {
737 goto done;
740 kdc_ip_string = get_kdc_ip_string(dname, realm, sitename, pss);
741 if (!kdc_ip_string) {
742 goto done;
745 enctypes = get_enctypes(fname);
746 if (enctypes == NULL) {
747 goto done;
750 #if !defined(SAMBA4_USES_HEIMDAL)
751 if (lp_include_system_krb5_conf()) {
752 include_system_krb5 = "include /etc/krb5.conf";
754 #endif
757 * We are setting 'dns_lookup_kdc' to true, because we want to lookup
758 * KDCs which are not configured via DNS SRV records, eg. if we do:
760 * net ads join -Uadmin@otherdomain
762 file_contents =
763 talloc_asprintf(fname,
764 "[libdefaults]\n"
765 "\tdefault_realm = %s\n"
766 "%s"
767 "\tdns_lookup_realm = false\n"
768 "\tdns_lookup_kdc = true\n\n"
769 "[realms]\n\t%s = {\n"
770 "%s\t}\n"
771 "\t%s = {\n"
772 "%s\t}\n"
773 "%s\n",
774 realm_upper,
775 enctypes,
776 realm_upper,
777 kdc_ip_string,
778 domain,
779 kdc_ip_string,
780 include_system_krb5);
782 if (!file_contents) {
783 goto done;
786 flen = strlen(file_contents);
788 mask = umask(S_IRWXO | S_IRWXG);
789 fd = mkstemp(tmpname);
790 umask(mask);
791 if (fd == -1) {
792 DBG_ERR("mkstemp failed, for file %s. Errno %s\n",
793 tmpname,
794 strerror(errno));
795 goto done;
798 if (fchmod(fd, 0644)==-1) {
799 DEBUG(0,("create_local_private_krb5_conf_for_domain: fchmod failed for %s."
800 " Errno %s\n",
801 tmpname, strerror(errno) ));
802 unlink(tmpname);
803 close(fd);
804 goto done;
807 ret = write(fd, file_contents, flen);
808 if (flen != ret) {
809 DEBUG(0,("create_local_private_krb5_conf_for_domain: write failed,"
810 " returned %d (should be %u). Errno %s\n",
811 (int)ret, (unsigned int)flen, strerror(errno) ));
812 unlink(tmpname);
813 close(fd);
814 goto done;
816 if (close(fd)==-1) {
817 DEBUG(0,("create_local_private_krb5_conf_for_domain: close failed."
818 " Errno %s\n", strerror(errno) ));
819 unlink(tmpname);
820 goto done;
823 if (rename(tmpname, fname) == -1) {
824 DEBUG(0,("create_local_private_krb5_conf_for_domain: rename "
825 "of %s to %s failed. Errno %s\n",
826 tmpname, fname, strerror(errno) ));
827 unlink(tmpname);
828 goto done;
831 DBG_INFO("wrote file %s with realm %s KDC list:\n%s\n",
832 fname, realm_upper, kdc_ip_string);
834 /* Set the environment variable to this file. */
835 setenv("KRB5_CONFIG", fname, 1);
837 result = true;
839 #if defined(OVERWRITE_SYSTEM_KRB5_CONF)
841 #define SYSTEM_KRB5_CONF_PATH "/etc/krb5.conf"
842 /* Insanity, sheer insanity..... */
844 if (strequal(realm, lp_realm())) {
845 SMB_STRUCT_STAT sbuf;
847 if (sys_lstat(SYSTEM_KRB5_CONF_PATH, &sbuf, false) == 0) {
848 if (S_ISLNK(sbuf.st_ex_mode) && sbuf.st_ex_size) {
849 int lret;
850 size_t alloc_size = sbuf.st_ex_size + 1;
851 char *linkpath = talloc_array(talloc_tos(), char,
852 alloc_size);
853 if (!linkpath) {
854 goto done;
856 lret = readlink(SYSTEM_KRB5_CONF_PATH, linkpath,
857 alloc_size - 1);
858 if (lret == -1) {
859 TALLOC_FREE(linkpath);
860 goto done;
862 linkpath[lret] = '\0';
864 if (strcmp(linkpath, fname) == 0) {
865 /* Symlink already exists. */
866 TALLOC_FREE(linkpath);
867 goto done;
869 TALLOC_FREE(linkpath);
873 /* Try and replace with a symlink. */
874 if (symlink(fname, SYSTEM_KRB5_CONF_PATH) == -1) {
875 const char *newpath = SYSTEM_KRB5_CONF_PATH ".saved";
876 if (errno != EEXIST) {
877 DEBUG(0,("create_local_private_krb5_conf_for_domain: symlink "
878 "of %s to %s failed. Errno %s\n",
879 fname, SYSTEM_KRB5_CONF_PATH, strerror(errno) ));
880 goto done; /* Not a fatal error. */
883 /* Yes, this is a race conditon... too bad. */
884 if (rename(SYSTEM_KRB5_CONF_PATH, newpath) == -1) {
885 DEBUG(0,("create_local_private_krb5_conf_for_domain: rename "
886 "of %s to %s failed. Errno %s\n",
887 SYSTEM_KRB5_CONF_PATH, newpath,
888 strerror(errno) ));
889 goto done; /* Not a fatal error. */
892 if (symlink(fname, SYSTEM_KRB5_CONF_PATH) == -1) {
893 DEBUG(0,("create_local_private_krb5_conf_for_domain: "
894 "forced symlink of %s to /etc/krb5.conf failed. Errno %s\n",
895 fname, strerror(errno) ));
896 goto done; /* Not a fatal error. */
900 #endif
902 done:
903 TALLOC_FREE(tmpname);
904 TALLOC_FREE(dname);
906 return result;
908 #endif