s3:libads: Fix code spelling
[samba.git] / source3 / libads / kerberos.c
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.
9
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.
14
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.
19
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/>.
22 */
23
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"
34
35 #ifdef HAVE_KRB5
36
37 #define LIBADS_CCACHE_NAME "MEMORY:libads"
38
39 /*
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 ...
43 */
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[])
50 {
51         if (num_prompts == 0) return 0;
52         if (num_prompts == 2) {
53                 /*
54                  * only heimdal has a prompt type and we need to deal with it here to
55                  * avoid loops.
56                  *
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.
60                  */
61
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) {
65                         /*
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
74                          */
75                         return KRB5KDC_ERR_KEY_EXPIRED;
76                 }
77 #elif defined(HAVE_KRB5_GET_PROMPT_TYPES) /* MIT */
78                 krb5_prompt_type *prompt_types = NULL;
79
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;
85                         }
86                 }
87 #endif
88         }
89
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;
98                 }
99         }
100         return 0;
101 }
102
103 /*
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
107 */
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)
121 {
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;
133
134         ZERO_STRUCT(my_creds);
135
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;
142         }
143
144         if (time_offset != 0) {
145                 krb5_set_real_time(ctx, time(NULL) + time_offset, 0);
146         }
147
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"));
152
153         if ((code = krb5_cc_resolve(ctx, cache_name ? cache_name : krb5_cc_default_name(ctx), &cc))) {
154                 goto out;
155         }
156
157         if ((code = smb_krb5_parse_name(ctx, given_principal, &me))) {
158                 goto out;
159         }
160
161         if ((code = krb5_get_init_creds_opt_alloc(ctx, &opt))) {
162                 goto out;
163         }
164
165         krb5_get_init_creds_opt_set_renew_life(opt, renewable_time);
166         krb5_get_init_creds_opt_set_forwardable(opt, True);
167
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
179
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;
184                 }
185         }
186 #endif
187         if (add_netbios_addr) {
188                 if ((code = smb_krb5_gen_netbios_krb5_address(&addr,
189                                                         lp_netbios_name()))) {
190                         goto out;
191                 }
192                 krb5_get_init_creds_opt_set_address_list(opt, addr->addrs);
193         }
194
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;
199         }
200
201         canon_princ = my_creds.client;
202
203         code = smb_krb5_unparse_name(frame,
204                                      ctx,
205                                      canon_princ,
206                                      &canon_principal);
207         if (code != 0) {
208                 goto out;
209         }
210
211         DBG_DEBUG("%s mapped to %s\n", given_principal, canon_principal);
212
213         canon_realm = smb_krb5_principal_get_realm(frame, ctx, canon_princ);
214         if (canon_realm == NULL) {
215                 code = ENOMEM;
216                 goto out;
217         }
218
219         if ((code = krb5_cc_initialize(ctx, cc, canon_princ))) {
220                 goto out;
221         }
222
223         if ((code = krb5_cc_store_cred(ctx, cc, &my_creds))) {
224                 goto out;
225         }
226
227         if (expire_time) {
228                 *expire_time = (time_t) my_creds.times.endtime;
229         }
230
231         if (renew_till_time) {
232                 *renew_till_time = (time_t) my_creds.times.renew_till;
233         }
234
235         if (_canon_principal != NULL) {
236                 *_canon_principal = talloc_move(mem_ctx, &canon_principal);
237         }
238         if (_canon_realm != NULL) {
239                 *_canon_realm = talloc_move(mem_ctx, &canon_realm);
240         }
241  out:
242         if (ntstatus) {
243                 /* fast path */
244                 if (code == 0) {
245                         *ntstatus = NT_STATUS_OK;
246                         goto cleanup;
247                 }
248
249                 /* fall back to self-made-mapping */
250                 *ntstatus = krb5_to_nt_status(code);
251         }
252
253  cleanup:
254         krb5_free_cred_contents(ctx, &my_creds);
255         if (me) {
256                 krb5_free_principal(ctx, me);
257         }
258         if (addr) {
259                 smb_krb5_free_addresses(ctx, addr);
260         }
261         if (opt) {
262                 krb5_get_init_creds_opt_free(ctx, opt);
263         }
264         if (cc) {
265                 krb5_cc_close(ctx, cc);
266         }
267         if (ctx) {
268                 krb5_free_context(ctx);
269         }
270         TALLOC_FREE(frame);
271         return code;
272 }
273
274 int ads_kdestroy(const char *cc_name)
275 {
276         krb5_error_code code;
277         krb5_context ctx = NULL;
278         krb5_ccache cc = NULL;
279
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;
285         }
286
287         if (!cc_name) {
288                 if ((code = krb5_cc_default(ctx, &cc))) {
289                         krb5_free_context(ctx);
290                         return code;
291                 }
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;
298                 }
299         }
300
301         if ((code = krb5_cc_destroy (ctx, cc))) {
302                 DEBUG(3, ("ads_kdestroy: krb5_cc_destroy failed: %s\n",
303                         error_message(code)));
304         }
305
306         krb5_free_context (ctx);
307         return code;
308 }
309
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)
317 {
318         int ret;
319         /*
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.
323          */
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;
328                 }
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;
333         }
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;
341 }
342
343 /************************************************************************
344 ************************************************************************/
345
346 int kerberos_kinit_password(const char *principal,
347                             const char *password,
348                             int time_offset,
349                             const char *cache_name)
350 {
351         return kerberos_kinit_password_ext(principal,
352                                            password,
353                                            time_offset,
354                                            0,
355                                            0,
356                                            cache_name,
357                                            False,
358                                            False,
359                                            0,
360                                            NULL,
361                                            NULL,
362                                            NULL,
363                                            NULL);
364 }
365
366 /************************************************************************
367 ************************************************************************/
368
369 /************************************************************************
370  Create a string list of available kdc's, possibly searching by sitename.
371  Does DNS queries.
372
373  If "sitename" is given, the DC's in that site are listed first.
374
375 ************************************************************************/
376
377 static void add_sockaddr_unique(struct sockaddr_storage *addrs, size_t *num_addrs,
378                                 const struct sockaddr_storage *addr)
379 {
380         size_t i;
381
382         for (i=0; i<*num_addrs; i++) {
383                 if (sockaddr_equal((const struct sockaddr *)&addrs[i],
384                                    (const struct sockaddr *)addr)) {
385                         return;
386                 }
387         }
388         addrs[i] = *addr;
389         *num_addrs += 1;
390 }
391
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 */
396
397 static char *print_canonical_sockaddr_with_port(TALLOC_CTX *mem_ctx,
398                                                 const struct sockaddr_storage *pss)
399 {
400         char *str = NULL;
401
402         str = print_canonical_sockaddr(mem_ctx, pss);
403         if (str == NULL) {
404                 return NULL;
405         }
406
407         if (pss->ss_family != AF_INET6) {
408                 return str;
409         }
410
411 #if defined(HAVE_IPV6)
412         str = talloc_asprintf_append(str, ":88");
413 #endif
414         return str;
415 }
416
417 static char *get_kdc_ip_string(char *mem_ctx,
418                 const char *realm,
419                 const char *sitename,
420                 const struct sockaddr_storage *pss)
421 {
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;
439
440         SMB_ASSERT(pss != NULL);
441
442         canon_sockaddr = print_canonical_sockaddr_with_port(frame, pss);
443         if (canon_sockaddr == NULL) {
444                 goto out;
445         }
446
447         kdc_str = talloc_asprintf(frame,
448                                   "\t\tkdc = %s\n",
449                                   canon_sockaddr);
450         if (kdc_str == NULL) {
451                 goto out;
452         }
453
454         ok = sockaddr_storage_to_samba_sockaddr(&sa, pss);
455         if (!ok) {
456                 goto out;
457         }
458
459         /*
460          * First get the KDC's only in this site, the rest will be
461          * appended later
462          */
463
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;
474                 }
475                 DBG_DEBUG("got %zu addresses from site %s search\n",
476                         count_site,
477                         sitename);
478         }
479
480         /* Get all KDC's. */
481
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;
491         }
492         DBG_DEBUG("got %zu addresses from site-less search\n", count_nonsite);
493
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;
498         }
499
500
501         dc_addrs = talloc_array(talloc_tos(), struct sockaddr_storage,
502                                 count_site + count_nonsite);
503         if (dc_addrs == NULL) {
504                 goto out;
505         }
506
507         num_dcs = 0;
508
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);
513                 }
514         }
515
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);
520                 }
521         }
522
523         DBG_DEBUG("%zu additional KDCs to test\n", num_dcs);
524         if (num_dcs == 0) {
525                 /*
526                  * We do not have additional KDCs, but we have the one passed
527                  * in via `pss`. So just use that one and leave.
528                  */
529                 result = talloc_move(mem_ctx, &kdc_str);
530                 goto out;
531         }
532
533         dc_addrs2 = talloc_zero_array(talloc_tos(),
534                                       struct tsocket_address *,
535                                       num_dcs);
536         if (dc_addrs2 == NULL) {
537                 goto out;
538         }
539
540         for (i=0; i<num_dcs; i++) {
541                 char addr[INET6_ADDRSTRLEN];
542                 int ret;
543
544                 print_sockaddr(addr, sizeof(addr), &dc_addrs[i]);
545
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;
554                 }
555         }
556
557         dc_addrs3 = (const struct tsocket_address * const *)dc_addrs2;
558
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;
566
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;
571         }
572
573         for (i=0; i<num_dcs; i++) {
574                 char *new_kdc_str;
575
576                 if (responses[i] == NULL) {
577                         continue;
578                 }
579
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;
588                 }
589                 kdc_str = new_kdc_str;
590         }
591
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");
598         }
599
600         TALLOC_FREE(frame);
601         return result;
602 }
603
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 ************************************************************************/
610
611 #if !defined(SAMBA4_USES_HEIMDAL) /* MIT version */
612 static char *get_enctypes(TALLOC_CTX *mem_ctx)
613 {
614         char *aes_enctypes = NULL;
615         const char *legacy_enctypes = "";
616         char *enctypes = NULL;
617
618         aes_enctypes = talloc_strdup(mem_ctx, "");
619         if (aes_enctypes == NULL) {
620                 goto done;
621         }
622
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;
629                 }
630                 aes_enctypes = talloc_asprintf_append(
631                     aes_enctypes, "%s", "aes128-cts-hmac-sha1-96");
632                 if (aes_enctypes == NULL) {
633                         goto done;
634                 }
635         }
636
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";
641         }
642
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;
652 }
653 #else /* Heimdal version */
654 static char *get_enctypes(TALLOC_CTX *mem_ctx)
655 {
656         const char *aes_enctypes = "";
657         const char *legacy_enctypes = "";
658         char *enctypes = NULL;
659
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";
664         }
665
666         if (lp_kerberos_encryption_types() == KERBEROS_ETYPES_ALL ||
667             lp_kerberos_encryption_types() == KERBEROS_ETYPES_LEGACY) {
668                 legacy_enctypes = "arcfour-hmac-md5";
669         }
670
671         enctypes = talloc_asprintf(mem_ctx, "\tdefault_etypes = %s %s\n",
672                                    aes_enctypes, legacy_enctypes);
673
674         return enctypes;
675 }
676 #endif
677
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)
682 {
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;
696
697         if (!lp_create_krb5_conf()) {
698                 return false;
699         }
700
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;
705         }
706
707         if (domain == NULL || pss == NULL) {
708                 return false;
709         }
710
711         dname = lock_path(talloc_tos(), "smb_krb5");
712         if (!dname) {
713                 return false;
714         }
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;
720         }
721
722         tmpname = lock_path(talloc_tos(), "smb_tmp_krb5.XXXXXX");
723         if (!tmpname) {
724                 goto done;
725         }
726
727         fname = talloc_asprintf(dname, "%s/krb5.conf.%s", dname, domain);
728         if (!fname) {
729                 goto done;
730         }
731
732         DEBUG(10,("create_local_private_krb5_conf_for_domain: fname = %s, realm = %s, domain = %s\n",
733                 fname, realm, domain ));
734
735         realm_upper = talloc_strdup(fname, realm);
736         if (!strupper_m(realm_upper)) {
737                 goto done;
738         }
739
740         kdc_ip_string = get_kdc_ip_string(dname, realm, sitename, pss);
741         if (!kdc_ip_string) {
742                 goto done;
743         }
744
745         enctypes = get_enctypes(fname);
746         if (enctypes == NULL) {
747                 goto done;
748         }
749
750 #if !defined(SAMBA4_USES_HEIMDAL)
751         if (lp_include_system_krb5_conf()) {
752                 include_system_krb5 = "include /etc/krb5.conf";
753         }
754 #endif
755
756         /*
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:
759          *
760          *     net ads join -Uadmin@otherdomain
761          */
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);
781
782         if (!file_contents) {
783                 goto done;
784         }
785
786         flen = strlen(file_contents);
787
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;
796         }
797
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;
805         }
806
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;
815         }
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;
821         }
822
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;
829         }
830
831         DBG_INFO("wrote file %s with realm %s KDC list:\n%s\n",
832                  fname, realm_upper, kdc_ip_string);
833
834         /* Set the environment variable to this file. */
835         setenv("KRB5_CONFIG", fname, 1);
836
837         result = true;
838
839 #if defined(OVERWRITE_SYSTEM_KRB5_CONF)
840
841 #define SYSTEM_KRB5_CONF_PATH "/etc/krb5.conf"
842         /* Insanity, sheer insanity..... */
843
844         if (strequal(realm, lp_realm())) {
845                 SMB_STRUCT_STAT sbuf;
846
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;
855                                 }
856                                 lret = readlink(SYSTEM_KRB5_CONF_PATH, linkpath,
857                                                 alloc_size - 1);
858                                 if (lret == -1) {
859                                         TALLOC_FREE(linkpath);
860                                         goto done;
861                                 }
862                                 linkpath[lret] = '\0';
863
864                                 if (strcmp(linkpath, fname) == 0) {
865                                         /* Symlink already exists. */
866                                         TALLOC_FREE(linkpath);
867                                         goto done;
868                                 }
869                                 TALLOC_FREE(linkpath);
870                         }
871                 }
872
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. */
881                         }
882
883                         /* Yes, this is a race condition... 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. */
890                         }
891
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. */
897                         }
898                 }
899         }
900 #endif
901
902 done:
903         TALLOC_FREE(tmpname);
904         TALLOC_FREE(dname);
905
906         return result;
907 }
908 #endif