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(SQLSOCK*, SQL_CONFIG*);
58 /*************************************************************************
60 * Function: sql_create_socket
62 * Purpose: Establish connection to the db
64 *************************************************************************/
65 static int sql_init_socket(SQLSOCK *sqlsocket, SQL_CONFIG *config)
67 rlm_sql_mysql_sock *mysql_sock;
68 unsigned long sql_flags;
69 unsigned int timeout = config->query_timeout;
71 if (!sqlsocket->conn) {
72 sqlsocket->conn = (rlm_sql_mysql_sock *)rad_malloc(sizeof(rlm_sql_mysql_sock));
73 if (!sqlsocket->conn) {
77 mysql_sock = sqlsocket->conn;
78 memset(mysql_sock, 0, sizeof(*mysql_sock));
80 radlog(L_INFO, "rlm_sql_mysql: Starting connect to MySQL server");
82 mysql_init(&(mysql_sock->conn));
83 mysql_options(&(mysql_sock->conn), MYSQL_READ_DEFAULT_GROUP, "freeradius");
85 #if (MYSQL_VERSION_ID >= 50000)
87 mysql_options(&(mysql_sock->conn), MYSQL_OPT_CONNECT_TIMEOUT, &timeout);
88 mysql_options(&(mysql_sock->conn), MYSQL_OPT_READ_TIMEOUT, &timeout);
89 mysql_options(&(mysql_sock->conn), MYSQL_OPT_WRITE_TIMEOUT, &timeout);
93 #if (MYSQL_VERSION_ID >= 40100)
94 sql_flags = CLIENT_MULTI_RESULTS | CLIENT_FOUND_ROWS;
96 sql_flags = CLIENT_FOUND_ROWS;
99 #ifdef CLIENT_MULTI_STATEMENTS
100 sql_flags |= CLIENT_MULTI_STATEMENTS;
102 if (!(mysql_sock->sock = mysql_real_connect(&(mysql_sock->conn),
105 config->sql_password,
107 atoi(config->sql_port),
110 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);
111 radlog(L_ERR, "rlm_sql_mysql: Mysql error '%s'", mysql_error(&mysql_sock->conn));
112 mysql_sock->sock = NULL;
121 /*************************************************************************
123 * Function: sql_destroy_socket
125 * Purpose: Free socket and any private connection data
127 *************************************************************************/
128 static int sql_destroy_socket(SQLSOCK *sqlsocket, UNUSED SQL_CONFIG *config)
130 free(sqlsocket->conn);
131 sqlsocket->conn = NULL;
137 /*************************************************************************
139 * Function: sql_check_error
141 * Purpose: check the error to see if the server is down
143 *************************************************************************/
144 static int sql_check_error(int error)
147 case CR_SERVER_GONE_ERROR:
150 radlog(L_DBG, "rlm_sql_mysql: MYSQL check_error: %d, returning SQL_DOWN", error);
156 case CR_OUT_OF_MEMORY:
157 case CR_COMMANDS_OUT_OF_SYNC:
158 case CR_UNKNOWN_ERROR:
160 radlog(L_DBG, "rlm_sql_mysql: MYSQL check_error: %d received", error);
167 /*************************************************************************
169 * Function: sql_query
171 * Purpose: Issue a query to the database
173 *************************************************************************/
174 static int sql_query(SQLSOCK * sqlsocket, SQL_CONFIG *config, char *querystr)
176 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
178 if (config->sqltrace)
179 radlog(L_DBG,"rlm_sql_mysql: query: %s", querystr);
180 if (mysql_sock->sock == NULL) {
181 radlog(L_ERR, "rlm_sql_mysql: Socket not connected");
185 mysql_query(mysql_sock->sock, querystr);
186 return sql_check_error(mysql_errno(mysql_sock->sock));
190 /*************************************************************************
192 * Function: sql_store_result
194 * Purpose: database specific store_result function. Returns a result
195 * set for the query. In case of multiple results, get the
196 * first non-empty one.
198 *************************************************************************/
199 static int sql_store_result(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
201 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
204 if (mysql_sock->sock == NULL) {
205 radlog(L_ERR, "rlm_sql_mysql: Socket not connected");
209 if (!(mysql_sock->result = mysql_store_result(mysql_sock->sock))) {
210 status = sql_check_error(mysql_errno(mysql_sock->sock));
212 radlog(L_ERR, "rlm_sql_mysql: Cannot store result");
213 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
214 mysql_error(mysql_sock->sock));
217 #if (MYSQL_VERSION_ID >= 40100)
218 status = mysql_next_result(mysql_sock->sock);
220 /* there are more results */
221 goto retry_store_result;
222 } else if (status > 0) {
223 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
224 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
225 mysql_error(mysql_sock->sock));
226 return sql_check_error(status);
234 /*************************************************************************
236 * Function: sql_num_fields
238 * Purpose: database specific num_fields function. Returns number
239 * of columns from query
241 *************************************************************************/
242 static int sql_num_fields(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
245 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
247 #if MYSQL_VERSION_ID >= 32224
248 if (!(num = mysql_field_count(mysql_sock->sock))) {
250 if (!(num = mysql_num_fields(mysql_sock->sock))) {
252 radlog(L_ERR, "rlm_sql_mysql: MYSQL Error: No Fields");
253 radlog(L_ERR, "rlm_sql_mysql: MYSQL error: %s",
254 mysql_error(mysql_sock->sock));
260 /*************************************************************************
262 * Function: sql_select_query
264 * Purpose: Issue a select query to the database
266 *************************************************************************/
267 static int sql_select_query(SQLSOCK *sqlsocket, SQL_CONFIG *config,
272 ret = sql_query(sqlsocket, config, querystr);
275 ret = sql_store_result(sqlsocket, config);
280 /* Why? Per http://www.mysql.com/doc/n/o/node_591.html,
281 * this cannot return an error. Perhaps just to complain if no
284 sql_num_fields(sqlsocket, config);
290 /*************************************************************************
292 * Function: sql_num_rows
294 * Purpose: database specific num_rows. Returns number of rows in
297 *************************************************************************/
298 static int sql_num_rows(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
300 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
302 if (mysql_sock->result)
303 return mysql_num_rows(mysql_sock->result);
309 /*************************************************************************
311 * Function: sql_fetch_row
313 * Purpose: database specific fetch_row. Returns a SQL_ROW struct
314 * with all the data for the query in 'sqlsocket->row'. Returns
315 * 0 on success, -1 on failure, SQL_DOWN if database is down.
317 *************************************************************************/
318 static int sql_fetch_row(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
320 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
324 * Check pointer before de-referencing it.
326 if (!mysql_sock->result) {
331 sqlsocket->row = mysql_fetch_row(mysql_sock->result);
333 if (sqlsocket->row == NULL) {
334 status = sql_check_error(mysql_errno(mysql_sock->sock));
336 radlog(L_ERR, "rlm_sql_mysql: Cannot fetch row");
337 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
338 mysql_error(mysql_sock->sock));
341 #if (MYSQL_VERSION_ID >= 40100)
342 sql_free_result(sqlsocket, config);
343 status = mysql_next_result(mysql_sock->sock);
345 /* there are more results */
346 if ((sql_store_result(sqlsocket, config) == 0)
347 && (mysql_sock->result != NULL))
348 goto retry_fetch_row;
349 } else if (status > 0) {
350 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
351 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
352 mysql_error(mysql_sock->sock));
353 return sql_check_error(status);
361 /*************************************************************************
363 * Function: sql_free_result
365 * Purpose: database specific free_result. Frees memory allocated
368 *************************************************************************/
369 static int sql_free_result(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
371 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
373 if (mysql_sock->result) {
374 mysql_free_result(mysql_sock->result);
375 mysql_sock->result = NULL;
383 /*************************************************************************
385 * Function: sql_error
387 * Purpose: database specific error. Returns error associated with
390 *************************************************************************/
391 static const char *sql_error(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
393 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
395 if (mysql_sock == NULL || mysql_sock->sock == NULL) {
396 return "rlm_sql_mysql: no connection to db";
398 return mysql_error(mysql_sock->sock);
402 /*************************************************************************
404 * Function: sql_close
406 * Purpose: database specific close. Closes an open database
409 *************************************************************************/
410 static int sql_close(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
412 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
414 if (mysql_sock && mysql_sock->sock){
415 mysql_close(mysql_sock->sock);
416 mysql_sock->sock = NULL;
423 /*************************************************************************
425 * Function: sql_finish_query
427 * Purpose: As a single SQL statement may return multiple results
428 * sets, (for example stored procedures) it is necessary to check
429 * whether more results exist and process them in turn if so.
431 *************************************************************************/
432 static int sql_finish_query(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
434 #if (MYSQL_VERSION_ID >= 40100)
435 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
439 status = sql_store_result(sqlsocket, config);
442 } else if (mysql_sock->result != NULL) {
443 radlog(L_DBG, "rlm_sql_mysql: SQL statement returned unexpected result");
444 sql_free_result(sqlsocket, config);
446 status = mysql_next_result(mysql_sock->sock);
448 /* there are more results */
449 goto skip_next_result;
450 } else if (status > 0) {
451 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
452 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
453 mysql_error(mysql_sock->sock));
454 return sql_check_error(status);
462 /*************************************************************************
464 * Function: sql_finish_select_query
466 * Purpose: End the select query, such as freeing memory or result
468 *************************************************************************/
469 static int sql_finish_select_query(SQLSOCK * sqlsocket, SQL_CONFIG *config)
471 #if (MYSQL_VERSION_ID >= 40100)
473 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
475 sql_free_result(sqlsocket, config);
476 #if (MYSQL_VERSION_ID >= 40100)
477 status = mysql_next_result(mysql_sock->sock);
479 /* there are more results */
480 sql_finish_query(sqlsocket, config);
481 } else if (status > 0) {
482 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
483 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
484 mysql_error(mysql_sock->sock));
485 return sql_check_error(status);
492 /*************************************************************************
494 * Function: sql_affected_rows
496 * Purpose: End the select query, such as freeing memory or result
498 *************************************************************************/
499 static int sql_affected_rows(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
501 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
503 return mysql_affected_rows(mysql_sock->sock);
507 /* Exported to rlm_sql */
508 rlm_sql_module_t rlm_sql_mysql = {
522 sql_finish_select_query,