xref: /minix3/external/bsd/bind/dist/bin/named/tsigconf.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: tsigconf.c,v 1.5 2014/12/10 04:37:52 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004-2007, 2009, 2011, 2012  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 1999-2001  Internet Software Consortium.
6*00b67f09SDavid van Moolenbroek  *
7*00b67f09SDavid van Moolenbroek  * Permission to use, copy, modify, and/or distribute this software for any
8*00b67f09SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
9*00b67f09SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
10*00b67f09SDavid van Moolenbroek  *
11*00b67f09SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12*00b67f09SDavid van Moolenbroek  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13*00b67f09SDavid van Moolenbroek  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14*00b67f09SDavid van Moolenbroek  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15*00b67f09SDavid van Moolenbroek  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16*00b67f09SDavid van Moolenbroek  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17*00b67f09SDavid van Moolenbroek  * PERFORMANCE OF THIS SOFTWARE.
18*00b67f09SDavid van Moolenbroek  */
19*00b67f09SDavid van Moolenbroek 
20*00b67f09SDavid van Moolenbroek /* Id: tsigconf.c,v 1.35 2011/01/11 23:47:12 tbox Exp  */
21*00b67f09SDavid van Moolenbroek 
22*00b67f09SDavid van Moolenbroek /*! \file */
23*00b67f09SDavid van Moolenbroek 
24*00b67f09SDavid van Moolenbroek #include <config.h>
25*00b67f09SDavid van Moolenbroek 
26*00b67f09SDavid van Moolenbroek #include <isc/base64.h>
27*00b67f09SDavid van Moolenbroek #include <isc/buffer.h>
28*00b67f09SDavid van Moolenbroek #include <isc/mem.h>
29*00b67f09SDavid van Moolenbroek #include <isc/string.h>
30*00b67f09SDavid van Moolenbroek 
31*00b67f09SDavid van Moolenbroek #include <isccfg/cfg.h>
32*00b67f09SDavid van Moolenbroek 
33*00b67f09SDavid van Moolenbroek #include <dns/tsig.h>
34*00b67f09SDavid van Moolenbroek #include <dns/result.h>
35*00b67f09SDavid van Moolenbroek 
36*00b67f09SDavid van Moolenbroek #include <named/log.h>
37*00b67f09SDavid van Moolenbroek 
38*00b67f09SDavid van Moolenbroek #include <named/config.h>
39*00b67f09SDavid van Moolenbroek #include <named/tsigconf.h>
40*00b67f09SDavid van Moolenbroek 
41*00b67f09SDavid van Moolenbroek static isc_result_t
add_initial_keys(const cfg_obj_t * list,dns_tsig_keyring_t * ring,isc_mem_t * mctx)42*00b67f09SDavid van Moolenbroek add_initial_keys(const cfg_obj_t *list, dns_tsig_keyring_t *ring,
43*00b67f09SDavid van Moolenbroek 		 isc_mem_t *mctx)
44*00b67f09SDavid van Moolenbroek {
45*00b67f09SDavid van Moolenbroek 	dns_tsigkey_t *tsigkey = NULL;
46*00b67f09SDavid van Moolenbroek 	const cfg_listelt_t *element;
47*00b67f09SDavid van Moolenbroek 	const cfg_obj_t *key = NULL;
48*00b67f09SDavid van Moolenbroek 	const char *keyid = NULL;
49*00b67f09SDavid van Moolenbroek 	unsigned char *secret = NULL;
50*00b67f09SDavid van Moolenbroek 	int secretalloc = 0;
51*00b67f09SDavid van Moolenbroek 	int secretlen = 0;
52*00b67f09SDavid van Moolenbroek 	isc_result_t ret;
53*00b67f09SDavid van Moolenbroek 	isc_stdtime_t now;
54*00b67f09SDavid van Moolenbroek 	isc_uint16_t bits;
55*00b67f09SDavid van Moolenbroek 
56*00b67f09SDavid van Moolenbroek 	for (element = cfg_list_first(list);
57*00b67f09SDavid van Moolenbroek 	     element != NULL;
58*00b67f09SDavid van Moolenbroek 	     element = cfg_list_next(element))
59*00b67f09SDavid van Moolenbroek 	{
60*00b67f09SDavid van Moolenbroek 		const cfg_obj_t *algobj = NULL;
61*00b67f09SDavid van Moolenbroek 		const cfg_obj_t *secretobj = NULL;
62*00b67f09SDavid van Moolenbroek 		dns_name_t keyname;
63*00b67f09SDavid van Moolenbroek 		dns_name_t *alg;
64*00b67f09SDavid van Moolenbroek 		const char *algstr;
65*00b67f09SDavid van Moolenbroek 		char keynamedata[1024];
66*00b67f09SDavid van Moolenbroek 		isc_buffer_t keynamesrc, keynamebuf;
67*00b67f09SDavid van Moolenbroek 		const char *secretstr;
68*00b67f09SDavid van Moolenbroek 		isc_buffer_t secretbuf;
69*00b67f09SDavid van Moolenbroek 
70*00b67f09SDavid van Moolenbroek 		key = cfg_listelt_value(element);
71*00b67f09SDavid van Moolenbroek 		keyid = cfg_obj_asstring(cfg_map_getname(key));
72*00b67f09SDavid van Moolenbroek 
73*00b67f09SDavid van Moolenbroek 		algobj = NULL;
74*00b67f09SDavid van Moolenbroek 		secretobj = NULL;
75*00b67f09SDavid van Moolenbroek 		(void)cfg_map_get(key, "algorithm", &algobj);
76*00b67f09SDavid van Moolenbroek 		(void)cfg_map_get(key, "secret", &secretobj);
77*00b67f09SDavid van Moolenbroek 		INSIST(algobj != NULL && secretobj != NULL);
78*00b67f09SDavid van Moolenbroek 
79*00b67f09SDavid van Moolenbroek 		/*
80*00b67f09SDavid van Moolenbroek 		 * Create the key name.
81*00b67f09SDavid van Moolenbroek 		 */
82*00b67f09SDavid van Moolenbroek 		dns_name_init(&keyname, NULL);
83*00b67f09SDavid van Moolenbroek 		isc_buffer_constinit(&keynamesrc, keyid, strlen(keyid));
84*00b67f09SDavid van Moolenbroek 		isc_buffer_add(&keynamesrc, strlen(keyid));
85*00b67f09SDavid van Moolenbroek 		isc_buffer_init(&keynamebuf, keynamedata, sizeof(keynamedata));
86*00b67f09SDavid van Moolenbroek 		ret = dns_name_fromtext(&keyname, &keynamesrc, dns_rootname,
87*00b67f09SDavid van Moolenbroek 					DNS_NAME_DOWNCASE, &keynamebuf);
88*00b67f09SDavid van Moolenbroek 		if (ret != ISC_R_SUCCESS)
89*00b67f09SDavid van Moolenbroek 			goto failure;
90*00b67f09SDavid van Moolenbroek 
91*00b67f09SDavid van Moolenbroek 		/*
92*00b67f09SDavid van Moolenbroek 		 * Create the algorithm.
93*00b67f09SDavid van Moolenbroek 		 */
94*00b67f09SDavid van Moolenbroek 		algstr = cfg_obj_asstring(algobj);
95*00b67f09SDavid van Moolenbroek 		if (ns_config_getkeyalgorithm(algstr, &alg, &bits)
96*00b67f09SDavid van Moolenbroek 		    != ISC_R_SUCCESS) {
97*00b67f09SDavid van Moolenbroek 			cfg_obj_log(algobj, ns_g_lctx, ISC_LOG_ERROR,
98*00b67f09SDavid van Moolenbroek 				    "key '%s': has a unsupported algorithm '%s'",
99*00b67f09SDavid van Moolenbroek 				    keyid, algstr);
100*00b67f09SDavid van Moolenbroek 			ret = DNS_R_BADALG;
101*00b67f09SDavid van Moolenbroek 			goto failure;
102*00b67f09SDavid van Moolenbroek 		}
103*00b67f09SDavid van Moolenbroek 
104*00b67f09SDavid van Moolenbroek 		secretstr = cfg_obj_asstring(secretobj);
105*00b67f09SDavid van Moolenbroek 		secretalloc = secretlen = strlen(secretstr) * 3 / 4;
106*00b67f09SDavid van Moolenbroek 		secret = isc_mem_get(mctx, secretlen);
107*00b67f09SDavid van Moolenbroek 		if (secret == NULL) {
108*00b67f09SDavid van Moolenbroek 			ret = ISC_R_NOMEMORY;
109*00b67f09SDavid van Moolenbroek 			goto failure;
110*00b67f09SDavid van Moolenbroek 		}
111*00b67f09SDavid van Moolenbroek 		isc_buffer_init(&secretbuf, secret, secretlen);
112*00b67f09SDavid van Moolenbroek 		ret = isc_base64_decodestring(secretstr, &secretbuf);
113*00b67f09SDavid van Moolenbroek 		if (ret != ISC_R_SUCCESS)
114*00b67f09SDavid van Moolenbroek 			goto failure;
115*00b67f09SDavid van Moolenbroek 		secretlen = isc_buffer_usedlength(&secretbuf);
116*00b67f09SDavid van Moolenbroek 
117*00b67f09SDavid van Moolenbroek 		isc_stdtime_get(&now);
118*00b67f09SDavid van Moolenbroek 		ret = dns_tsigkey_create(&keyname, alg, secret, secretlen,
119*00b67f09SDavid van Moolenbroek 					 ISC_FALSE, NULL, now, now,
120*00b67f09SDavid van Moolenbroek 					 mctx, ring, &tsigkey);
121*00b67f09SDavid van Moolenbroek 		isc_mem_put(mctx, secret, secretalloc);
122*00b67f09SDavid van Moolenbroek 		secret = NULL;
123*00b67f09SDavid van Moolenbroek 		if (ret != ISC_R_SUCCESS)
124*00b67f09SDavid van Moolenbroek 			goto failure;
125*00b67f09SDavid van Moolenbroek 		/*
126*00b67f09SDavid van Moolenbroek 		 * Set digest bits.
127*00b67f09SDavid van Moolenbroek 		 */
128*00b67f09SDavid van Moolenbroek 		dst_key_setbits(tsigkey->key, bits);
129*00b67f09SDavid van Moolenbroek 		dns_tsigkey_detach(&tsigkey);
130*00b67f09SDavid van Moolenbroek 	}
131*00b67f09SDavid van Moolenbroek 
132*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
133*00b67f09SDavid van Moolenbroek 
134*00b67f09SDavid van Moolenbroek  failure:
135*00b67f09SDavid van Moolenbroek 	cfg_obj_log(key, ns_g_lctx, ISC_LOG_ERROR,
136*00b67f09SDavid van Moolenbroek 		    "configuring key '%s': %s", keyid,
137*00b67f09SDavid van Moolenbroek 		    isc_result_totext(ret));
138*00b67f09SDavid van Moolenbroek 
139*00b67f09SDavid van Moolenbroek 	if (secret != NULL)
140*00b67f09SDavid van Moolenbroek 		isc_mem_put(mctx, secret, secretalloc);
141*00b67f09SDavid van Moolenbroek 	return (ret);
142*00b67f09SDavid van Moolenbroek }
143*00b67f09SDavid van Moolenbroek 
144*00b67f09SDavid van Moolenbroek isc_result_t
ns_tsigkeyring_fromconfig(const cfg_obj_t * config,const cfg_obj_t * vconfig,isc_mem_t * mctx,dns_tsig_keyring_t ** ringp)145*00b67f09SDavid van Moolenbroek ns_tsigkeyring_fromconfig(const cfg_obj_t *config, const cfg_obj_t *vconfig,
146*00b67f09SDavid van Moolenbroek 			  isc_mem_t *mctx, dns_tsig_keyring_t **ringp)
147*00b67f09SDavid van Moolenbroek {
148*00b67f09SDavid van Moolenbroek 	const cfg_obj_t *maps[3];
149*00b67f09SDavid van Moolenbroek 	const cfg_obj_t *keylist;
150*00b67f09SDavid van Moolenbroek 	dns_tsig_keyring_t *ring = NULL;
151*00b67f09SDavid van Moolenbroek 	isc_result_t result;
152*00b67f09SDavid van Moolenbroek 	int i;
153*00b67f09SDavid van Moolenbroek 
154*00b67f09SDavid van Moolenbroek 	REQUIRE(ringp != NULL && *ringp == NULL);
155*00b67f09SDavid van Moolenbroek 
156*00b67f09SDavid van Moolenbroek 	i = 0;
157*00b67f09SDavid van Moolenbroek 	if (config != NULL)
158*00b67f09SDavid van Moolenbroek 		maps[i++] = config;
159*00b67f09SDavid van Moolenbroek 	if (vconfig != NULL)
160*00b67f09SDavid van Moolenbroek 		maps[i++] = cfg_tuple_get(vconfig, "options");
161*00b67f09SDavid van Moolenbroek 	maps[i] = NULL;
162*00b67f09SDavid van Moolenbroek 
163*00b67f09SDavid van Moolenbroek 	result = dns_tsigkeyring_create(mctx, &ring);
164*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
165*00b67f09SDavid van Moolenbroek 		return (result);
166*00b67f09SDavid van Moolenbroek 
167*00b67f09SDavid van Moolenbroek 	for (i = 0; ; i++) {
168*00b67f09SDavid van Moolenbroek 		if (maps[i] == NULL)
169*00b67f09SDavid van Moolenbroek 			break;
170*00b67f09SDavid van Moolenbroek 		keylist = NULL;
171*00b67f09SDavid van Moolenbroek 		result = cfg_map_get(maps[i], "key", &keylist);
172*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
173*00b67f09SDavid van Moolenbroek 			continue;
174*00b67f09SDavid van Moolenbroek 		result = add_initial_keys(keylist, ring, mctx);
175*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
176*00b67f09SDavid van Moolenbroek 			goto failure;
177*00b67f09SDavid van Moolenbroek 	}
178*00b67f09SDavid van Moolenbroek 
179*00b67f09SDavid van Moolenbroek 	*ringp = ring;
180*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
181*00b67f09SDavid van Moolenbroek 
182*00b67f09SDavid van Moolenbroek  failure:
183*00b67f09SDavid van Moolenbroek 	dns_tsigkeyring_detach(&ring);
184*00b67f09SDavid van Moolenbroek 	return (result);
185*00b67f09SDavid van Moolenbroek }
186