/*************************************************************************** sqlite.c (c) 2000-2011 BenoƮt Minisini <gambas@users.sourceforge.net> This program 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, or (at your option) any later version. This program 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. ***************************************************************************/ #define __SQLITE_C #include <stdio.h> #include <stdlib.h> #include <stdarg.h> #include <string.h> #include <sys/types.h> #include <sys/stat.h> #include <dirent.h> #include <unistd.h> #include <libgen.h> #include "main.h" /* Internal function to check whether a file is a sqlite database file */ static bool is_sqlite2_database(char *filename) { FILE* fp; bool res; char magic_text[48]; fp = fopen(filename, "r"); if (!fp) return FALSE; res = fread(magic_text, 1, 47, fp) == 47; fclose(fp); if (!res) return FALSE; magic_text[47] = '\0'; if (strcmp(magic_text, "** This file contains an SQLite 2.1 database **")) return FALSE; return TRUE; } static bool is_sqlite3_database(char *filename) { FILE *fp; bool res; char magic_text[16]; fp = fopen(filename, "r"); if (!fp) return FALSE; res = fread(magic_text, 1, 15, fp) == 15; fclose(fp); if (!res) return FALSE; magic_text[15] = '\0'; if (strcmp(magic_text, "SQLite format 3")) return FALSE; return TRUE; } static bool IsDatabaseFile(char *filename) { return is_sqlite3_database(filename) || is_sqlite2_database(filename); } static char *FindDatabase(char *name, char *hostName) { char *dbhome = NULL; char *fullpath = NULL; /* Does Name includes fullpath */ if (strcmp(basename(name), name)) { if (IsDatabaseFile(name)) fullpath = GB.NewZeroString(name); return fullpath; } /* Hostname contains home area */ fullpath = GB.NewZeroString(hostName); GB.AddString(&fullpath, "/", 0); GB.AddString(&fullpath, name, 0); if (IsDatabaseFile(fullpath)) { return fullpath; } GB.FreeString(&fullpath); /* Check the GAMBAS_SQLITE_DBHOME setting */ dbhome = getenv("GAMBAS_SQLITE_DBHOME"); if (dbhome != NULL) { fullpath = GB.NewZeroString(dbhome); GB.AddString(&fullpath, "/", 0); GB.AddString(&fullpath, name, 0); if (IsDatabaseFile(fullpath)) { return fullpath; } } fullpath = GB.NewZeroString(GB.TempDir()); GB.AddString(&fullpath, "/sqlite/", 0); GB.AddString(&fullpath, name, 0); if (IsDatabaseFile(fullpath)) { return fullpath; } GB.FreeString(&fullpath); return NULL; } /***************************************************************************** open_database() *****************************************************************************/ static int open_database(DB_DESC *desc, DB_DATABASE * db) { char *db_fullpath = NULL; bool ver2 = FALSE; if (!desc->name) // memory database goto __SQLITE; db_fullpath = FindDatabase(desc->name, desc->host); if (!db_fullpath) { GB.Error("Unable to locate database: &1", desc->name); return TRUE; } ver2 = is_sqlite2_database(db_fullpath); GB.FreeString(&db_fullpath); if (ver2) goto __SQLITE2; else goto __SQLITE3; __SQLITE: GB.LoadComponent("gb.db.sqlite3"); GB.Error(NULL); if (GB.ExistComponent("gb.db.sqlite3")) goto __SQLITE3; else goto __SQLITE2; __SQLITE2: DB_TryAnother("sqlite2"); return TRUE; __SQLITE3: DB_TryAnother("sqlite3"); return TRUE; } /***************************************************************************** The driver interface *****************************************************************************/ DB_DRIVER DB_sqlite_pseudo_driver = { "sqlite", open_database, 0 };