/* 
 * $Id: utils.c,v 1.12 2004/12/02 21:16:02 andrei Exp $ 
 *
 * MySQL module utilities
 *
 * Copyright (C) 2001-2003 FhG Fokus
 *
 * This file is part of ser, a free SIP server.
 *
 * ser is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version
 *
 * For a license to use the ser software under conditions
 * other than those described here, or to purchase support for this
 * software, please contact iptel.org by e-mail at the following addresses:
 *    info@iptel.org
 *
 * ser is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License 
 * along with this program; if not, write to the Free Software 
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *
 * History:
 * -------
 * 2003-04-14 tm_isdst in struct tm set to -1 to let mktime 
 *            guess daylight saving (janakj)
 */


#define _XOPEN_SOURCE 4     /* bsd */
#define _XOPEN_SOURCE_EXTENDED 1    /* solaris */

#include <strings.h>
#include <string.h>
#include <time.h>  /*strptime, XOPEN issue must be >=4 */
#include "utils.h"


/*
 * Convert time_t structure to format accepted by MySQL database
 */
int time2mysql(time_t _time, char* _result, int _res_len)
{
	struct tm* t;
	
	     /*
	       if (_time == MAX_TIME_T) {
	       snprintf(_result, _res_len, "0000-00-00 00:00:00");
	       }
	     */

	t = localtime(&_time);
	return strftime(_result, _res_len, "%Y-%m-%d %H:%M:%S", t);
}


/*
 * Convert MySQL time representation to time_t structure
 */
time_t mysql2time(const char* _str)
{
	struct tm time;
	
	     /* It is necessary to zero tm structure first */
	memset(&time, '\0', sizeof(struct tm));
	strptime(_str, "%Y-%m-%d %H:%M:%S", &time);

	     /* Daylight saving information got lost in the database
	      * so let mktime to guess it. This eliminates the bug when
	      * contacts reloaded from the database have different time
	      * of expiration by one hour when daylight saving is used
	      */ 
	time.tm_isdst = -1;   
	return mktime(&time);
}


/*
 * Parse a mysql database URL of form 
 * mysql://[username[:password]@]hostname[:port]/database
 *
 * Returns 0 if parsing was successful and -1 otherwise
 */
int parse_mysql_url(char* _url, char** _user, char** _pass,
		    char** _host, char** _port, char** _db)
{
#define SHORTEST_MYSQL_URL "mysql://a/b"
#define SHORTEST_MYSQL_URL_LEN (sizeof(SHORTEST_MYSQL_URL) - 1)

#define MYSQL_URL_PREFIX "mysql://"
#define MYSQL_URL_PREFIX_LEN (sizeof(MYSQL_URL_PREFIX) - 1)

	enum state {
		ST_USER_HOST,  /* Username or hostname */
		ST_PASS_PORT,  /* Password or port part */
		ST_HOST,       /* Hostname part */
		ST_PORT,       /* Port part */
		ST_DB          /* Database part */
	};

	enum state st;
	int len, i;
	char* begin, *prev_begin;

	if (!_url || !_user || !_pass || !_host || !_port || !_db) {
		return -1;
	}
	
	len = strlen(_url);
	if (len < SHORTEST_MYSQL_URL_LEN) {
		return -1;
	}
	
	if (strncasecmp(_url, MYSQL_URL_PREFIX, MYSQL_URL_PREFIX_LEN)) {
		return -1;
	}

	     /* Skip the prefix part */
	_url += MYSQL_URL_PREFIX_LEN;
	len -= MYSQL_URL_PREFIX_LEN;
	
	     /* Initialize all variables */
	*_user = '\0';
	*_pass = '\0';
	*_host = '\0';
	*_port = '\0';
	*_db   = '\0';

	st = ST_USER_HOST;
	begin = _url;
	prev_begin = 0;

	for(i = 0; i < len; i++) {
		switch(st) {
		case ST_USER_HOST:
			switch(_url[i]) {
			case '@':
				st = ST_HOST;
				*_user = begin;
				begin = _url + i + 1;
				_url[i] = '\0';
				break;

			case ':':
				st = ST_PASS_PORT;
				prev_begin = begin;
				begin = _url + i + 1;
				_url[i] = '\0';
				break;

			case '/':
				*_host = begin;
				_url[i] = '\0';

				*_db = _url + i + 1;
				return 0;
			}

		case ST_PASS_PORT:
			switch(_url[i]) {
			case '@':
				st = ST_HOST;
				*_user = prev_begin;
				*_pass = begin;
				begin = _url + i + 1;
				_url[i] = '\0';
				break;

			case '/':
				*_host = prev_begin;
				*_port = begin;
				_url[i] = '\0';

				*_db = _url + i + 1;
				return 0;
			}

		case ST_HOST:
			switch(_url[i]) {
			case ':':
				st = ST_PORT;
				*_host = begin;
				begin = _url + i + 1;
				_url[i] = '\0';
				break;

			case '/':
				*_host = begin;
				_url[i] = '\0';

				*_db = _url + i + 1;
				return 0;
			}

		case ST_PORT:
			switch(_url[i]) {
			case '/':
				*_port = begin;
				_url[i] = '\0';

				*_db = _url + i + 1;
				return 0;
			}
			break;
			
		case ST_DB:
			break;
		}
	}

	if (st != ST_DB) {
		return -1;
	}

	return 0;
}


syntax highlighted by Code2HTML, v. 0.9.1