2008-06-17 20:27:32 +04:00
/* -------------------------------------------------------------------------- */
2010-02-22 20:00:30 +03:00
/* Copyright 2002-2010, OpenNebula Project Leads (OpenNebula.org) */
2008-06-17 20:27:32 +04:00
/* */
/* Licensed under the Apache License, Version 2.0 (the "License"); you may */
/* not use this file except in compliance with the License. You may obtain */
/* a copy of the License at */
/* */
/* http://www.apache.org/licenses/LICENSE-2.0 */
/* */
/* Unless required by applicable law or agreed to in writing, software */
/* distributed under the License is distributed on an "AS IS" BASIS, */
/* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */
/* See the License for the specific language governing permissions and */
/* limitations under the License. */
/* -------------------------------------------------------------------------- */
# ifndef SQLITE_DB_H_
# define SQLITE_DB_H_
# include <string>
# include <sstream>
# include <stdexcept>
# include <sqlite3.h>
# include <sys/time.h>
# include <sys/types.h>
# include <unistd.h>
# include "Log.h"
2010-03-29 15:57:33 +04:00
# include "SqlDB.h"
2008-06-17 20:27:32 +04:00
using namespace std ;
/**
* SqliteDB class . Provides a wrapper to the sqlite3 database interface . It also
2010-03-29 15:57:33 +04:00
* provides " global " synchronization mechanism to use it in a multithread
2008-06-17 20:27:32 +04:00
* environment .
*/
2010-03-29 15:57:33 +04:00
class SqliteDB : public SqlDB
2008-06-17 20:27:32 +04:00
{
public :
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
SqliteDB (
string & db_name ,
Log : : LogFunction _log = 0
) : log ( _log )
{
int rc ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
pthread_mutex_init ( & mutex , 0 ) ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
rc = sqlite3_open ( db_name . c_str ( ) , & db ) ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
if ( rc ! = SQLITE_OK )
{
throw runtime_error ( " Could not open database. " ) ;
2010-03-29 15:57:33 +04:00
}
2008-06-17 20:27:32 +04:00
} ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
~ SqliteDB ( )
{
pthread_mutex_destroy ( & mutex ) ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
sqlite3_close ( db ) ;
} ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
/**
* Wraps the sqlite3_exec function call , and locks the DB mutex .
* @ param sql_cmd the SQL command
* @ param callbak function to execute on each data returned , watch the
* mutex you block in the callback .
* @ param arg to pass to the callback function
* @ return 0 on success
2010-03-29 15:57:33 +04:00
*/
int exec ( ostringstream & cmd , SqlCallback cbk = 0 , void * arg = 0 )
2008-06-17 20:27:32 +04:00
{
int rc ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
const char * c_str ;
string str ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
int counter = 0 ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
char * err_msg ;
char * * ptr = ( log = = 0 ) ? 0 : & err_msg ;
2010-03-29 15:57:33 +04:00
str = cmd . str ( ) ;
2008-06-17 20:27:32 +04:00
c_str = str . c_str ( ) ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
lock ( ) ;
do
{
counter + + ;
2010-03-29 15:57:33 +04:00
rc = sqlite3_exec ( db , c_str , cbk , arg , ptr ) ;
2008-06-17 20:27:32 +04:00
if ( rc = = SQLITE_BUSY | | rc = = SQLITE_IOERR_BLOCKED )
{
struct timeval timeout ;
fd_set zero ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
FD_ZERO ( & zero ) ;
timeout . tv_sec = 0 ;
timeout . tv_usec = 100000 ;
2010-03-29 15:57:33 +04:00
select ( 0 , & zero , & zero , & zero , & timeout ) ;
2008-06-17 20:27:32 +04:00
}
2010-03-29 15:57:33 +04:00
} while ( ( rc = = SQLITE_BUSY | | rc = = SQLITE_IOERR_BLOCKED ) & &
2008-06-17 20:27:32 +04:00
( counter < 10 ) ) ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
unlock ( ) ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
if ( rc ! = SQLITE_OK )
2010-03-29 15:57:33 +04:00
{
2008-06-17 20:27:32 +04:00
if ( ( log ! = 0 ) & & ( err_msg ! = 0 ) )
{
ostringstream oss ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
oss < < " SQL command was: " < < c_str < < " , error: " < < err_msg ;
2008-06-18 19:58:44 +04:00
log ( " ONE " , Log : : ERROR , oss , 0 , Log : : ERROR ) ;
2010-03-29 15:57:33 +04:00
sqlite3_free ( err_msg ) ;
2008-06-17 20:27:32 +04:00
}
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
return - 1 ;
}
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
return 0 ;
} ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
/**
2010-03-29 15:57:33 +04:00
* Performs a DB transaction
* @ param sql_cmd the SQL command
* @ param callbak function to execute on each data returned
* @ param arg to pass to the callback function
* @ return 0 on success
2008-06-17 20:27:32 +04:00
*/
2010-03-29 15:57:33 +04:00
int exec ( const char * cmd_c_str , SqlCallback cbk = 0 , void * arg = 0 )
2008-06-17 20:27:32 +04:00
{
2010-03-29 15:57:33 +04:00
string cmd_str = cmd_c_str ;
ostringstream cmd ;
cmd . str ( cmd_str ) ;
return exec ( cmd , cbk , arg ) ;
2008-06-17 20:27:32 +04:00
} ;
2010-03-29 15:57:33 +04:00
/**
* This function returns a legal SQL string that can be used in an SQL
* statement .
* @ param str the string to be escaped
* @ return a valid SQL string or NULL in case of failure
*/
char * escape_str ( const string & str )
{
return sqlite3_mprintf ( " %q " , str . c_str ( ) ) ;
} ;
/**
* Frees a previously scaped string
* @ param str pointer to the str
*/
void free_str ( char * str )
{
sqlite3_free ( str ) ;
} ;
2008-06-17 20:27:32 +04:00
private :
/**
* Fine - grain mutex for DB access
*/
pthread_mutex_t mutex ;
/**
* Pointer to the database .
*/
sqlite3 * db ;
/**
* Log facility
2010-03-29 15:57:33 +04:00
*/
2008-06-17 20:27:32 +04:00
Log : : LogFunction log ;
2010-03-29 15:57:33 +04:00
2008-06-17 20:27:32 +04:00
/**
* Function to lock the DB
*/
void lock ( )
{
pthread_mutex_lock ( & mutex ) ;
} ;
/**
* Function to unlock the DB
*/
void unlock ( )
{
pthread_mutex_unlock ( & mutex ) ;
2010-03-29 15:57:33 +04:00
} ;
2008-06-17 20:27:32 +04:00
} ;
# endif /*SQLITE_DB_H_*/