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;
70 if (!sqlsocket->conn) {
71 sqlsocket->conn = (rlm_sql_mysql_sock *)rad_malloc(sizeof(rlm_sql_mysql_sock));
72 if (!sqlsocket->conn) {
76 mysql_sock = sqlsocket->conn;
77 memset(mysql_sock, 0, sizeof(*mysql_sock));
79 radlog(L_INFO, "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");
83 #if (MYSQL_VERSION_ID >= 40100)
84 sql_flags = CLIENT_MULTI_RESULTS | CLIENT_FOUND_ROWS;
86 sql_flags = CLIENT_FOUND_ROWS;
89 #ifdef CLIENT_MULTI_STATEMENTS
90 sql_flags |= CLIENT_MULTI_STATEMENTS;
92 if (!(mysql_sock->sock = mysql_real_connect(&(mysql_sock->conn),
97 atoi(config->sql_port),
100 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);
101 radlog(L_ERR, "rlm_sql_mysql: Mysql error '%s'", mysql_error(&mysql_sock->conn));
102 mysql_sock->sock = NULL;
111 /*************************************************************************
113 * Function: sql_destroy_socket
115 * Purpose: Free socket and any private connection data
117 *************************************************************************/
118 static int sql_destroy_socket(SQLSOCK *sqlsocket, UNUSED SQL_CONFIG *config)
120 free(sqlsocket->conn);
121 sqlsocket->conn = NULL;
127 /*************************************************************************
129 * Function: sql_check_error
131 * Purpose: check the error to see if the server is down
133 *************************************************************************/
134 static int sql_check_error(int error)
137 case CR_SERVER_GONE_ERROR:
140 radlog(L_DBG, "rlm_sql_mysql: MYSQL check_error: %d, returning SQL_DOWN", error);
146 case CR_OUT_OF_MEMORY:
147 case CR_COMMANDS_OUT_OF_SYNC:
148 case CR_UNKNOWN_ERROR:
150 radlog(L_DBG, "rlm_sql_mysql: MYSQL check_error: %d received", error);
157 /*************************************************************************
159 * Function: sql_query
161 * Purpose: Issue a query to the database
163 *************************************************************************/
164 static int sql_query(SQLSOCK * sqlsocket, SQL_CONFIG *config, char *querystr)
166 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
168 if (config->sqltrace)
169 radlog(L_DBG,"rlm_sql_mysql: query: %s", querystr);
170 if (mysql_sock->sock == NULL) {
171 radlog(L_ERR, "rlm_sql_mysql: Socket not connected");
175 mysql_query(mysql_sock->sock, querystr);
176 return sql_check_error(mysql_errno(mysql_sock->sock));
180 /*************************************************************************
182 * Function: sql_store_result
184 * Purpose: database specific store_result function. Returns a result
185 * set for the query. In case of multiple results, get the
186 * first non-empty one.
188 *************************************************************************/
189 static int sql_store_result(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
191 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
194 if (mysql_sock->sock == NULL) {
195 radlog(L_ERR, "rlm_sql_mysql: Socket not connected");
199 if (!(mysql_sock->result = mysql_store_result(mysql_sock->sock))) {
200 status = sql_check_error(mysql_errno(mysql_sock->sock));
202 radlog(L_ERR, "rlm_sql_mysql: Cannot store result");
203 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
204 mysql_error(mysql_sock->sock));
207 #if (MYSQL_VERSION_ID >= 40100)
208 status = mysql_next_result(mysql_sock->sock);
210 /* there are more results */
211 goto retry_store_result;
212 } else if (status > 0) {
213 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
214 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
215 mysql_error(mysql_sock->sock));
216 return sql_check_error(status);
224 /*************************************************************************
226 * Function: sql_num_fields
228 * Purpose: database specific num_fields function. Returns number
229 * of columns from query
231 *************************************************************************/
232 static int sql_num_fields(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
235 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
237 #if MYSQL_VERSION_ID >= 32224
238 if (!(num = mysql_field_count(mysql_sock->sock))) {
240 if (!(num = mysql_num_fields(mysql_sock->sock))) {
242 radlog(L_ERR, "rlm_sql_mysql: MYSQL Error: No Fields");
243 radlog(L_ERR, "rlm_sql_mysql: MYSQL error: %s",
244 mysql_error(mysql_sock->sock));
250 /*************************************************************************
252 * Function: sql_select_query
254 * Purpose: Issue a select query to the database
256 *************************************************************************/
257 static int sql_select_query(SQLSOCK *sqlsocket, SQL_CONFIG *config,
262 ret = sql_query(sqlsocket, config, querystr);
265 ret = sql_store_result(sqlsocket, config);
270 /* Why? Per http://www.mysql.com/doc/n/o/node_591.html,
271 * this cannot return an error. Perhaps just to complain if no
274 sql_num_fields(sqlsocket, config);
280 /*************************************************************************
282 * Function: sql_num_rows
284 * Purpose: database specific num_rows. Returns number of rows in
287 *************************************************************************/
288 static int sql_num_rows(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
290 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
292 if (mysql_sock->result)
293 return mysql_num_rows(mysql_sock->result);
299 /*************************************************************************
301 * Function: sql_fetch_row
303 * Purpose: database specific fetch_row. Returns a SQL_ROW struct
304 * with all the data for the query in 'sqlsocket->row'. Returns
305 * 0 on success, -1 on failure, SQL_DOWN if database is down.
307 *************************************************************************/
308 static int sql_fetch_row(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
310 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
314 * Check pointer before de-referencing it.
316 if (!mysql_sock->result) {
321 sqlsocket->row = mysql_fetch_row(mysql_sock->result);
323 if (sqlsocket->row == NULL) {
324 status = sql_check_error(mysql_errno(mysql_sock->sock));
326 radlog(L_ERR, "rlm_sql_mysql: Cannot fetch row");
327 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
328 mysql_error(mysql_sock->sock));
331 #if (MYSQL_VERSION_ID >= 40100)
332 sql_free_result(sqlsocket, config);
333 status = mysql_next_result(mysql_sock->sock);
335 /* there are more results */
336 if ((sql_store_result(sqlsocket, config) == 0)
337 && (mysql_sock->result != NULL))
338 goto retry_fetch_row;
339 } else if (status > 0) {
340 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
341 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
342 mysql_error(mysql_sock->sock));
343 return sql_check_error(status);
351 /*************************************************************************
353 * Function: sql_free_result
355 * Purpose: database specific free_result. Frees memory allocated
358 *************************************************************************/
359 static int sql_free_result(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
361 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
363 if (mysql_sock->result) {
364 mysql_free_result(mysql_sock->result);
365 mysql_sock->result = NULL;
373 /*************************************************************************
375 * Function: sql_error
377 * Purpose: database specific error. Returns error associated with
380 *************************************************************************/
381 static const char *sql_error(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
383 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
385 if (mysql_sock == NULL || mysql_sock->sock == NULL) {
386 return "rlm_sql_mysql: no connection to db";
388 return mysql_error(mysql_sock->sock);
392 /*************************************************************************
394 * Function: sql_close
396 * Purpose: database specific close. Closes an open database
399 *************************************************************************/
400 static int sql_close(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
402 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
404 if (mysql_sock && mysql_sock->sock){
405 mysql_close(mysql_sock->sock);
406 mysql_sock->sock = NULL;
413 /*************************************************************************
415 * Function: sql_finish_query
417 * Purpose: As a single SQL statement may return multiple results
418 * sets, (for example stored procedures) it is necessary to check
419 * whether more results exist and process them in turn if so.
421 *************************************************************************/
422 static int sql_finish_query(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
424 #if (MYSQL_VERSION_ID >= 40100)
425 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
429 status = sql_store_result(sqlsocket, config);
432 } else if (mysql_sock->result != NULL) {
433 radlog(L_DBG, "rlm_sql_mysql: SQL statement returned unexpected result");
434 sql_free_result(sqlsocket, config);
436 status = mysql_next_result(mysql_sock->sock);
438 /* there are more results */
439 goto skip_next_result;
440 } else if (status > 0) {
441 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
442 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
443 mysql_error(mysql_sock->sock));
444 return sql_check_error(status);
452 /*************************************************************************
454 * Function: sql_finish_select_query
456 * Purpose: End the select query, such as freeing memory or result
458 *************************************************************************/
459 static int sql_finish_select_query(SQLSOCK * sqlsocket, SQL_CONFIG *config)
461 #if (MYSQL_VERSION_ID >= 40100)
463 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
465 sql_free_result(sqlsocket, config);
466 #if (MYSQL_VERSION_ID >= 40100)
467 status = mysql_next_result(mysql_sock->sock);
469 /* there are more results */
470 sql_finish_query(sqlsocket, config);
471 } else if (status > 0) {
472 radlog(L_ERR, "rlm_sql_mysql: Cannot get next result");
473 radlog(L_ERR, "rlm_sql_mysql: MySQL error '%s'",
474 mysql_error(mysql_sock->sock));
475 return sql_check_error(status);
482 /*************************************************************************
484 * Function: sql_affected_rows
486 * Purpose: End the select query, such as freeing memory or result
488 *************************************************************************/
489 static int sql_affected_rows(SQLSOCK * sqlsocket, UNUSED SQL_CONFIG *config)
491 rlm_sql_mysql_sock *mysql_sock = sqlsocket->conn;
493 return mysql_affected_rows(mysql_sock->sock);
497 /* Exported to rlm_sql */
498 rlm_sql_module_t rlm_sql_mysql = {
512 sql_finish_select_query,