2 * sql_mysql.c SQL Module
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20 * Copyright 2000-2007 The FreeRADIUS server project
21 * Copyright 2000 Mike Machado <mike@innercite.com>
22 * Copyright 2000 Alan DeKok <aland@ox.org>
25 #include <freeradius-devel/ident.h>
28 #include <freeradius-devel/radiusd.h>
34 #ifdef HAVE_MYSQL_MYSQL_H
35 #include <mysql/mysql_version.h>
36 #include <mysql/errmsg.h>
37 #include <mysql/mysql.h>
40 #include <mysql_version.h>
48 typedef struct rlm_sql_mysql_sock {
56 static int sql_free_result(rlm_sql_handle_t*, rlm_sql_config_t*);
58 /*************************************************************************
60 * Function: sql_create_socket
62 * Purpose: Establish connection to the db
64 *************************************************************************/
65 static int sql_init_socket(rlm_sql_handle_t *handle, rlm_sql_config_t *config)
67 rlm_sql_mysql_sock *mysql_sock;
68 unsigned long sql_flags;
71 handle->conn = (rlm_sql_mysql_sock *)rad_malloc(sizeof(rlm_sql_mysql_sock));
76 mysql_sock = handle->conn;
77 memset(mysql_sock, 0, sizeof(*mysql_sock));
79 DEBUG("rlm_sql_mysql: Starting connect to MySQL server");
81 mysql_init(&(mysql_sock->conn));
82 mysql_options(&(mysql_sock->conn), MYSQL_READ_DEFAULT_GROUP, "freeradius");
84 #if (MYSQL_VERSION_ID >= 50000)
85 if (config->query_timeout) {
86 unsigned int timeout = config->query_timeout;
89 * 3 retries are hard-coded into the MySQL library.
90 * We ensure that the REAL timeout is what the user
91 * set by accounting for that.
93 if (timeout > 3) timeout /= 3;
95 mysql_options(&(mysql_sock->conn), MYSQL_OPT_CONNECT_TIMEOUT,
97 mysql_options(&(mysql_sock->conn), MYSQL_OPT_READ_TIMEOUT,
99 mysql_options(&(mysql_sock->conn), MYSQL_OPT_WRITE_TIMEOUT,
104 #if (MYSQL_VERSION_ID >= 40100)
105 sql_flags = CLIENT_MULTI_RESULTS | CLIENT_FOUND_ROWS;
107 sql_flags = CLIENT_FOUND_ROWS;
110 #ifdef CLIENT_MULTI_STATEMENTS
111 sql_flags |= CLIENT_MULTI_STATEMENTS;
113 if (!(mysql_sock->sock = mysql_real_connect(&(mysql_sock->conn),
116 config->sql_password,
118 atoi(config->sql_port),
121 radlog(L_ERR, "rlm_sql_mysql: Couldn't connect socket to MySQL server %s@%s:%s", config->sql_login, config->sql_server, config->sql_db);
122 radlog(L_ERR, "rlm_sql_mysql: Mysql error '%s'", mysql_error(&mysql_sock->conn));
123 mysql_sock->sock = NULL;
132 /*************************************************************************
134 * Function: sql_destroy_socket
136 * Purpose: Free socket and any private connection data
138 *************************************************************************/
139 static int sql_destroy_socket(rlm_sql_handle_t *handle, UNUSED rlm_sql_config_t *config)
148 /*************************************************************************
150 * Function: sql_check_error
152 * Purpose: check the error to see if the server is down
154 *************************************************************************/
155 static int sql_check_error(int error)
158 case CR_SERVER_GONE_ERROR:
161 radlog(L_DBG, "rlm_sql_mysql: MYSQL check_error: %d, returning SQL_DOWN", error);
167 case CR_OUT_OF_MEMORY:
168 case CR_COMMANDS_OUT_OF_SYNC:
169 case CR_UNKNOWN_ERROR:
171 radlog(L_DBG, "rlm_sql_mysql: MYSQL check_error: %d received", error);
178 /*************************************************************************
180 * Function: sql_query
182 * Purpose: Issue a query to the database
184 *************************************************************************/
185 static int sql_query(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config,
188 rlm_sql_mysql_sock *mysql_sock = handle->conn;
190 if (mysql_sock->sock == NULL) {
191 radlog(L_ERR, "rlm_sql_mysql: Socket not connected");
195 mysql_query(mysql_sock->sock, querystr);
196 return sql_check_error(mysql_errno(mysql_sock->sock));
200 /*************************************************************************
202 * Function: sql_store_result
204 * Purpose: database specific store_result function. Returns a result
205 * set for the query. In case of multiple results, get the
206 * first non-empty one.
208 *************************************************************************/
209 static int sql_store_result(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
211 rlm_sql_mysql_sock *mysql_sock = handle->conn;
214 if (mysql_sock->sock == NULL) {
215 radlog(L_ERR, "rlm_sql_mysql: Socket not connected");
219 if (!(mysql_sock->result = mysql_store_result(mysql_sock->sock))) {
220 status = sql_check_error(mysql_errno(mysql_sock->sock));
222 radlog(L_ERR, "rlm_sql_mysql: Cannot store result");
223 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
224 mysql_error(mysql_sock->sock));
227 #if (MYSQL_VERSION_ID >= 40100)
228 status = mysql_next_result(mysql_sock->sock);
230 /* there are more results */
231 goto retry_store_result;
232 } else if (status > 0) {
233 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
234 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
235 mysql_error(mysql_sock->sock));
236 return sql_check_error(status);
244 /*************************************************************************
246 * Function: sql_num_fields
248 * Purpose: database specific num_fields function. Returns number
249 * of columns from query
251 *************************************************************************/
252 static int sql_num_fields(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
255 rlm_sql_mysql_sock *mysql_sock = handle->conn;
257 #if MYSQL_VERSION_ID >= 32224
258 if (!(num = mysql_field_count(mysql_sock->sock))) {
260 if (!(num = mysql_num_fields(mysql_sock->sock))) {
262 radlog(L_ERR, "rlm_sql_mysql: MYSQL Error: No Fields");
263 radlog(L_ERR, "rlm_sql_mysql: MYSQL error: %s",
264 mysql_error(mysql_sock->sock));
270 /*************************************************************************
272 * Function: sql_select_query
274 * Purpose: Issue a select query to the database
276 *************************************************************************/
277 static int sql_select_query(rlm_sql_handle_t *handle, rlm_sql_config_t *config,
282 ret = sql_query(handle, config, querystr);
285 ret = sql_store_result(handle, config);
290 /* Why? Per http://www.mysql.com/doc/n/o/node_591.html,
291 * this cannot return an error. Perhaps just to complain if no
294 sql_num_fields(handle, config);
300 /*************************************************************************
302 * Function: sql_num_rows
304 * Purpose: database specific num_rows. Returns number of rows in
307 *************************************************************************/
308 static int sql_num_rows(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
310 rlm_sql_mysql_sock *mysql_sock = handle->conn;
312 if (mysql_sock->result)
313 return mysql_num_rows(mysql_sock->result);
319 /*************************************************************************
321 * Function: sql_fetch_row
323 * Purpose: database specific fetch_row. Returns a rlm_sql_row_t struct
324 * with all the data for the query in 'handle->row'. Returns
325 * 0 on success, -1 on failure, SQL_DOWN if database is down.
327 *************************************************************************/
328 static int sql_fetch_row(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
330 rlm_sql_mysql_sock *mysql_sock = handle->conn;
334 * Check pointer before de-referencing it.
336 if (!mysql_sock->result) {
341 handle->row = mysql_fetch_row(mysql_sock->result);
343 if (handle->row == NULL) {
344 status = sql_check_error(mysql_errno(mysql_sock->sock));
346 radlog(L_ERR, "rlm_sql_mysql: Cannot fetch row");
347 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
348 mysql_error(mysql_sock->sock));
351 #if (MYSQL_VERSION_ID >= 40100)
352 sql_free_result(handle, config);
353 status = mysql_next_result(mysql_sock->sock);
355 /* there are more results */
356 if ((sql_store_result(handle, config) == 0)
357 && (mysql_sock->result != NULL))
358 goto retry_fetch_row;
359 } else if (status > 0) {
360 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
361 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
362 mysql_error(mysql_sock->sock));
363 return sql_check_error(status);
371 /*************************************************************************
373 * Function: sql_free_result
375 * Purpose: database specific free_result. Frees memory allocated
378 *************************************************************************/
379 static int sql_free_result(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
381 rlm_sql_mysql_sock *mysql_sock = handle->conn;
383 if (mysql_sock->result) {
384 mysql_free_result(mysql_sock->result);
385 mysql_sock->result = NULL;
393 /*************************************************************************
395 * Function: sql_error
397 * Purpose: database specific error. Returns error associated with
400 *************************************************************************/
401 static const char *sql_error(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
403 rlm_sql_mysql_sock *mysql_sock = handle->conn;
405 if (mysql_sock == NULL || mysql_sock->sock == NULL) {
406 return "rlm_sql_mysql: no connection to db";
408 return mysql_error(mysql_sock->sock);
412 /*************************************************************************
414 * Function: sql_close
416 * Purpose: database specific close. Closes an open database
419 *************************************************************************/
420 static int sql_close(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
422 rlm_sql_mysql_sock *mysql_sock = handle->conn;
424 if (mysql_sock && mysql_sock->sock){
425 mysql_close(mysql_sock->sock);
426 mysql_sock->sock = NULL;
433 /*************************************************************************
435 * Function: sql_finish_query
437 * Purpose: As a single SQL statement may return multiple results
438 * sets, (for example stored procedures) it is necessary to check
439 * whether more results exist and process them in turn if so.
441 *************************************************************************/
442 static int sql_finish_query(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
444 #if (MYSQL_VERSION_ID >= 40100)
445 rlm_sql_mysql_sock *mysql_sock = handle->conn;
449 status = sql_store_result(handle, config);
452 } else if (mysql_sock->result != NULL) {
453 radlog(L_DBG, "rlm_sql_mysql: SQL statement returned unexpected result");
454 sql_free_result(handle, config);
456 status = mysql_next_result(mysql_sock->sock);
458 /* there are more results */
459 goto skip_next_result;
460 } else if (status > 0) {
461 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
462 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
463 mysql_error(mysql_sock->sock));
464 return sql_check_error(status);
472 /*************************************************************************
474 * Function: sql_finish_select_query
476 * Purpose: End the select query, such as freeing memory or result
478 *************************************************************************/
479 static int sql_finish_select_query(rlm_sql_handle_t * handle, rlm_sql_config_t *config)
481 #if (MYSQL_VERSION_ID >= 40100)
483 rlm_sql_mysql_sock *mysql_sock = handle->conn;
485 sql_free_result(handle, config);
486 #if (MYSQL_VERSION_ID >= 40100)
487 status = mysql_next_result(mysql_sock->sock);
489 /* there are more results */
490 sql_finish_query(handle, config);
491 } else if (status > 0) {
492 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
493 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
494 mysql_error(mysql_sock->sock));
495 return sql_check_error(status);
502 /*************************************************************************
504 * Function: sql_affected_rows
506 * Purpose: End the select query, such as freeing memory or result
508 *************************************************************************/
509 static int sql_affected_rows(rlm_sql_handle_t * handle, UNUSED rlm_sql_config_t *config)
511 rlm_sql_mysql_sock *mysql_sock = handle->conn;
513 return mysql_affected_rows(mysql_sock->sock);
517 /* Exported to rlm_sql */
518 rlm_sql_module_t rlm_sql_mysql = {
532 sql_finish_select_query,