4 * Copyright (C) 2002 Stichting NLnet, Netherlands, stichting@nlnet.nl.
6 * Permission to use, copy, modify, and distribute this software for any
7 * purpose with or without fee is hereby granted, provided that the
8 * above copyright notice and this permission notice appear in all
11 * THE SOFTWARE IS PROVIDED "AS IS" AND STICHTING NLNET
12 * DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
13 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL
14 * STICHTING NLNET BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR
15 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
16 * OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
17 * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE
18 * USE OR PERFORMANCE OF THIS SOFTWARE.
20 * The development of Dynamically Loadable Zones (DLZ) for Bind 9 was
21 * conceived and contributed by Rob Butler.
23 * Permission to use, copy, modify, and distribute this software for any
24 * purpose with or without fee is hereby granted, provided that the
25 * above copyright notice and this permission notice appear in all
28 * THE SOFTWARE IS PROVIDED "AS IS" AND ROB BUTLER
29 * DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
30 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL
31 * ROB BUTLER BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR
32 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
33 * OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
34 * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE
35 * USE OR PERFORMANCE OF THIS SOFTWARE.
39 * Copyright (C) 1999-2001 Internet Software Consortium.
41 * Permission to use, copy, modify, and distribute this software for any
42 * purpose with or without fee is hereby granted, provided that the above
43 * copyright notice and this permission notice appear in all copies.
45 * THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM
46 * DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
47 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL
48 * INTERNET SOFTWARE CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT,
49 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING
50 * FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
51 * NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
52 * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
64 #include <dns/result.h>
67 #include <isc/platform.h>
68 #include <isc/print.h>
69 #include <isc/result.h>
70 #include <isc/string.h>
73 #include <named/globals.h>
75 #include <dlz/sdlz_helper.h>
76 #include <dlz/dlz_mysql_driver.h>
80 static dns_sdlzimplementation_t
*dlz_mysql
= NULL
;
82 #define dbc_search_limit 30
90 #define safeGet(in) in == NULL ? "" : in
97 * Allocates memory for a new string, and then constructs the new
98 * string by "escaping" the input string. The new string is
99 * safe to be used in queries. This is necessary because we cannot
100 * be sure of what types of strings are passed to us, and we don't
101 * want special characters in the string causing problems.
105 mysqldrv_escape_string(MYSQL
*mysql
, const char *instr
) {
115 outstr
= isc_mem_allocate(ns_g_mctx
,(2 * len
* sizeof(char)) + 1);
119 mysql_real_escape_string(mysql
, outstr
, instr
, len
);
125 * This function is the real core of the driver. Zone, record
126 * and client strings are passed in (or NULL is passed if the
127 * string is not available). The type of query we want to run
128 * is indicated by the query flag, and the dbdata object is passed
129 * passed in to. dbdata really holds a single database instance.
130 * The function will construct and run the query, hopefully getting
135 mysql_get_resultset(const char *zone
, const char *record
,
136 const char *client
, unsigned int query
,
137 void *dbdata
, MYSQL_RES
**rs
)
140 dbinstance_t
*dbi
= NULL
;
141 char *querystring
= NULL
;
146 if (query
!= COUNTZONE
)
147 REQUIRE(*rs
== NULL
);
151 /* get db instance / connection */
152 dbi
= (dbinstance_t
*) dbdata
;
154 /* if DBI is null, can't do anything else */
156 result
= ISC_R_FAILURE
;
160 /* what type of query are we going to run? */
164 * if the query was not passed in from the config file
165 * then we can't run it. return not_implemented, so
166 * it's like the code for that operation was never
167 * built into the driver.... AHHH flexibility!!!
169 if (dbi
->allnodes_q
== NULL
) {
170 result
= ISC_R_NOTIMPLEMENTED
;
175 /* same as comments as ALLNODES */
176 if (dbi
->allowxfr_q
== NULL
) {
177 result
= ISC_R_NOTIMPLEMENTED
;
182 /* same as comments as ALLNODES */
183 if (dbi
->authority_q
== NULL
) {
184 result
= ISC_R_NOTIMPLEMENTED
;
189 /* this is required. It's the whole point of DLZ! */
190 if (dbi
->findzone_q
== NULL
) {
191 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
192 DNS_LOGMODULE_DLZ
, ISC_LOG_DEBUG(2),
193 "No query specified for findzone. "
194 "Findzone requires a query");
195 result
= ISC_R_FAILURE
;
200 /* same as comments as ALLNODES */
201 if (dbi
->countzone_q
== NULL
) {
202 result
= ISC_R_NOTIMPLEMENTED
;
207 /* this is required. It's also a major point of DLZ! */
208 if (dbi
->lookup_q
== NULL
) {
209 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
210 DNS_LOGMODULE_DLZ
, ISC_LOG_DEBUG(2),
211 "No query specified for lookup. "
212 "Lookup requires a query");
213 result
= ISC_R_FAILURE
;
219 * this should never happen. If it does, the code is
222 UNEXPECTED_ERROR(__FILE__
, __LINE__
,
223 "Incorrect query flag passed to "
224 "mysql_get_resultset");
225 result
= ISC_R_UNEXPECTED
;
231 * was a zone string passed? If so, make it safe for use in
235 dbi
->zone
= mysqldrv_escape_string((MYSQL
*) dbi
->dbconn
,
237 if (dbi
->zone
== NULL
) {
238 result
= ISC_R_NOMEMORY
;
241 } else { /* no string passed, set the string pointer to NULL */
246 * was a record string passed? If so, make it safe for use in
249 if (record
!= NULL
) {
250 dbi
->record
= mysqldrv_escape_string((MYSQL
*) dbi
->dbconn
,
252 if (dbi
->record
== NULL
) {
253 result
= ISC_R_NOMEMORY
;
256 } else { /* no string passed, set the string pointer to NULL */
261 * was a client string passed? If so, make it safe for use in
264 if (client
!= NULL
) {
265 dbi
->client
= mysqldrv_escape_string((MYSQL
*) dbi
->dbconn
,
267 if (dbi
->client
== NULL
) {
268 result
= ISC_R_NOMEMORY
;
271 } else { /* no string passed, set the string pointer to NULL */
276 * what type of query are we going to run? this time we build
277 * the actual query to run.
281 querystring
= build_querystring(ns_g_mctx
, dbi
->allnodes_q
);
284 querystring
= build_querystring(ns_g_mctx
, dbi
->allowxfr_q
);
287 querystring
= build_querystring(ns_g_mctx
, dbi
->authority_q
);
290 querystring
= build_querystring(ns_g_mctx
, dbi
->findzone_q
);
293 querystring
= build_querystring(ns_g_mctx
, dbi
->countzone_q
);
296 querystring
= build_querystring(ns_g_mctx
, dbi
->lookup_q
);
300 * this should never happen. If it does, the code is
303 UNEXPECTED_ERROR(__FILE__
, __LINE__
,
304 "Incorrect query flag passed to "
305 "mysql_get_resultset");
306 result
= ISC_R_UNEXPECTED
;
310 /* if the querystring is null, Bummer, outta RAM. UPGRADE TIME!!! */
311 if (querystring
== NULL
) {
312 result
= ISC_R_NOMEMORY
;
317 * output the full query string during debug so we can see
318 * what lame error the query has.
320 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
321 DNS_LOGMODULE_DLZ
, ISC_LOG_DEBUG(1),
322 "\nQuery String: %s\n", querystring
);
324 /* attempt query up to 3 times. */
325 for (i
=0; i
< 3; i
++) {
326 qres
= mysql_query((MYSQL
*) dbi
->dbconn
, querystring
);
329 for (j
=0; mysql_ping((MYSQL
*) dbi
->dbconn
) != 0 && j
< 4; j
++)
334 result
= ISC_R_SUCCESS
;
335 if (query
!= COUNTZONE
) {
336 *rs
= mysql_store_result((MYSQL
*) dbi
->dbconn
);
338 result
= ISC_R_FAILURE
;
341 result
= ISC_R_FAILURE
;
346 /* it's always good to cleanup after yourself */
348 /* if we couldn't even get DBI, just return NULL */
350 return ISC_R_FAILURE
;
352 /* free dbi->zone string */
353 if (dbi
->zone
!= NULL
)
354 isc_mem_free(ns_g_mctx
, dbi
->zone
);
356 /* free dbi->record string */
357 if (dbi
->record
!= NULL
)
358 isc_mem_free(ns_g_mctx
, dbi
->record
);
360 /* free dbi->client string */
361 if (dbi
->client
!= NULL
)
362 isc_mem_free(ns_g_mctx
, dbi
->client
);
364 /* release query string */
365 if (querystring
!= NULL
)
366 isc_mem_free(ns_g_mctx
, querystring
);
373 * The processing of result sets for lookup and authority are
374 * exactly the same. So that functionality has been moved
375 * into this function to minimize code.
379 mysql_process_rs(dns_sdlzlookup_t
*lookup
, MYSQL_RES
*rs
)
381 isc_result_t result
= ISC_R_NOTFOUND
;
390 row
= mysql_fetch_row(rs
); /* get a row from the result set */
391 fields
= mysql_num_fields(rs
); /* how many columns in result set */
392 while (row
!= NULL
) {
396 * one column in rs, it's the data field. use
397 * default type of A record, and default TTL
400 result
= dns_sdlz_putrr(lookup
, "a", 86400,
405 * two columns, data field, and data type.
406 * use default TTL of 86400.
408 result
= dns_sdlz_putrr(lookup
, safeGet(row
[0]), 86400,
413 * three columns, all data no defaults.
414 * convert text to int, make sure it worked
417 ttl
= strtol(safeGet(row
[0]), &endp
, 10);
418 if (*endp
!= '\0' || ttl
< 0) {
419 isc_log_write(dns_lctx
,
420 DNS_LOGCATEGORY_DATABASE
,
421 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
422 "mysql driver ttl must be "
425 result
= dns_sdlz_putrr(lookup
, safeGet(row
[1]), ttl
,
430 * more than 3 fields, concatenate the last
431 * ones together. figure out how long to make
434 for (j
=2, len
=0; j
< fields
; j
++) {
435 len
+= strlen(safeGet(row
[j
])) + 1;
438 * allocate string memory, allow for NULL to
441 tmpString
= isc_mem_allocate(ns_g_mctx
, len
+ 1);
442 if (tmpString
== NULL
) {
443 /* major bummer, need more ram */
444 isc_log_write(dns_lctx
,
445 DNS_LOGCATEGORY_DATABASE
,
446 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
447 "mysql driver unable "
448 "to allocate memory for "
450 mysql_free_result(rs
);
451 return (ISC_R_FAILURE
); /* Yeah, I'd say! */
453 /* copy field to tmpString */
454 strcpy(tmpString
, safeGet(row
[2]));
458 * concat the rest of fields together, space
461 for (j
=3; j
< fields
; j
++) {
462 strcat(tmpString
, " ");
463 strcat(tmpString
, safeGet(row
[j
]));
465 /* convert text to int, make sure it worked right */
466 ttl
= strtol(safeGet(row
[0]), &endp
, 10);
467 if (*endp
!= '\0' || ttl
< 0) {
468 isc_log_write(dns_lctx
,
469 DNS_LOGCATEGORY_DATABASE
,
470 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
471 "mysql driver ttl must be "
474 /* ok, now tell Bind about it. */
475 result
= dns_sdlz_putrr(lookup
, safeGet(row
[1]),
477 /* done, get rid of this thing. */
478 isc_mem_free(ns_g_mctx
, tmpString
);
480 /* I sure hope we were successful */
481 if (result
!= ISC_R_SUCCESS
) {
482 /* nope, get rid of the Result set, and log a msg */
483 mysql_free_result(rs
);
484 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
485 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
486 "dns_sdlz_putrr returned error. "
487 "Error code was: %s",
488 isc_result_totext(result
));
489 return (ISC_R_FAILURE
);
491 row
= mysql_fetch_row(rs
); /* get next row */
494 /* free result set memory */
495 mysql_free_result(rs
);
497 /* return result code */
502 * SDLZ interface methods
505 /*% determine if the zone is supported by (in) the database */
508 mysql_findzone(void *driverarg
, void *dbdata
, const char *name
)
511 MYSQL_RES
*rs
= NULL
;
516 /* run the query and get the result set from the database. */
517 result
= mysql_get_resultset(name
, NULL
, NULL
, FINDZONE
, dbdata
, &rs
);
518 /* if we didn't get a result set, log an err msg. */
519 if (result
!= ISC_R_SUCCESS
|| rs
== NULL
) {
521 mysql_free_result(rs
);
522 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
523 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
524 "mysql driver unable to return "
525 "result set for findzone query");
526 return (ISC_R_FAILURE
);
528 /* count how many rows in result set */
529 rows
= mysql_num_rows(rs
);
530 /* get rid of result set, we are done with it. */
531 mysql_free_result(rs
);
533 /* if we returned any rows, zone is supported. */
535 mysql_get_resultset(name
, NULL
, NULL
, COUNTZONE
, dbdata
, NULL
);
536 return (ISC_R_SUCCESS
);
539 /* no rows returned, zone is not supported. */
540 return (ISC_R_NOTFOUND
);
543 /*% Determine if the client is allowed to perform a zone transfer */
545 mysql_allowzonexfr(void *driverarg
, void *dbdata
, const char *name
,
549 MYSQL_RES
*rs
= NULL
;
554 /* first check if the zone is supported by the database. */
555 result
= mysql_findzone(driverarg
, dbdata
, name
);
556 if (result
!= ISC_R_SUCCESS
)
557 return (ISC_R_NOTFOUND
);
560 * if we get to this point we know the zone is supported by
561 * the database the only questions now are is the zone
562 * transfer is allowed for this client and did the config file
563 * have an allow zone xfr query.
565 * Run our query, and get a result set from the database.
567 result
= mysql_get_resultset(name
, NULL
, client
, ALLOWXFR
,
569 /* if we get "not implemented", send it along. */
570 if (result
== ISC_R_NOTIMPLEMENTED
)
572 /* if we didn't get a result set, log an err msg. */
573 if (result
!= ISC_R_SUCCESS
|| rs
== NULL
) {
575 mysql_free_result(rs
);
576 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
577 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
578 "mysql driver unable to return "
579 "result set for allow xfr query");
580 return (ISC_R_FAILURE
);
582 /* count how many rows in result set */
583 rows
= mysql_num_rows(rs
);
584 /* get rid of result set, we are done with it. */
585 mysql_free_result(rs
);
587 /* if we returned any rows, zone xfr is allowed. */
589 return (ISC_R_SUCCESS
);
591 /* no rows returned, zone xfr not allowed */
592 return (ISC_R_NOPERM
);
596 * If the client is allowed to perform a zone transfer, the next order of
597 * business is to get all the nodes in the zone, so bind can respond to the
601 mysql_allnodes(const char *zone
, void *driverarg
, void *dbdata
,
602 dns_sdlzallnodes_t
*allnodes
)
605 MYSQL_RES
*rs
= NULL
;
616 /* run the query and get the result set from the database. */
617 result
= mysql_get_resultset(zone
, NULL
, NULL
, ALLNODES
, dbdata
, &rs
);
618 /* if we get "not implemented", send it along */
619 if (result
== ISC_R_NOTIMPLEMENTED
)
621 /* if we didn't get a result set, log an err msg. */
622 if (result
!= ISC_R_SUCCESS
) {
624 mysql_free_result(rs
);
625 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
626 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
627 "mysql driver unable to return "
628 "result set for all nodes query");
629 return (ISC_R_FAILURE
);
632 result
= ISC_R_NOTFOUND
;
634 row
= mysql_fetch_row(rs
); /* get a row from the result set */
635 fields
= mysql_num_fields(rs
); /* how many columns in result set */
636 while (row
!= NULL
) {
637 if (fields
< 4) { /* gotta have at least 4 columns */
638 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
639 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
640 "mysql driver too few fields returned "
641 "by all nodes query");
643 /* convert text to int, make sure it worked right */
644 ttl
= strtol(safeGet(row
[0]), &endp
, 10);
645 if (*endp
!= '\0' || ttl
< 0) {
646 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
647 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
648 "mysql driver ttl must be "
652 /* tell Bind about it. */
653 result
= dns_sdlz_putnamedrr(allnodes
, safeGet(row
[2]),
654 safeGet(row
[1]), ttl
,
658 * more than 4 fields, concatenate the last
659 * ones together. figure out how long to make
662 for (j
=3, len
=0; j
< fields
; j
++) {
663 len
+= strlen(safeGet(row
[j
])) + 1;
665 /* allocate memory, allow for NULL to term string */
666 tmpString
= isc_mem_allocate(ns_g_mctx
, len
+ 1);
667 if (tmpString
== NULL
) { /* we need more ram. */
668 isc_log_write(dns_lctx
,
669 DNS_LOGCATEGORY_DATABASE
,
670 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
671 "mysql driver unable "
672 "to allocate memory for "
674 mysql_free_result(rs
);
675 return (ISC_R_FAILURE
);
677 /* copy this field to tmpString */
678 strcpy(tmpString
, safeGet(row
[3]));
679 /* concatonate the rest, with spaces between */
680 for (j
=4; j
< fields
; j
++) {
681 strcat(tmpString
, " ");
682 strcat(tmpString
, safeGet(row
[j
]));
684 /* tell Bind about it. */
685 result
= dns_sdlz_putnamedrr(allnodes
, safeGet(row
[2]),
688 isc_mem_free(ns_g_mctx
, tmpString
);
690 /* if we weren't successful, log err msg */
691 if (result
!= ISC_R_SUCCESS
) {
692 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
693 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
694 "dns_sdlz_putnamedrr returned error. "
695 "Error code was: %s",
696 isc_result_totext(result
));
697 result
= ISC_R_FAILURE
;
700 /* get next row from the result set */
701 row
= mysql_fetch_row(rs
);
704 /* free result set memory */
705 mysql_free_result(rs
);
710 /*% if the lookup function does not return SOA or NS records for the zone,
711 * use this function to get that information for Bind.
715 mysql_authority(const char *zone
, void *driverarg
, void *dbdata
,
716 dns_sdlzlookup_t
*lookup
)
719 MYSQL_RES
*rs
= NULL
;
723 /* run the query and get the result set from the database. */
724 result
= mysql_get_resultset(zone
, NULL
, NULL
, AUTHORITY
, dbdata
, &rs
);
725 /* if we get "not implemented", send it along */
726 if (result
== ISC_R_NOTIMPLEMENTED
)
728 /* if we didn't get a result set, log an err msg. */
729 if (result
!= ISC_R_SUCCESS
) {
731 mysql_free_result(rs
);
732 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
733 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
734 "mysql driver unable to return "
735 "result set for authority query");
736 return (ISC_R_FAILURE
);
739 * lookup and authority result sets are processed in the same
740 * manner mysql_process_rs does the job for both functions.
742 return mysql_process_rs(lookup
, rs
);
745 /*% if zone is supported, lookup up a (or multiple) record(s) in it */
747 mysql_lookup(const char *zone
, const char *name
, void *driverarg
,
748 void *dbdata
, dns_sdlzlookup_t
*lookup
)
751 MYSQL_RES
*rs
= NULL
;
755 /* run the query and get the result set from the database. */
756 result
= mysql_get_resultset(zone
, name
, NULL
, LOOKUP
, dbdata
, &rs
);
757 /* if we didn't get a result set, log an err msg. */
758 if (result
!= ISC_R_SUCCESS
) {
760 mysql_free_result(rs
);
761 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
762 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
763 "mysql driver unable to return "
764 "result set for lookup query");
765 return (ISC_R_FAILURE
);
768 * lookup and authority result sets are processed in the same manner
769 * mysql_process_rs does the job for both functions.
771 return mysql_process_rs(lookup
, rs
);
775 * create an instance of the driver. Remember, only 1 copy of the driver's
776 * code is ever loaded, the driver has to remember which context it's
777 * operating in. This is done via use of the dbdata argument which is
778 * passed into all query functions.
781 mysql_create(const char *dlzname
, unsigned int argc
, char *argv
[],
782 void *driverarg
, void **dbdata
)
785 dbinstance_t
*dbi
= NULL
;
796 unsigned int flags
= 0;
797 #if MYSQL_VERSION_ID >= 50000
798 my_bool auto_reconnect
= 1;
804 /* verify we have at least 4 arg's passed to the driver */
806 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
807 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
808 "mysql driver requires "
809 "at least 4 command line args.");
810 return (ISC_R_FAILURE
);
813 /* no more than 8 arg's should be passed to the driver */
815 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
816 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
817 "mysql driver cannot accept "
818 "more than 7 command line args.");
819 return (ISC_R_FAILURE
);
822 /* parse connection string and get paramters. */
824 /* get db name - required */
825 dbname
= getParameterValue(argv
[1], "dbname=");
826 if (dbname
== NULL
) {
827 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
828 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
829 "mysql driver requires a dbname parameter.");
830 result
= ISC_R_FAILURE
;
834 /* get db port. Not required, but must be > 0 if specified */
835 tmp
= getParameterValue(argv
[1], "port=");
839 port
= strtol(tmp
, &endp
, 10);
840 if (*endp
!= '\0' || port
< 0) {
841 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
842 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
844 "must be a positive number.");
845 isc_mem_free(ns_g_mctx
, tmp
);
846 result
= ISC_R_FAILURE
;
849 isc_mem_free(ns_g_mctx
, tmp
);
852 /* how many queries were passed in from config file? */
855 result
= build_sqldbinstance(ns_g_mctx
, NULL
, NULL
, NULL
,
856 argv
[2], argv
[3], NULL
, &dbi
);
859 result
= build_sqldbinstance(ns_g_mctx
, NULL
, NULL
, argv
[4],
860 argv
[2], argv
[3], NULL
, &dbi
);
863 result
= build_sqldbinstance(ns_g_mctx
, argv
[5], NULL
, argv
[4],
864 argv
[2], argv
[3], NULL
, &dbi
);
867 result
= build_sqldbinstance(ns_g_mctx
, argv
[5],
869 argv
[2], argv
[3], NULL
, &dbi
);
872 result
= build_sqldbinstance(ns_g_mctx
, argv
[5],
874 argv
[2], argv
[3], argv
[7], &dbi
);
877 /* not really needed, should shut up compiler. */
878 result
= ISC_R_FAILURE
;
881 /* unsuccessful?, log err msg and cleanup. */
882 if (result
!= ISC_R_SUCCESS
) {
883 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
884 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
885 "mysql driver could not create "
886 "database instance object.");
887 result
= ISC_R_FAILURE
;
891 /* create and set db connection */
892 dbi
->dbconn
= mysql_init(NULL
);
894 /* if db connection cannot be created, log err msg and cleanup. */
895 if (dbi
->dbconn
== NULL
) {
896 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
897 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
898 "mysql driver could not allocate "
899 "memory for database connection");
900 result
= ISC_R_FAILURE
;
904 tmp
= getParameterValue(argv
[1], "compress=");
906 if (strcasecmp(tmp
, "true") == 0)
907 flags
= CLIENT_COMPRESS
;
908 isc_mem_free(ns_g_mctx
, tmp
);
911 tmp
= getParameterValue(argv
[1], "ssl=");
913 if (strcasecmp(tmp
, "true") == 0)
914 flags
= flags
| CLIENT_SSL
;
915 isc_mem_free(ns_g_mctx
, tmp
);
918 tmp
= getParameterValue(argv
[1], "space=");
920 if (strcasecmp(tmp
, "ignore") == 0)
921 flags
= flags
| CLIENT_IGNORE_SPACE
;
922 isc_mem_free(ns_g_mctx
, tmp
);
926 host
= getParameterValue(argv
[1], "host=");
927 user
= getParameterValue(argv
[1], "user=");
928 pass
= getParameterValue(argv
[1], "pass=");
929 socket
= getParameterValue(argv
[1], "socket=");
931 #if MYSQL_VERSION_ID >= 50000
932 /* enable automatic reconnection. */
933 if (mysql_options((MYSQL
*) dbi
->dbconn
, MYSQL_OPT_RECONNECT
,
934 &auto_reconnect
) != 0) {
935 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
936 DNS_LOGMODULE_DLZ
, ISC_LOG_WARNING
,
937 "mysql driver failed to set "
938 "MYSQL_OPT_RECONNECT option, continuing");
942 for (j
=0; dbc
== NULL
&& j
< 4; j
++)
943 dbc
= mysql_real_connect((MYSQL
*) dbi
->dbconn
, host
,
944 user
, pass
, dbname
, port
, socket
,
947 /* let user know if we couldn't connect. */
949 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
950 DNS_LOGMODULE_DLZ
, ISC_LOG_ERROR
,
951 "mysql driver failed to create "
952 "database connection after 4 attempts");
953 result
= ISC_R_FAILURE
;
957 /* return db connection via dbdata */
960 result
= ISC_R_SUCCESS
;
965 destroy_sqldbinstance(dbi
);
970 isc_mem_free(ns_g_mctx
, dbname
);
972 isc_mem_free(ns_g_mctx
, host
);
974 isc_mem_free(ns_g_mctx
, user
);
976 isc_mem_free(ns_g_mctx
, pass
);
978 isc_mem_free(ns_g_mctx
, socket
);
985 * destroy the driver. Remember, only 1 copy of the driver's
986 * code is ever loaded, the driver has to remember which context it's
987 * operating in. This is done via use of the dbdata argument.
988 * so we really only need to clean it up since we are not using driverarg.
992 mysql_destroy(void *driverarg
, void *dbdata
)
998 dbi
= (dbinstance_t
*) dbdata
;
1000 /* release DB connection */
1001 if (dbi
->dbconn
!= NULL
)
1002 mysql_close((MYSQL
*) dbi
->dbconn
);
1004 /* destroy DB instance */
1005 destroy_sqldbinstance(dbi
);
1008 /* pointers to all our runtime methods. */
1009 /* this is used during driver registration */
1010 /* i.e. in dlz_mysql_init below. */
1011 static dns_sdlzmethods_t dlz_mysql_methods
= {
1022 * Wrapper around dns_sdlzregister().
1025 dlz_mysql_init(void) {
1026 isc_result_t result
;
1029 * Write debugging message to log
1031 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
1032 DNS_LOGMODULE_DLZ
, ISC_LOG_DEBUG(2),
1033 "Registering DLZ mysql driver.");
1035 /* Driver is always threadsafe. Because of the way MySQL handles
1036 * threads the MySQL driver can only be used when bind is run single
1037 * threaded. Using MySQL with Bind running multi-threaded is not
1038 * allowed. When using the MySQL driver "-n1" should always be
1039 * passed to Bind to guarantee single threaded operation.
1041 result
= dns_sdlzregister("mysql", &dlz_mysql_methods
, NULL
,
1042 DNS_SDLZFLAG_RELATIVEOWNER
|
1043 DNS_SDLZFLAG_RELATIVERDATA
|
1044 DNS_SDLZFLAG_THREADSAFE
,
1045 ns_g_mctx
, &dlz_mysql
);
1046 /* if we can't register the driver, there are big problems. */
1047 if (result
!= ISC_R_SUCCESS
) {
1048 UNEXPECTED_ERROR(__FILE__
, __LINE__
,
1049 "dns_sdlzregister() failed: %s",
1050 isc_result_totext(result
));
1051 result
= ISC_R_UNEXPECTED
;
1059 * Wrapper around dns_sdlzunregister().
1062 dlz_mysql_clear(void) {
1065 * Write debugging message to log
1067 isc_log_write(dns_lctx
, DNS_LOGCATEGORY_DATABASE
,
1068 DNS_LOGMODULE_DLZ
, ISC_LOG_DEBUG(2),
1069 "Unregistering DLZ mysql driver.");
1071 /* unregister the driver. */
1072 if (dlz_mysql
!= NULL
)
1073 dns_sdlzunregister(&dlz_mysql
);