12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480 |
- /*
- +----------------------------------------------------------------------+
- | Copyright (c) The PHP Group |
- +----------------------------------------------------------------------+
- | This source file is subject to version 3.01 of the PHP license, |
- | that is bundled with this package in the file LICENSE, and is |
- | available through the world-wide-web at the following url: |
- | https://www.php.net/license/3_01.txt |
- | If you did not receive a copy of the PHP license and are unable to |
- | obtain it through the world-wide-web, please send a note to |
- | license@php.net so we can mail you a copy immediately. |
- +----------------------------------------------------------------------+
- | Authors: Scott MacVicar <scottmac@php.net> |
- +----------------------------------------------------------------------+
- */
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif
- #include "php.h"
- #include "php_ini.h"
- #include "ext/standard/info.h"
- #include "php_sqlite3.h"
- #include "php_sqlite3_structs.h"
- #include "sqlite3_arginfo.h"
- #include "main/SAPI.h"
- #include <sqlite3.h>
- #include "zend_exceptions.h"
- #include "SAPI.h"
- ZEND_DECLARE_MODULE_GLOBALS(sqlite3)
- static PHP_GINIT_FUNCTION(sqlite3);
- static int php_sqlite3_authorizer(void *autharg, int action, const char *arg1, const char *arg2, const char *arg3, const char *arg4);
- static void sqlite3_param_dtor(zval *data);
- static int php_sqlite3_compare_stmt_zval_free(php_sqlite3_free_list **free_list, zval *statement);
- /* {{{ Error Handler */
- static void php_sqlite3_error(php_sqlite3_db_object *db_obj, char *format, ...)
- {
- va_list arg;
- char *message;
- va_start(arg, format);
- vspprintf(&message, 0, format, arg);
- va_end(arg);
- if (db_obj && db_obj->exception) {
- zend_throw_exception(zend_ce_exception, message, 0);
- } else {
- php_error_docref(NULL, E_WARNING, "%s", message);
- }
- if (message) {
- efree(message);
- }
- }
- /* }}} */
- #define SQLITE3_CHECK_INITIALIZED(db_obj, member, class_name) \
- if (!(db_obj) || !(member)) { \
- zend_throw_error(NULL, "The " #class_name " object has not been correctly initialised or is already closed"); \
- RETURN_THROWS(); \
- }
- #define SQLITE3_CHECK_INITIALIZED_STMT(member, class_name) \
- if (!(member)) { \
- zend_throw_error(NULL, "The " #class_name " object has not been correctly initialised or is already closed"); \
- RETURN_THROWS(); \
- }
- /* {{{ PHP_INI */
- PHP_INI_BEGIN()
- STD_PHP_INI_ENTRY("sqlite3.extension_dir", NULL, PHP_INI_SYSTEM, OnUpdateString, extension_dir, zend_sqlite3_globals, sqlite3_globals)
- #if SQLITE_VERSION_NUMBER >= 3026000
- STD_PHP_INI_BOOLEAN("sqlite3.defensive", "1", PHP_INI_SYSTEM, OnUpdateBool, dbconfig_defensive, zend_sqlite3_globals, sqlite3_globals)
- #endif
- PHP_INI_END()
- /* }}} */
- /* Handlers */
- static zend_object_handlers sqlite3_object_handlers;
- static zend_object_handlers sqlite3_stmt_object_handlers;
- static zend_object_handlers sqlite3_result_object_handlers;
- /* Class entries */
- zend_class_entry *php_sqlite3_sc_entry;
- zend_class_entry *php_sqlite3_stmt_entry;
- zend_class_entry *php_sqlite3_result_entry;
- /* {{{ Opens a SQLite 3 Database, if the build includes encryption then it will attempt to use the key. */
- PHP_METHOD(SQLite3, open)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- char *filename, *encryption_key, *fullpath;
- size_t filename_len, encryption_key_len = 0;
- zend_long flags = SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE;
- int rc;
- db_obj = Z_SQLITE3_DB_P(object);
- if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "p|ls", &filename, &filename_len, &flags, &encryption_key, &encryption_key_len)) {
- RETURN_THROWS();
- }
- if (db_obj->initialised) {
- zend_throw_exception(zend_ce_exception, "Already initialised DB Object", 0);
- RETURN_THROWS();
- }
- if (filename_len != 0 && (filename_len != sizeof(":memory:")-1 ||
- memcmp(filename, ":memory:", sizeof(":memory:")-1) != 0)) {
- if (!(fullpath = expand_filepath(filename, NULL))) {
- zend_throw_exception(zend_ce_exception, "Unable to expand filepath", 0);
- RETURN_THROWS();
- }
- if (php_check_open_basedir(fullpath)) {
- zend_throw_exception_ex(zend_ce_exception, 0, "open_basedir prohibits opening %s", fullpath);
- efree(fullpath);
- RETURN_THROWS();
- }
- } else {
- /* filename equals "" or ":memory:" */
- fullpath = filename;
- }
- rc = sqlite3_open_v2(fullpath, &(db_obj->db), flags, NULL);
- if (rc != SQLITE_OK) {
- zend_throw_exception_ex(zend_ce_exception, 0, "Unable to open database: %s",
- #ifdef HAVE_SQLITE3_ERRSTR
- db_obj->db ? sqlite3_errmsg(db_obj->db) : sqlite3_errstr(rc));
- #else
- db_obj->db ? sqlite3_errmsg(db_obj->db) : "");
- #endif
- sqlite3_close(db_obj->db);
- if (fullpath != filename) {
- efree(fullpath);
- }
- return;
- }
- #ifdef SQLITE_HAS_CODEC
- if (encryption_key_len > 0) {
- if (sqlite3_key(db_obj->db, encryption_key, encryption_key_len) != SQLITE_OK) {
- zend_throw_exception_ex(zend_ce_exception, 0, "Unable to open database: %s", sqlite3_errmsg(db_obj->db));
- sqlite3_close(db_obj->db);
- RETURN_THROWS();
- }
- }
- #endif
- db_obj->initialised = 1;
- db_obj->authorizer_fci = empty_fcall_info;
- db_obj->authorizer_fcc = empty_fcall_info_cache;
- sqlite3_set_authorizer(db_obj->db, php_sqlite3_authorizer, db_obj);
- #if SQLITE_VERSION_NUMBER >= 3026000
- if (SQLITE3G(dbconfig_defensive)) {
- sqlite3_db_config(db_obj->db, SQLITE_DBCONFIG_DEFENSIVE, 1, NULL);
- }
- #endif
- if (fullpath != filename) {
- efree(fullpath);
- }
- }
- /* }}} */
- /* {{{ Close a SQLite 3 Database. */
- PHP_METHOD(SQLite3, close)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- int errcode;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- if (db_obj->initialised) {
- zend_llist_clean(&(db_obj->free_list));
- if(db_obj->db) {
- errcode = sqlite3_close(db_obj->db);
- if (errcode != SQLITE_OK) {
- php_sqlite3_error(db_obj, "Unable to close database: %d, %s", errcode, sqlite3_errmsg(db_obj->db));
- RETURN_FALSE;
- }
- }
- db_obj->initialised = 0;
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ Executes a result-less query against a given database. */
- PHP_METHOD(SQLite3, exec)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- zend_string *sql;
- char *errtext = NULL;
- db_obj = Z_SQLITE3_DB_P(object);
- if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "S", &sql)) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- if (sqlite3_exec(db_obj->db, ZSTR_VAL(sql), NULL, NULL, &errtext) != SQLITE_OK) {
- php_sqlite3_error(db_obj, "%s", errtext);
- sqlite3_free(errtext);
- RETURN_FALSE;
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ Returns the SQLite3 Library version as a string constant and as a number. */
- PHP_METHOD(SQLite3, version)
- {
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- array_init(return_value);
- add_assoc_string(return_value, "versionString", (char*)sqlite3_libversion());
- add_assoc_long(return_value, "versionNumber", sqlite3_libversion_number());
- return;
- }
- /* }}} */
- /* {{{ Returns the rowid of the most recent INSERT into the database from the database connection. */
- PHP_METHOD(SQLite3, lastInsertRowID)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- RETURN_LONG((zend_long) sqlite3_last_insert_rowid(db_obj->db));
- }
- /* }}} */
- /* {{{ Returns the numeric result code of the most recent failed sqlite API call for the database connection. */
- PHP_METHOD(SQLite3, lastErrorCode)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->db, SQLite3)
- if (db_obj->initialised) {
- RETURN_LONG(sqlite3_errcode(db_obj->db));
- } else {
- RETURN_LONG(0);
- }
- }
- /* }}} */
- /* {{{ Returns the numeric extended result code of the most recent failed sqlite API call for the database connection. */
- PHP_METHOD(SQLite3, lastExtendedErrorCode)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->db, SQLite3)
- if (db_obj->initialised) {
- RETURN_LONG(sqlite3_extended_errcode(db_obj->db));
- } else {
- RETURN_LONG(0);
- }
- }
- /* }}} */
- /* {{{ Turns on or off the extended result codes feature of SQLite. */
- PHP_METHOD(SQLite3, enableExtendedResultCodes)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- bool enable = 1;
- db_obj = Z_SQLITE3_DB_P(object);
- int ret;
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "|b", &enable) == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->db, SQLite3)
- if (db_obj->initialised) {
- ret = sqlite3_extended_result_codes(db_obj->db, enable ? 1 : 0);
- if (ret == SQLITE_OK)
- {
- RETURN_TRUE;
- }
- }
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ Returns english text describing the most recent failed sqlite API call for the database connection. */
- PHP_METHOD(SQLite3, lastErrorMsg)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->db, SQLite3)
- if (db_obj->initialised) {
- RETURN_STRING((char *)sqlite3_errmsg(db_obj->db));
- } else {
- RETURN_EMPTY_STRING();
- }
- }
- /* }}} */
- /* {{{ Sets a busy handler that will sleep until database is not locked or timeout is reached. Passing a value less than or equal to zero turns off all busy handlers. */
- PHP_METHOD(SQLite3, busyTimeout)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- zend_long ms;
- #ifdef SQLITE_ENABLE_API_ARMOR
- int return_code;
- #endif
- db_obj = Z_SQLITE3_DB_P(object);
- if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "l", &ms)) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- #ifdef SQLITE_ENABLE_API_ARMOR
- return_code = sqlite3_busy_timeout(db_obj->db, ms);
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(db_obj, "Unable to set busy timeout: %d, %s", return_code, sqlite3_errmsg(db_obj->db));
- RETURN_FALSE;
- }
- #else
- php_ignore_value(sqlite3_busy_timeout(db_obj->db, ms));
- #endif
- RETURN_TRUE;
- }
- /* }}} */
- #ifndef SQLITE_OMIT_LOAD_EXTENSION
- /* {{{ Attempts to load an SQLite extension library. */
- PHP_METHOD(SQLite3, loadExtension)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- char *extension, *lib_path, *extension_dir, *errtext = NULL;
- char fullpath[MAXPATHLEN];
- size_t extension_len, extension_dir_len;
- db_obj = Z_SQLITE3_DB_P(object);
- if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "s", &extension, &extension_len)) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- #ifdef ZTS
- if ((strncmp(sapi_module.name, "cgi", 3) != 0) &&
- (strcmp(sapi_module.name, "cli") != 0) &&
- (strncmp(sapi_module.name, "embed", 5) != 0)
- ) { php_sqlite3_error(db_obj, "Not supported in multithreaded Web servers");
- RETURN_FALSE;
- }
- #endif
- if (!SQLITE3G(extension_dir)) {
- php_sqlite3_error(db_obj, "SQLite Extension are disabled");
- RETURN_FALSE;
- }
- if (extension_len == 0) {
- php_sqlite3_error(db_obj, "Empty string as an extension");
- RETURN_FALSE;
- }
- extension_dir = SQLITE3G(extension_dir);
- extension_dir_len = strlen(SQLITE3G(extension_dir));
- if (IS_SLASH(extension_dir[extension_dir_len-1])) {
- spprintf(&lib_path, 0, "%s%s", extension_dir, extension);
- } else {
- spprintf(&lib_path, 0, "%s%c%s", extension_dir, DEFAULT_SLASH, extension);
- }
- if (!VCWD_REALPATH(lib_path, fullpath)) {
- php_sqlite3_error(db_obj, "Unable to load extension at '%s'", lib_path);
- efree(lib_path);
- RETURN_FALSE;
- }
- efree(lib_path);
- if (strncmp(fullpath, extension_dir, extension_dir_len) != 0) {
- php_sqlite3_error(db_obj, "Unable to open extensions outside the defined directory");
- RETURN_FALSE;
- }
- /* Extension loading should only be enabled for when we attempt to load */
- sqlite3_enable_load_extension(db_obj->db, 1);
- if (sqlite3_load_extension(db_obj->db, fullpath, 0, &errtext) != SQLITE_OK) {
- php_sqlite3_error(db_obj, "%s", errtext);
- sqlite3_free(errtext);
- sqlite3_enable_load_extension(db_obj->db, 0);
- RETURN_FALSE;
- }
- sqlite3_enable_load_extension(db_obj->db, 0);
- RETURN_TRUE;
- }
- /* }}} */
- #endif
- /* {{{ Returns the number of database rows that were changed (or inserted or deleted) by the most recent SQL statement. */
- PHP_METHOD(SQLite3, changes)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- RETURN_LONG(sqlite3_changes(db_obj->db));
- }
- /* }}} */
- /* {{{ Returns a string that has been properly escaped. */
- PHP_METHOD(SQLite3, escapeString)
- {
- zend_string *sql;
- char *ret;
- if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "S", &sql)) {
- RETURN_THROWS();
- }
- if (ZSTR_LEN(sql)) {
- ret = sqlite3_mprintf("%q", ZSTR_VAL(sql));
- if (ret) {
- RETVAL_STRING(ret);
- sqlite3_free(ret);
- }
- } else {
- RETURN_EMPTY_STRING();
- }
- }
- /* }}} */
- /* {{{ Returns a prepared SQL statement for execution. */
- PHP_METHOD(SQLite3, prepare)
- {
- php_sqlite3_db_object *db_obj;
- php_sqlite3_stmt *stmt_obj;
- zval *object = ZEND_THIS;
- zend_string *sql;
- int errcode;
- php_sqlite3_free_list *free_item;
- db_obj = Z_SQLITE3_DB_P(object);
- if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "S", &sql)) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- if (!ZSTR_LEN(sql)) {
- RETURN_FALSE;
- }
- object_init_ex(return_value, php_sqlite3_stmt_entry);
- stmt_obj = Z_SQLITE3_STMT_P(return_value);
- stmt_obj->db_obj = db_obj;
- ZVAL_OBJ_COPY(&stmt_obj->db_obj_zval, Z_OBJ_P(object));
- errcode = sqlite3_prepare_v2(db_obj->db, ZSTR_VAL(sql), ZSTR_LEN(sql), &(stmt_obj->stmt), NULL);
- if (errcode != SQLITE_OK) {
- php_sqlite3_error(db_obj, "Unable to prepare statement: %d, %s", errcode, sqlite3_errmsg(db_obj->db));
- zval_ptr_dtor(return_value);
- RETURN_FALSE;
- }
- stmt_obj->initialised = 1;
- free_item = emalloc(sizeof(php_sqlite3_free_list));
- free_item->stmt_obj = stmt_obj;
- ZVAL_OBJ(&free_item->stmt_obj_zval, Z_OBJ_P(return_value));
- zend_llist_add_element(&(db_obj->free_list), &free_item);
- }
- /* }}} */
- /* {{{ Returns true or false, for queries that return data it will return a SQLite3Result object. */
- PHP_METHOD(SQLite3, query)
- {
- php_sqlite3_db_object *db_obj;
- php_sqlite3_result *result;
- php_sqlite3_stmt *stmt_obj;
- zval *object = ZEND_THIS;
- zval stmt;
- zend_string *sql;
- char *errtext = NULL;
- int return_code;
- db_obj = Z_SQLITE3_DB_P(object);
- if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "S", &sql)) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- if (!ZSTR_LEN(sql)) {
- RETURN_FALSE;
- }
- /* If there was no return value then just execute the query */
- if (!USED_RET()) {
- if (sqlite3_exec(db_obj->db, ZSTR_VAL(sql), NULL, NULL, &errtext) != SQLITE_OK) {
- php_sqlite3_error(db_obj, "%s", errtext);
- sqlite3_free(errtext);
- }
- RETURN_FALSE;
- }
- object_init_ex(&stmt, php_sqlite3_stmt_entry);
- stmt_obj = Z_SQLITE3_STMT_P(&stmt);
- stmt_obj->db_obj = db_obj;
- ZVAL_OBJ_COPY(&stmt_obj->db_obj_zval, Z_OBJ_P(object));
- return_code = sqlite3_prepare_v2(db_obj->db, ZSTR_VAL(sql), ZSTR_LEN(sql), &(stmt_obj->stmt), NULL);
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(db_obj, "Unable to prepare statement: %d, %s", return_code, sqlite3_errmsg(db_obj->db));
- zval_ptr_dtor(&stmt);
- RETURN_FALSE;
- }
- stmt_obj->initialised = 1;
- object_init_ex(return_value, php_sqlite3_result_entry);
- result = Z_SQLITE3_RESULT_P(return_value);
- result->db_obj = db_obj;
- result->stmt_obj = stmt_obj;
- ZVAL_OBJ(&result->stmt_obj_zval, Z_OBJ(stmt));
- return_code = sqlite3_step(result->stmt_obj->stmt);
- switch (return_code) {
- case SQLITE_ROW: /* Valid Row */
- case SQLITE_DONE: /* Valid but no results */
- {
- php_sqlite3_free_list *free_item;
- free_item = emalloc(sizeof(php_sqlite3_free_list));
- free_item->stmt_obj = stmt_obj;
- free_item->stmt_obj_zval = stmt;
- zend_llist_add_element(&(db_obj->free_list), &free_item);
- sqlite3_reset(result->stmt_obj->stmt);
- break;
- }
- default:
- if (!EG(exception)) {
- php_sqlite3_error(db_obj, "Unable to execute statement: %s", sqlite3_errmsg(db_obj->db));
- }
- sqlite3_finalize(stmt_obj->stmt);
- stmt_obj->initialised = 0;
- zval_ptr_dtor(return_value);
- RETURN_FALSE;
- }
- }
- /* }}} */
- static void sqlite_value_to_zval(sqlite3_stmt *stmt, int column, zval *data) /* {{{ */
- {
- sqlite3_int64 val;
- switch (sqlite3_column_type(stmt, column)) {
- case SQLITE_INTEGER:
- val = sqlite3_column_int64(stmt, column);
- #if LONG_MAX <= 2147483647
- if (val > ZEND_LONG_MAX || val < ZEND_LONG_MIN) {
- ZVAL_STRINGL(data, (char *)sqlite3_column_text(stmt, column), sqlite3_column_bytes(stmt, column));
- } else {
- #endif
- ZVAL_LONG(data, (zend_long) val);
- #if LONG_MAX <= 2147483647
- }
- #endif
- break;
- case SQLITE_FLOAT:
- ZVAL_DOUBLE(data, sqlite3_column_double(stmt, column));
- break;
- case SQLITE_NULL:
- ZVAL_NULL(data);
- break;
- case SQLITE3_TEXT:
- ZVAL_STRING(data, (char*)sqlite3_column_text(stmt, column));
- break;
- case SQLITE_BLOB:
- default:
- ZVAL_STRINGL(data, (char*)sqlite3_column_blob(stmt, column), sqlite3_column_bytes(stmt, column));
- }
- }
- /* }}} */
- /* {{{ Returns a string of the first column, or an array of the entire row. */
- PHP_METHOD(SQLite3, querySingle)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- zend_string *sql;
- char *errtext = NULL;
- int return_code;
- bool entire_row = 0;
- sqlite3_stmt *stmt;
- db_obj = Z_SQLITE3_DB_P(object);
- if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "S|b", &sql, &entire_row)) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- if (!ZSTR_LEN(sql)) {
- RETURN_FALSE;
- }
- /* If there was no return value then just execute the query */
- if (!USED_RET()) {
- if (sqlite3_exec(db_obj->db, ZSTR_VAL(sql), NULL, NULL, &errtext) != SQLITE_OK) {
- php_sqlite3_error(db_obj, "%s", errtext);
- sqlite3_free(errtext);
- }
- RETURN_FALSE;
- }
- return_code = sqlite3_prepare_v2(db_obj->db, ZSTR_VAL(sql), ZSTR_LEN(sql), &stmt, NULL);
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(db_obj, "Unable to prepare statement: %d, %s", return_code, sqlite3_errmsg(db_obj->db));
- RETURN_FALSE;
- }
- return_code = sqlite3_step(stmt);
- switch (return_code) {
- case SQLITE_ROW: /* Valid Row */
- {
- if (!entire_row) {
- sqlite_value_to_zval(stmt, 0, return_value);
- } else {
- int i = 0;
- array_init(return_value);
- for (i = 0; i < sqlite3_data_count(stmt); i++) {
- zval data;
- sqlite_value_to_zval(stmt, i, &data);
- add_assoc_zval(return_value, (char*)sqlite3_column_name(stmt, i), &data);
- }
- }
- break;
- }
- case SQLITE_DONE: /* Valid but no results */
- {
- if (!entire_row) {
- RETVAL_NULL();
- } else {
- RETVAL_EMPTY_ARRAY();
- }
- break;
- }
- default:
- if (!EG(exception)) {
- php_sqlite3_error(db_obj, "Unable to execute statement: %s", sqlite3_errmsg(db_obj->db));
- }
- RETVAL_FALSE;
- }
- sqlite3_finalize(stmt);
- }
- /* }}} */
- static int sqlite3_do_callback(struct php_sqlite3_fci *fc, zval *cb, int argc, sqlite3_value **argv, sqlite3_context *context, int is_agg) /* {{{ */
- {
- zval *zargs = NULL;
- zval retval;
- int i;
- int ret;
- int fake_argc;
- php_sqlite3_agg_context *agg_context = NULL;
- if (is_agg) {
- is_agg = 2;
- }
- fake_argc = argc + is_agg;
- fc->fci.size = sizeof(fc->fci);
- ZVAL_COPY_VALUE(&fc->fci.function_name, cb);
- fc->fci.object = NULL;
- fc->fci.retval = &retval;
- fc->fci.param_count = fake_argc;
- /* build up the params */
- if (fake_argc) {
- zargs = (zval *)safe_emalloc(fake_argc, sizeof(zval), 0);
- }
- if (is_agg) {
- /* summon the aggregation context */
- agg_context = (php_sqlite3_agg_context *)sqlite3_aggregate_context(context, sizeof(php_sqlite3_agg_context));
- if (Z_ISUNDEF(agg_context->zval_context)) {
- ZVAL_NULL(&agg_context->zval_context);
- }
- ZVAL_COPY(&zargs[0], &agg_context->zval_context);
- ZVAL_LONG(&zargs[1], agg_context->row_count);
- }
- for (i = 0; i < argc; i++) {
- switch (sqlite3_value_type(argv[i])) {
- case SQLITE_INTEGER:
- #if ZEND_LONG_MAX > 2147483647
- ZVAL_LONG(&zargs[i + is_agg], sqlite3_value_int64(argv[i]));
- #else
- ZVAL_LONG(&zargs[i + is_agg], sqlite3_value_int(argv[i]));
- #endif
- break;
- case SQLITE_FLOAT:
- ZVAL_DOUBLE(&zargs[i + is_agg], sqlite3_value_double(argv[i]));
- break;
- case SQLITE_NULL:
- ZVAL_NULL(&zargs[i + is_agg]);
- break;
- case SQLITE_BLOB:
- case SQLITE3_TEXT:
- default:
- ZVAL_STRINGL(&zargs[i + is_agg], (char*)sqlite3_value_text(argv[i]), sqlite3_value_bytes(argv[i]));
- break;
- }
- }
- fc->fci.params = zargs;
- if ((ret = zend_call_function(&fc->fci, &fc->fcc)) == FAILURE) {
- php_error_docref(NULL, E_WARNING, "An error occurred while invoking the callback");
- }
- if (is_agg) {
- zval_ptr_dtor(&zargs[0]);
- }
- /* clean up the params */
- if (fake_argc) {
- for (i = is_agg; i < argc + is_agg; i++) {
- zval_ptr_dtor(&zargs[i]);
- }
- if (is_agg) {
- zval_ptr_dtor(&zargs[1]);
- }
- efree(zargs);
- }
- if (!is_agg || !argv) {
- /* only set the sqlite return value if we are a scalar function,
- * or if we are finalizing an aggregate */
- if (!Z_ISUNDEF(retval)) {
- switch (Z_TYPE(retval)) {
- case IS_LONG:
- #if ZEND_LONG_MAX > 2147483647
- sqlite3_result_int64(context, Z_LVAL(retval));
- #else
- sqlite3_result_int(context, Z_LVAL(retval));
- #endif
- break;
- case IS_NULL:
- sqlite3_result_null(context);
- break;
- case IS_DOUBLE:
- sqlite3_result_double(context, Z_DVAL(retval));
- break;
- default: {
- zend_string *str = zval_try_get_string(&retval);
- if (UNEXPECTED(!str)) {
- ret = FAILURE;
- break;
- }
- sqlite3_result_text(context, ZSTR_VAL(str), ZSTR_LEN(str), SQLITE_TRANSIENT);
- zend_string_release(str);
- break;
- }
- }
- } else {
- sqlite3_result_error(context, "failed to invoke callback", 0);
- }
- if (agg_context && !Z_ISUNDEF(agg_context->zval_context)) {
- zval_ptr_dtor(&agg_context->zval_context);
- }
- } else {
- /* we're stepping in an aggregate; the return value goes into
- * the context */
- if (agg_context && !Z_ISUNDEF(agg_context->zval_context)) {
- zval_ptr_dtor(&agg_context->zval_context);
- }
- ZVAL_COPY_VALUE(&agg_context->zval_context, &retval);
- ZVAL_UNDEF(&retval);
- }
- if (!Z_ISUNDEF(retval)) {
- zval_ptr_dtor(&retval);
- }
- return ret;
- }
- /* }}}*/
- static void php_sqlite3_callback_func(sqlite3_context *context, int argc, sqlite3_value **argv) /* {{{ */
- {
- php_sqlite3_func *func = (php_sqlite3_func *)sqlite3_user_data(context);
- sqlite3_do_callback(&func->afunc, &func->func, argc, argv, context, 0);
- }
- /* }}}*/
- static void php_sqlite3_callback_step(sqlite3_context *context, int argc, sqlite3_value **argv) /* {{{ */
- {
- php_sqlite3_func *func = (php_sqlite3_func *)sqlite3_user_data(context);
- php_sqlite3_agg_context *agg_context = (php_sqlite3_agg_context *)sqlite3_aggregate_context(context, sizeof(php_sqlite3_agg_context));
- agg_context->row_count++;
- sqlite3_do_callback(&func->astep, &func->step, argc, argv, context, 1);
- }
- /* }}} */
- static void php_sqlite3_callback_final(sqlite3_context *context) /* {{{ */
- {
- php_sqlite3_func *func = (php_sqlite3_func *)sqlite3_user_data(context);
- php_sqlite3_agg_context *agg_context = (php_sqlite3_agg_context *)sqlite3_aggregate_context(context, sizeof(php_sqlite3_agg_context));
- agg_context->row_count = 0;
- sqlite3_do_callback(&func->afini, &func->fini, 0, NULL, context, 1);
- }
- /* }}} */
- static int php_sqlite3_callback_compare(void *coll, int a_len, const void *a, int b_len, const void* b) /* {{{ */
- {
- php_sqlite3_collation *collation = (php_sqlite3_collation*)coll;
- zval zargs[2];
- zval retval;
- int ret;
- // Exception occurred on previous callback. Don't attempt to call function.
- if (EG(exception)) {
- return 0;
- }
- collation->fci.fci.size = (sizeof(collation->fci.fci));
- ZVAL_COPY_VALUE(&collation->fci.fci.function_name, &collation->cmp_func);
- collation->fci.fci.object = NULL;
- collation->fci.fci.retval = &retval;
- collation->fci.fci.param_count = 2;
- ZVAL_STRINGL(&zargs[0], a, a_len);
- ZVAL_STRINGL(&zargs[1], b, b_len);
- collation->fci.fci.params = zargs;
- if ((ret = zend_call_function(&collation->fci.fci, &collation->fci.fcc)) == FAILURE) {
- php_error_docref(NULL, E_WARNING, "An error occurred while invoking the compare callback");
- }
- zval_ptr_dtor(&zargs[0]);
- zval_ptr_dtor(&zargs[1]);
- if (EG(exception)) {
- ret = 0;
- } else if (Z_TYPE(retval) != IS_LONG){
- //retval ought to contain a ZVAL_LONG by now
- // (the result of a comparison, i.e. most likely -1, 0, or 1)
- //I suppose we could accept any scalar return type, though.
- php_error_docref(NULL, E_WARNING, "An error occurred while invoking the compare callback (invalid return type). Collation behaviour is undefined.");
- } else {
- ret = Z_LVAL(retval);
- }
- zval_ptr_dtor(&retval);
- return ret;
- }
- /* }}} */
- /* {{{ Allows registration of a PHP function as a SQLite UDF that can be called within SQL statements. */
- PHP_METHOD(SQLite3, createFunction)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- php_sqlite3_func *func;
- char *sql_func;
- size_t sql_func_len;
- zend_fcall_info fci;
- zend_fcall_info_cache fcc;
- zend_long sql_func_num_args = -1;
- zend_long flags = 0;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "sf|ll", &sql_func, &sql_func_len, &fci, &fcc, &sql_func_num_args, &flags) == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- if (!sql_func_len) {
- RETURN_FALSE;
- }
- func = (php_sqlite3_func *)ecalloc(1, sizeof(*func));
- if (sqlite3_create_function(db_obj->db, sql_func, sql_func_num_args, flags | SQLITE_UTF8, func, php_sqlite3_callback_func, NULL, NULL) == SQLITE_OK) {
- func->func_name = estrdup(sql_func);
- ZVAL_COPY(&func->func, &fci.function_name);
- func->argc = sql_func_num_args;
- func->next = db_obj->funcs;
- db_obj->funcs = func;
- RETURN_TRUE;
- }
- efree(func);
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ Allows registration of a PHP function for use as an aggregate. */
- PHP_METHOD(SQLite3, createAggregate)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- php_sqlite3_func *func;
- char *sql_func;
- size_t sql_func_len;
- zend_fcall_info step_fci, fini_fci;
- zend_fcall_info_cache step_fcc, fini_fcc;
- zend_long sql_func_num_args = -1;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "sff|l", &sql_func, &sql_func_len, &step_fci, &step_fcc, &fini_fci, &fini_fcc, &sql_func_num_args) == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- if (!sql_func_len) {
- RETURN_FALSE;
- }
- func = (php_sqlite3_func *)ecalloc(1, sizeof(*func));
- if (sqlite3_create_function(db_obj->db, sql_func, sql_func_num_args, SQLITE_UTF8, func, NULL, php_sqlite3_callback_step, php_sqlite3_callback_final) == SQLITE_OK) {
- func->func_name = estrdup(sql_func);
- ZVAL_COPY(&func->step, &step_fci.function_name);
- ZVAL_COPY(&func->fini, &fini_fci.function_name);
- func->argc = sql_func_num_args;
- func->next = db_obj->funcs;
- db_obj->funcs = func;
- RETURN_TRUE;
- }
- efree(func);
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ Registers a PHP function as a comparator that can be used with the SQL COLLATE operator. Callback must accept two strings and return an integer (as strcmp()). */
- PHP_METHOD(SQLite3, createCollation)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- php_sqlite3_collation *collation;
- char *collation_name;
- size_t collation_name_len;
- zend_fcall_info fci;
- zend_fcall_info_cache fcc;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "sf", &collation_name, &collation_name_len, &fci, &fcc) == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- if (!collation_name_len) {
- RETURN_FALSE;
- }
- collation = (php_sqlite3_collation *)ecalloc(1, sizeof(*collation));
- if (sqlite3_create_collation(db_obj->db, collation_name, SQLITE_UTF8, collation, php_sqlite3_callback_compare) == SQLITE_OK) {
- collation->collation_name = estrdup(collation_name);
- ZVAL_COPY(&collation->cmp_func, &fci.function_name);
- collation->next = db_obj->collations;
- db_obj->collations = collation;
- RETURN_TRUE;
- }
- efree(collation);
- RETURN_FALSE;
- }
- /* }}} */
- typedef struct {
- sqlite3_blob *blob;
- size_t position;
- size_t size;
- int flags;
- } php_stream_sqlite3_data;
- static ssize_t php_sqlite3_stream_write(php_stream *stream, const char *buf, size_t count)
- {
- php_stream_sqlite3_data *sqlite3_stream = (php_stream_sqlite3_data *) stream->abstract;
- if (sqlite3_stream->flags & SQLITE_OPEN_READONLY) {
- php_error_docref(NULL, E_WARNING, "Can't write to blob stream: is open as read only");
- return -1;
- }
- if (sqlite3_stream->position + count > sqlite3_stream->size) {
- php_error_docref(NULL, E_WARNING, "It is not possible to increase the size of a BLOB");
- return -1;
- }
- if (sqlite3_blob_write(sqlite3_stream->blob, buf, count, sqlite3_stream->position) != SQLITE_OK) {
- return -1;
- }
- if (sqlite3_stream->position + count >= sqlite3_stream->size) {
- stream->eof = 1;
- sqlite3_stream->position = sqlite3_stream->size;
- }
- else {
- sqlite3_stream->position += count;
- }
- return count;
- }
- static ssize_t php_sqlite3_stream_read(php_stream *stream, char *buf, size_t count)
- {
- php_stream_sqlite3_data *sqlite3_stream = (php_stream_sqlite3_data *) stream->abstract;
- if (sqlite3_stream->position + count >= sqlite3_stream->size) {
- count = sqlite3_stream->size - sqlite3_stream->position;
- stream->eof = 1;
- }
- if (count) {
- if (sqlite3_blob_read(sqlite3_stream->blob, buf, count, sqlite3_stream->position) != SQLITE_OK) {
- return -1;
- }
- sqlite3_stream->position += count;
- }
- return count;
- }
- static int php_sqlite3_stream_close(php_stream *stream, int close_handle)
- {
- php_stream_sqlite3_data *sqlite3_stream = (php_stream_sqlite3_data *) stream->abstract;
- if (sqlite3_blob_close(sqlite3_stream->blob) != SQLITE_OK) {
- /* Error occurred, but it still closed */
- }
- efree(sqlite3_stream);
- return 0;
- }
- static int php_sqlite3_stream_flush(php_stream *stream)
- {
- /* do nothing */
- return 0;
- }
- /* {{{ */
- static int php_sqlite3_stream_seek(php_stream *stream, zend_off_t offset, int whence, zend_off_t *newoffs)
- {
- php_stream_sqlite3_data *sqlite3_stream = (php_stream_sqlite3_data *) stream->abstract;
- switch(whence) {
- case SEEK_CUR:
- if (offset < 0) {
- if (sqlite3_stream->position < (size_t)(-offset)) {
- sqlite3_stream->position = 0;
- *newoffs = -1;
- return -1;
- } else {
- sqlite3_stream->position = sqlite3_stream->position + offset;
- *newoffs = sqlite3_stream->position;
- stream->eof = 0;
- return 0;
- }
- } else {
- if (sqlite3_stream->position + (size_t)(offset) > sqlite3_stream->size) {
- sqlite3_stream->position = sqlite3_stream->size;
- *newoffs = -1;
- return -1;
- } else {
- sqlite3_stream->position = sqlite3_stream->position + offset;
- *newoffs = sqlite3_stream->position;
- stream->eof = 0;
- return 0;
- }
- }
- case SEEK_SET:
- if (sqlite3_stream->size < (size_t)(offset)) {
- sqlite3_stream->position = sqlite3_stream->size;
- *newoffs = -1;
- return -1;
- } else {
- sqlite3_stream->position = offset;
- *newoffs = sqlite3_stream->position;
- stream->eof = 0;
- return 0;
- }
- case SEEK_END:
- if (offset > 0) {
- sqlite3_stream->position = sqlite3_stream->size;
- *newoffs = -1;
- return -1;
- } else if (sqlite3_stream->size < (size_t)(-offset)) {
- sqlite3_stream->position = 0;
- *newoffs = -1;
- return -1;
- } else {
- sqlite3_stream->position = sqlite3_stream->size + offset;
- *newoffs = sqlite3_stream->position;
- stream->eof = 0;
- return 0;
- }
- default:
- *newoffs = sqlite3_stream->position;
- return -1;
- }
- }
- /* }}} */
- static int php_sqlite3_stream_cast(php_stream *stream, int castas, void **ret)
- {
- return FAILURE;
- }
- static int php_sqlite3_stream_stat(php_stream *stream, php_stream_statbuf *ssb)
- {
- php_stream_sqlite3_data *sqlite3_stream = (php_stream_sqlite3_data *) stream->abstract;
- ssb->sb.st_size = sqlite3_stream->size;
- return 0;
- }
- static const php_stream_ops php_stream_sqlite3_ops = {
- php_sqlite3_stream_write,
- php_sqlite3_stream_read,
- php_sqlite3_stream_close,
- php_sqlite3_stream_flush,
- "SQLite3",
- php_sqlite3_stream_seek,
- php_sqlite3_stream_cast,
- php_sqlite3_stream_stat,
- NULL
- };
- /* {{{ Open a blob as a stream which we can read / write to. */
- PHP_METHOD(SQLite3, openBlob)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- char *table, *column, *dbname = "main", *mode = "rb";
- size_t table_len, column_len, dbname_len;
- zend_long rowid, flags = SQLITE_OPEN_READONLY, sqlite_flags = 0;
- sqlite3_blob *blob = NULL;
- php_stream_sqlite3_data *sqlite3_stream;
- php_stream *stream;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "ssl|pl", &table, &table_len, &column, &column_len, &rowid, &dbname, &dbname_len, &flags) == FAILURE) {
- RETURN_THROWS();
- }
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- sqlite_flags = (flags & SQLITE_OPEN_READWRITE) ? 1 : 0;
- if (sqlite3_blob_open(db_obj->db, dbname, table, column, rowid, sqlite_flags, &blob) != SQLITE_OK) {
- php_sqlite3_error(db_obj, "Unable to open blob: %s", sqlite3_errmsg(db_obj->db));
- RETURN_FALSE;
- }
- sqlite3_stream = emalloc(sizeof(php_stream_sqlite3_data));
- sqlite3_stream->blob = blob;
- sqlite3_stream->flags = flags;
- sqlite3_stream->position = 0;
- sqlite3_stream->size = sqlite3_blob_bytes(blob);
- if (sqlite_flags != 0) {
- mode = "r+b";
- }
- stream = php_stream_alloc(&php_stream_sqlite3_ops, sqlite3_stream, 0, mode);
- if (stream) {
- php_stream_to_zval(stream, return_value);
- } else {
- RETURN_FALSE;
- }
- }
- /* }}} */
- /* {{{ Enables an exception error mode. */
- PHP_METHOD(SQLite3, enableExceptions)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- bool enableExceptions = 0;
- db_obj = Z_SQLITE3_DB_P(object);
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "|b", &enableExceptions) == FAILURE) {
- RETURN_THROWS();
- }
- RETVAL_BOOL(db_obj->exception);
- db_obj->exception = enableExceptions;
- }
- /* }}} */
- /* {{{ Register a callback function to be used as an authorizer by SQLite. The callback should return SQLite3::OK, SQLite3::IGNORE or SQLite3::DENY. */
- PHP_METHOD(SQLite3, setAuthorizer)
- {
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- db_obj = Z_SQLITE3_DB_P(object);
- zend_fcall_info fci;
- zend_fcall_info_cache fcc;
- ZEND_PARSE_PARAMETERS_START(1, 1)
- Z_PARAM_FUNC_OR_NULL(fci, fcc)
- ZEND_PARSE_PARAMETERS_END();
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- /* Clear previously set callback */
- if (ZEND_FCI_INITIALIZED(db_obj->authorizer_fci)) {
- zval_ptr_dtor(&db_obj->authorizer_fci.function_name);
- db_obj->authorizer_fci.size = 0;
- }
- /* Only enable userland authorizer if argument is not NULL */
- if (ZEND_FCI_INITIALIZED(fci)) {
- db_obj->authorizer_fci = fci;
- Z_ADDREF(db_obj->authorizer_fci.function_name);
- db_obj->authorizer_fcc = fcc;
- }
- RETURN_TRUE;
- }
- /* }}} */
- #if SQLITE_VERSION_NUMBER >= 3006011
- /* {{{ Backups the current database to another one. */
- PHP_METHOD(SQLite3, backup)
- {
- php_sqlite3_db_object *source_obj;
- php_sqlite3_db_object *destination_obj;
- char *source_dbname = "main", *destination_dbname = "main";
- size_t source_dbname_length, destination_dbname_length;
- zval *source_zval = ZEND_THIS;
- zval *destination_zval;
- sqlite3_backup *dbBackup;
- int rc; // Return code
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "O|pp", &destination_zval, php_sqlite3_sc_entry, &source_dbname, &source_dbname_length, &destination_dbname, &destination_dbname_length) == FAILURE) {
- RETURN_THROWS();
- }
- source_obj = Z_SQLITE3_DB_P(source_zval);
- SQLITE3_CHECK_INITIALIZED(source_obj, source_obj->initialised, SQLite3)
- destination_obj = Z_SQLITE3_DB_P(destination_zval);
- SQLITE3_CHECK_INITIALIZED(destination_obj, destination_obj->initialised, SQLite3)
- dbBackup = sqlite3_backup_init(destination_obj->db, destination_dbname, source_obj->db, source_dbname);
- if (dbBackup) {
- do {
- rc = sqlite3_backup_step(dbBackup, -1);
- } while (rc == SQLITE_OK);
- /* Release resources allocated by backup_init(). */
- rc = sqlite3_backup_finish(dbBackup);
- }
- else {
- rc = sqlite3_errcode(source_obj->db);
- }
- if (rc != SQLITE_OK) {
- if (rc == SQLITE_BUSY) {
- php_sqlite3_error(source_obj, "Backup failed: source database is busy");
- }
- else if (rc == SQLITE_LOCKED) {
- php_sqlite3_error(source_obj, "Backup failed: source database is locked");
- }
- else {
- php_sqlite3_error(source_obj, "Backup failed: %d, %s", rc, sqlite3_errmsg(source_obj->db));
- }
- RETURN_FALSE;
- }
- RETURN_TRUE;
- }
- /* }}} */
- #endif
- /* {{{ Returns the number of parameters within the prepared statement. */
- PHP_METHOD(SQLite3Stmt, paramCount)
- {
- php_sqlite3_stmt *stmt_obj;
- zval *object = ZEND_THIS;
- stmt_obj = Z_SQLITE3_STMT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(stmt_obj->db_obj, stmt_obj->initialised, SQLite3);
- SQLITE3_CHECK_INITIALIZED_STMT(stmt_obj->stmt, SQLite3Stmt);
- RETURN_LONG(sqlite3_bind_parameter_count(stmt_obj->stmt));
- }
- /* }}} */
- /* {{{ Closes the prepared statement. */
- PHP_METHOD(SQLite3Stmt, close)
- {
- php_sqlite3_stmt *stmt_obj;
- zval *object = ZEND_THIS;
- stmt_obj = Z_SQLITE3_STMT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(stmt_obj->db_obj, stmt_obj->initialised, SQLite3);
- if(stmt_obj->db_obj) {
- zend_llist_del_element(&(stmt_obj->db_obj->free_list), object, (int (*)(void *, void *)) php_sqlite3_compare_stmt_zval_free);
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ Reset the prepared statement to the state before it was executed, bindings still remain. */
- PHP_METHOD(SQLite3Stmt, reset)
- {
- php_sqlite3_stmt *stmt_obj;
- zval *object = ZEND_THIS;
- stmt_obj = Z_SQLITE3_STMT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(stmt_obj->db_obj, stmt_obj->initialised, SQLite3);
- SQLITE3_CHECK_INITIALIZED_STMT(stmt_obj->stmt, SQLite3Stmt);
- if (sqlite3_reset(stmt_obj->stmt) != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to reset statement: %s", sqlite3_errmsg(sqlite3_db_handle(stmt_obj->stmt)));
- RETURN_FALSE;
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ Clear all current bound parameters. */
- PHP_METHOD(SQLite3Stmt, clear)
- {
- php_sqlite3_stmt *stmt_obj;
- zval *object = ZEND_THIS;
- stmt_obj = Z_SQLITE3_STMT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(stmt_obj->db_obj, stmt_obj->initialised, SQLite3);
- SQLITE3_CHECK_INITIALIZED_STMT(stmt_obj->stmt, SQLite3Stmt);
- if (sqlite3_clear_bindings(stmt_obj->stmt) != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to clear statement: %s", sqlite3_errmsg(sqlite3_db_handle(stmt_obj->stmt)));
- RETURN_FALSE;
- }
- if (stmt_obj->bound_params) {
- zend_hash_destroy(stmt_obj->bound_params);
- FREE_HASHTABLE(stmt_obj->bound_params);
- stmt_obj->bound_params = NULL;
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ Returns true if a statement is definitely read only */
- PHP_METHOD(SQLite3Stmt, readOnly)
- {
- php_sqlite3_stmt *stmt_obj;
- zval *object = ZEND_THIS;
- stmt_obj = Z_SQLITE3_STMT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(stmt_obj->db_obj, stmt_obj->initialised, SQLite3);
- SQLITE3_CHECK_INITIALIZED_STMT(stmt_obj->stmt, SQLite3Stmt);
- if (sqlite3_stmt_readonly(stmt_obj->stmt)) {
- RETURN_TRUE;
- }
- RETURN_FALSE;
- }
- /* }}} */
- /* bind parameters to a statement before execution */
- static int php_sqlite3_bind_params(php_sqlite3_stmt *stmt_obj) /* {{{ */
- {
- struct php_sqlite3_bound_param *param;
- int return_code;
- if (stmt_obj->bound_params) {
- ZEND_HASH_FOREACH_PTR(stmt_obj->bound_params, param) {
- zval *parameter;
- /* parameter must be a reference? */
- if (Z_ISREF(param->parameter)) {
- parameter = Z_REFVAL(param->parameter);
- } else {
- parameter = ¶m->parameter;
- }
- /* If the ZVAL is null then it should be bound as that */
- if (Z_TYPE_P(parameter) == IS_NULL) {
- return_code = sqlite3_bind_null(stmt_obj->stmt, param->param_number);
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to bind parameter number " ZEND_LONG_FMT " (%d)", param->param_number, return_code);
- }
- continue;
- }
- switch (param->type) {
- case SQLITE_INTEGER:
- convert_to_long(parameter);
- #if ZEND_LONG_MAX > 2147483647
- return_code = sqlite3_bind_int64(stmt_obj->stmt, param->param_number, Z_LVAL_P(parameter));
- #else
- return_code = sqlite3_bind_int(stmt_obj->stmt, param->param_number, Z_LVAL_P(parameter));
- #endif
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to bind parameter number " ZEND_LONG_FMT " (%d)", param->param_number, return_code);
- }
- break;
- case SQLITE_FLOAT:
- convert_to_double(parameter);
- return_code = sqlite3_bind_double(stmt_obj->stmt, param->param_number, Z_DVAL_P(parameter));
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to bind parameter number " ZEND_LONG_FMT " (%d)", param->param_number, return_code);
- }
- break;
- case SQLITE_BLOB:
- {
- php_stream *stream = NULL;
- zend_string *buffer = NULL;
- if (Z_TYPE_P(parameter) == IS_RESOURCE) {
- php_stream_from_zval_no_verify(stream, parameter);
- if (stream == NULL) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to read stream for parameter %ld", param->param_number);
- return FAILURE;
- }
- buffer = php_stream_copy_to_mem(stream, PHP_STREAM_COPY_ALL, 0);
- } else {
- buffer = zval_get_string(parameter);
- }
- if (buffer) {
- return_code = sqlite3_bind_blob(stmt_obj->stmt, param->param_number, ZSTR_VAL(buffer), ZSTR_LEN(buffer), SQLITE_TRANSIENT);
- zend_string_release_ex(buffer, 0);
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to bind parameter number " ZEND_LONG_FMT " (%d)", param->param_number, return_code);
- }
- } else {
- return_code = sqlite3_bind_null(stmt_obj->stmt, param->param_number);
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to bind parameter number " ZEND_LONG_FMT " (%d)", param->param_number, return_code);
- }
- }
- break;
- }
- case SQLITE3_TEXT: {
- zend_string *str = zval_try_get_string(parameter);
- if (UNEXPECTED(!str)) {
- return FAILURE;
- }
- return_code = sqlite3_bind_text(stmt_obj->stmt, param->param_number, ZSTR_VAL(str), ZSTR_LEN(str), SQLITE_TRANSIENT);
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to bind parameter number " ZEND_LONG_FMT " (%d)", param->param_number, return_code);
- }
- zend_string_release(str);
- break;
- }
- case SQLITE_NULL:
- return_code = sqlite3_bind_null(stmt_obj->stmt, param->param_number);
- if (return_code != SQLITE_OK) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to bind parameter number " ZEND_LONG_FMT " (%d)", param->param_number, return_code);
- }
- break;
- default:
- php_sqlite3_error(stmt_obj->db_obj, "Unknown parameter type: %pd for parameter %pd", param->type, param->param_number);
- return FAILURE;
- }
- } ZEND_HASH_FOREACH_END();
- }
- return SUCCESS;
- }
- /* }}} */
- /* {{{ Returns the SQL statement used to prepare the query. If expanded is true, binded parameters and values will be expanded. */
- PHP_METHOD(SQLite3Stmt, getSQL)
- {
- php_sqlite3_stmt *stmt_obj;
- bool expanded = 0;
- zval *object = getThis();
- stmt_obj = Z_SQLITE3_STMT_P(object);
- int bind_rc;
- ZEND_PARSE_PARAMETERS_START(0, 1)
- Z_PARAM_OPTIONAL
- Z_PARAM_BOOL(expanded)
- ZEND_PARSE_PARAMETERS_END();
- SQLITE3_CHECK_INITIALIZED(stmt_obj->db_obj, stmt_obj->initialised, SQLite3);
- SQLITE3_CHECK_INITIALIZED_STMT(stmt_obj->stmt, SQLite3Stmt);
- bind_rc = php_sqlite3_bind_params(stmt_obj);
- if (bind_rc == FAILURE || EG(exception)) {
- RETURN_FALSE;
- }
- if (expanded) {
- #ifdef HAVE_SQLITE3_EXPANDED_SQL
- char *sql = sqlite3_expanded_sql(stmt_obj->stmt);
- RETVAL_STRING(sql);
- sqlite3_free(sql);
- #else
- php_sqlite3_error(stmt_obj->db_obj, "The expanded parameter requires SQLite3 >= 3.14 and %s is installed", sqlite3_libversion());
- RETURN_FALSE;
- #endif
- } else {
- const char *sql = sqlite3_sql(stmt_obj->stmt);
- RETVAL_STRING(sql);
- }
- }
- /* }}} */
- static int register_bound_parameter_to_sqlite(struct php_sqlite3_bound_param *param, php_sqlite3_stmt *stmt) /* {{{ */
- {
- HashTable *hash;
- hash = stmt->bound_params;
- if (!hash) {
- ALLOC_HASHTABLE(hash);
- zend_hash_init(hash, 13, NULL, sqlite3_param_dtor, 0);
- stmt->bound_params = hash;
- }
- /* We need a : prefix to resolve a name to a parameter number */
- if (param->name) {
- if (ZSTR_VAL(param->name)[0] != ':' && ZSTR_VAL(param->name)[0] != '@') {
- /* pre-increment for character + 1 for null */
- zend_string *temp = zend_string_alloc(ZSTR_LEN(param->name) + 1, 0);
- ZSTR_VAL(temp)[0] = ':';
- memmove(ZSTR_VAL(temp) + 1, ZSTR_VAL(param->name), ZSTR_LEN(param->name) + 1);
- param->name = temp;
- } else {
- param->name = zend_string_copy(param->name);
- }
- /* do lookup*/
- param->param_number = sqlite3_bind_parameter_index(stmt->stmt, ZSTR_VAL(param->name));
- }
- if (param->param_number < 1) {
- if (param->name) {
- zend_string_release_ex(param->name, 0);
- }
- return 0;
- }
- if (param->param_number >= 1) {
- zend_hash_index_del(hash, param->param_number);
- }
- if (param->name) {
- zend_hash_update_mem(hash, param->name, param, sizeof(struct php_sqlite3_bound_param));
- } else {
- zend_hash_index_update_mem(hash, param->param_number, param, sizeof(struct php_sqlite3_bound_param));
- }
- return 1;
- }
- /* }}} */
- /* {{{ Best try to map between PHP and SQLite. Default is still text. */
- #define PHP_SQLITE3_SET_TYPE(z, p) \
- switch (Z_TYPE_P(z)) { \
- default: \
- (p).type = SQLITE_TEXT; \
- break; \
- case IS_LONG: \
- case IS_TRUE: \
- case IS_FALSE: \
- (p).type = SQLITE_INTEGER; \
- break; \
- case IS_DOUBLE: \
- (p).type = SQLITE_FLOAT; \
- break; \
- case IS_NULL: \
- (p).type = SQLITE_NULL; \
- break; \
- }
- /* }}} */
- /* {{{ Common implementation of ::bindParam() and ::bindValue */
- static void sqlite3stmt_bind(INTERNAL_FUNCTION_PARAMETERS)
- {
- php_sqlite3_stmt *stmt_obj;
- zval *object = ZEND_THIS;
- struct php_sqlite3_bound_param param = {0};
- zval *parameter;
- stmt_obj = Z_SQLITE3_STMT_P(object);
- param.param_number = -1;
- param.type = SQLITE3_TEXT;
- ZEND_PARSE_PARAMETERS_START(2, 3)
- Z_PARAM_STR_OR_LONG(param.name, param.param_number)
- Z_PARAM_ZVAL(parameter)
- Z_PARAM_OPTIONAL
- Z_PARAM_LONG(param.type)
- ZEND_PARSE_PARAMETERS_END();
- SQLITE3_CHECK_INITIALIZED(stmt_obj->db_obj, stmt_obj->initialised, SQLite3);
- SQLITE3_CHECK_INITIALIZED_STMT(stmt_obj->stmt, SQLite3Stmt);
- ZVAL_COPY(¶m.parameter, parameter);
- if (ZEND_NUM_ARGS() < 3) {
- PHP_SQLITE3_SET_TYPE(parameter, param);
- }
- if (!register_bound_parameter_to_sqlite(¶m, stmt_obj)) {
- if (!Z_ISUNDEF(param.parameter)) {
- zval_ptr_dtor(&(param.parameter));
- ZVAL_UNDEF(¶m.parameter);
- }
- RETURN_FALSE;
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ Bind Parameter to a stmt variable. */
- PHP_METHOD(SQLite3Stmt, bindParam)
- {
- sqlite3stmt_bind(INTERNAL_FUNCTION_PARAM_PASSTHRU);
- }
- /* }}} */
- /* {{{ Bind Value of a parameter to a stmt variable. */
- PHP_METHOD(SQLite3Stmt, bindValue)
- {
- sqlite3stmt_bind(INTERNAL_FUNCTION_PARAM_PASSTHRU);
- }
- /* }}} */
- #undef PHP_SQLITE3_SET_TYPE
- /* {{{ Executes a prepared statement and returns a result set object. */
- PHP_METHOD(SQLite3Stmt, execute)
- {
- php_sqlite3_stmt *stmt_obj;
- php_sqlite3_result *result;
- zval *object = ZEND_THIS;
- int return_code = 0;
- int bind_rc = 0;
- stmt_obj = Z_SQLITE3_STMT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(stmt_obj->db_obj, stmt_obj->initialised, SQLite3);
- /* Always reset statement before execution, see bug #77051 */
- sqlite3_reset(stmt_obj->stmt);
- /* Bind parameters to the statement */
- bind_rc = php_sqlite3_bind_params(stmt_obj);
- if (bind_rc == FAILURE || EG(exception)) {
- RETURN_FALSE;
- }
- return_code = sqlite3_step(stmt_obj->stmt);
- switch (return_code) {
- case SQLITE_ROW: /* Valid Row */
- case SQLITE_DONE: /* Valid but no results */
- {
- sqlite3_reset(stmt_obj->stmt);
- object_init_ex(return_value, php_sqlite3_result_entry);
- result = Z_SQLITE3_RESULT_P(return_value);
- result->is_prepared_statement = 1;
- result->db_obj = stmt_obj->db_obj;
- result->stmt_obj = stmt_obj;
- ZVAL_OBJ_COPY(&result->stmt_obj_zval, Z_OBJ_P(object));
- break;
- }
- case SQLITE_ERROR:
- sqlite3_reset(stmt_obj->stmt);
- ZEND_FALLTHROUGH;
- default:
- if (!EG(exception)) {
- php_sqlite3_error(stmt_obj->db_obj, "Unable to execute statement: %s", sqlite3_errmsg(sqlite3_db_handle(stmt_obj->stmt)));
- }
- zval_ptr_dtor(return_value);
- RETURN_FALSE;
- }
- return;
- }
- /* }}} */
- /* {{{ __constructor for SQLite3Stmt. */
- PHP_METHOD(SQLite3Stmt, __construct)
- {
- php_sqlite3_stmt *stmt_obj;
- php_sqlite3_db_object *db_obj;
- zval *object = ZEND_THIS;
- zval *db_zval;
- zend_string *sql;
- int errcode;
- zend_error_handling error_handling;
- php_sqlite3_free_list *free_item;
- stmt_obj = Z_SQLITE3_STMT_P(object);
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "OS", &db_zval, php_sqlite3_sc_entry, &sql) == FAILURE) {
- RETURN_THROWS();
- }
- db_obj = Z_SQLITE3_DB_P(db_zval);
- zend_replace_error_handling(EH_THROW, NULL, &error_handling);
- SQLITE3_CHECK_INITIALIZED(db_obj, db_obj->initialised, SQLite3)
- zend_restore_error_handling(&error_handling);
- if (!ZSTR_LEN(sql)) {
- RETURN_FALSE;
- }
- stmt_obj->db_obj = db_obj;
- ZVAL_OBJ_COPY(&stmt_obj->db_obj_zval, Z_OBJ_P(db_zval));
- errcode = sqlite3_prepare_v2(db_obj->db, ZSTR_VAL(sql), ZSTR_LEN(sql), &(stmt_obj->stmt), NULL);
- if (errcode != SQLITE_OK) {
- php_sqlite3_error(db_obj, "Unable to prepare statement: %d, %s", errcode, sqlite3_errmsg(db_obj->db));
- zval_ptr_dtor(return_value);
- RETURN_FALSE;
- }
- stmt_obj->initialised = 1;
- free_item = emalloc(sizeof(php_sqlite3_free_list));
- free_item->stmt_obj = stmt_obj;
- //?? free_item->stmt_obj_zval = ZEND_THIS;
- ZVAL_OBJ(&free_item->stmt_obj_zval, Z_OBJ_P(object));
- zend_llist_add_element(&(db_obj->free_list), &free_item);
- }
- /* }}} */
- /* {{{ Number of columns in the result set. */
- PHP_METHOD(SQLite3Result, numColumns)
- {
- php_sqlite3_result *result_obj;
- zval *object = ZEND_THIS;
- result_obj = Z_SQLITE3_RESULT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(result_obj->db_obj, result_obj->stmt_obj->initialised, SQLite3Result)
- RETURN_LONG(sqlite3_column_count(result_obj->stmt_obj->stmt));
- }
- /* }}} */
- /* {{{ Returns the name of the nth column. */
- PHP_METHOD(SQLite3Result, columnName)
- {
- php_sqlite3_result *result_obj;
- zval *object = ZEND_THIS;
- zend_long column = 0;
- char *column_name;
- result_obj = Z_SQLITE3_RESULT_P(object);
- ZEND_PARSE_PARAMETERS_START(1, 1)
- Z_PARAM_LONG(column)
- ZEND_PARSE_PARAMETERS_END();
- SQLITE3_CHECK_INITIALIZED(result_obj->db_obj, result_obj->stmt_obj->initialised, SQLite3Result)
- column_name = (char*) sqlite3_column_name(result_obj->stmt_obj->stmt, column);
- if (column_name == NULL) {
- RETURN_FALSE;
- }
- RETVAL_STRING(column_name);
- }
- /* }}} */
- /* {{{ Returns the type of the nth column. */
- PHP_METHOD(SQLite3Result, columnType)
- {
- php_sqlite3_result *result_obj;
- zval *object = ZEND_THIS;
- zend_long column = 0;
- result_obj = Z_SQLITE3_RESULT_P(object);
- ZEND_PARSE_PARAMETERS_START(1, 1)
- Z_PARAM_LONG(column)
- ZEND_PARSE_PARAMETERS_END();
- SQLITE3_CHECK_INITIALIZED(result_obj->db_obj, result_obj->stmt_obj->initialised, SQLite3Result)
- if (!sqlite3_data_count(result_obj->stmt_obj->stmt)) {
- RETURN_FALSE;
- }
- RETURN_LONG(sqlite3_column_type(result_obj->stmt_obj->stmt, column));
- }
- /* }}} */
- /* {{{ Fetch a result row as both an associative or numerically indexed array or both. */
- PHP_METHOD(SQLite3Result, fetchArray)
- {
- php_sqlite3_result *result_obj;
- zval *object = ZEND_THIS;
- int i, ret;
- zend_long mode = PHP_SQLITE3_BOTH;
- result_obj = Z_SQLITE3_RESULT_P(object);
- ZEND_PARSE_PARAMETERS_START(0, 1)
- Z_PARAM_OPTIONAL
- Z_PARAM_LONG(mode)
- ZEND_PARSE_PARAMETERS_END();
- SQLITE3_CHECK_INITIALIZED(result_obj->db_obj, result_obj->stmt_obj->initialised, SQLite3Result)
- ret = sqlite3_step(result_obj->stmt_obj->stmt);
- switch (ret) {
- case SQLITE_ROW:
- /* If there was no return value then just skip fetching */
- if (!USED_RET()) {
- RETURN_FALSE;
- }
- array_init(return_value);
-
- int column_count = sqlite3_data_count(result_obj->stmt_obj->stmt);
- for (i = 0; i < column_count; i++) {
- zval data;
- sqlite_value_to_zval(result_obj->stmt_obj->stmt, i, &data);
- if (mode & PHP_SQLITE3_NUM) {
- add_index_zval(return_value, i, &data);
- }
- if (mode & PHP_SQLITE3_ASSOC) {
- if (mode & PHP_SQLITE3_NUM) {
- if (Z_REFCOUNTED(data)) {
- Z_ADDREF(data);
- }
- }
- add_assoc_zval(return_value, (char*)sqlite3_column_name(result_obj->stmt_obj->stmt, i), &data);
- }
- }
- break;
- case SQLITE_DONE:
- RETURN_FALSE;
- break;
- default:
- php_sqlite3_error(result_obj->db_obj, "Unable to execute statement: %s", sqlite3_errmsg(sqlite3_db_handle(result_obj->stmt_obj->stmt)));
- }
- }
- /* }}} */
- /* {{{ Resets the result set back to the first row. */
- PHP_METHOD(SQLite3Result, reset)
- {
- php_sqlite3_result *result_obj;
- zval *object = ZEND_THIS;
- result_obj = Z_SQLITE3_RESULT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(result_obj->db_obj, result_obj->stmt_obj->initialised, SQLite3Result)
- if (sqlite3_reset(result_obj->stmt_obj->stmt) != SQLITE_OK) {
- RETURN_FALSE;
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ Closes the result set. */
- PHP_METHOD(SQLite3Result, finalize)
- {
- php_sqlite3_result *result_obj;
- zval *object = ZEND_THIS;
- result_obj = Z_SQLITE3_RESULT_P(object);
- ZEND_PARSE_PARAMETERS_NONE();
- SQLITE3_CHECK_INITIALIZED(result_obj->db_obj, result_obj->stmt_obj->initialised, SQLite3Result)
- /* We need to finalize an internal statement */
- if (result_obj->is_prepared_statement == 0) {
- zend_llist_del_element(&(result_obj->db_obj->free_list), &result_obj->stmt_obj_zval,
- (int (*)(void *, void *)) php_sqlite3_compare_stmt_zval_free);
- } else {
- sqlite3_reset(result_obj->stmt_obj->stmt);
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ __constructor for SQLite3Result. */
- PHP_METHOD(SQLite3Result, __construct)
- {
- zend_throw_exception(zend_ce_exception, "SQLite3Result cannot be directly instantiated", 0);
- }
- /* }}} */
- /* {{{ Authorization Callback */
- static int php_sqlite3_authorizer(void *autharg, int action, const char *arg1, const char *arg2, const char *arg3, const char *arg4)
- {
- /* Check open_basedir restrictions first */
- if (PG(open_basedir) && *PG(open_basedir)) {
- if (action == SQLITE_ATTACH) {
- if (!arg1) {
- return SQLITE_DENY;
- }
- if (memcmp(arg1, ":memory:", sizeof(":memory:")) && *arg1) {
- if (strncmp(arg1, "file:", 5) == 0) {
- /* starts with "file:" */
- if (!arg1[5]) {
- return SQLITE_DENY;
- }
- if (php_check_open_basedir(arg1 + 5)) {
- return SQLITE_DENY;
- }
- }
- if (php_check_open_basedir(arg1)) {
- return SQLITE_DENY;
- }
- }
- }
- }
- php_sqlite3_db_object *db_obj = (php_sqlite3_db_object *)autharg;
- zend_fcall_info *fci = &db_obj->authorizer_fci;
- /* fallback to access allowed if authorizer callback is not defined */
- if (fci->size == 0) {
- return SQLITE_OK;
- }
- /* call userland authorizer callback, if set */
- zval retval;
- zval argv[5];
- ZVAL_LONG(&argv[0], action);
- if (NULL == arg1) {
- ZVAL_NULL(&argv[1]);
- } else {
- ZVAL_STRING(&argv[1], arg1);
- }
- if (NULL == arg2) {
- ZVAL_NULL(&argv[2]);
- } else {
- ZVAL_STRING(&argv[2], arg2);
- }
- if (NULL == arg3) {
- ZVAL_NULL(&argv[3]);
- } else {
- ZVAL_STRING(&argv[3], arg3);
- }
- if (NULL == arg4) {
- ZVAL_NULL(&argv[4]);
- } else {
- ZVAL_STRING(&argv[4], arg4);
- }
- fci->retval = &retval;
- fci->param_count = 5;
- fci->params = argv;
- int authreturn = SQLITE_DENY;
- if (zend_call_function(fci, &db_obj->authorizer_fcc) != SUCCESS || Z_ISUNDEF(retval)) {
- php_sqlite3_error(db_obj, "An error occurred while invoking the authorizer callback");
- } else {
- if (Z_TYPE(retval) != IS_LONG) {
- php_sqlite3_error(db_obj, "The authorizer callback returned an invalid type: expected int");
- } else {
- authreturn = Z_LVAL(retval);
- if (authreturn != SQLITE_OK && authreturn != SQLITE_IGNORE && authreturn != SQLITE_DENY) {
- php_sqlite3_error(db_obj, "The authorizer callback returned an invalid value");
- authreturn = SQLITE_DENY;
- }
- }
- }
- zend_fcall_info_args_clear(fci, 0);
- zval_ptr_dtor(&retval);
- return authreturn;
- }
- /* }}} */
- /* {{{ php_sqlite3_free_list_dtor */
- static void php_sqlite3_free_list_dtor(void **item)
- {
- php_sqlite3_free_list *free_item = (php_sqlite3_free_list *)*item;
- if (free_item->stmt_obj && free_item->stmt_obj->initialised) {
- sqlite3_finalize(free_item->stmt_obj->stmt);
- free_item->stmt_obj->initialised = 0;
- }
- efree(*item);
- }
- /* }}} */
- static int php_sqlite3_compare_stmt_zval_free(php_sqlite3_free_list **free_list, zval *statement ) /* {{{ */
- {
- return ((*free_list)->stmt_obj->initialised && Z_PTR_P(statement) == Z_PTR((*free_list)->stmt_obj_zval));
- }
- /* }}} */
- static int php_sqlite3_compare_stmt_free( php_sqlite3_free_list **free_list, sqlite3_stmt *statement ) /* {{{ */
- {
- return ((*free_list)->stmt_obj->initialised && statement == (*free_list)->stmt_obj->stmt);
- }
- /* }}} */
- static void php_sqlite3_object_free_storage(zend_object *object) /* {{{ */
- {
- php_sqlite3_db_object *intern = php_sqlite3_db_from_obj(object);
- php_sqlite3_func *func;
- php_sqlite3_collation *collation;
- if (!intern) {
- return;
- }
- /* Release function_name from authorizer */
- if (intern->authorizer_fci.size > 0) {
- zval_ptr_dtor(&intern->authorizer_fci.function_name);
- }
- while (intern->funcs) {
- func = intern->funcs;
- intern->funcs = func->next;
- if (intern->initialised && intern->db) {
- sqlite3_create_function(intern->db, func->func_name, func->argc, SQLITE_UTF8, func, NULL, NULL, NULL);
- }
- efree((char*)func->func_name);
- if (!Z_ISUNDEF(func->func)) {
- zval_ptr_dtor(&func->func);
- }
- if (!Z_ISUNDEF(func->step)) {
- zval_ptr_dtor(&func->step);
- }
- if (!Z_ISUNDEF(func->fini)) {
- zval_ptr_dtor(&func->fini);
- }
- efree(func);
- }
- while (intern->collations){
- collation = intern->collations;
- intern->collations = collation->next;
- if (intern->initialised && intern->db){
- sqlite3_create_collation(intern->db, collation->collation_name, SQLITE_UTF8, NULL, NULL);
- }
- efree((char*)collation->collation_name);
- if (!Z_ISUNDEF(collation->cmp_func)) {
- zval_ptr_dtor(&collation->cmp_func);
- }
- efree(collation);
- }
- if (intern->initialised && intern->db) {
- sqlite3_close(intern->db);
- intern->initialised = 0;
- }
- zend_object_std_dtor(&intern->zo);
- }
- /* }}} */
- static void php_sqlite3_stmt_object_free_storage(zend_object *object) /* {{{ */
- {
- php_sqlite3_stmt *intern = php_sqlite3_stmt_from_obj(object);
- if (!intern) {
- return;
- }
- if (intern->bound_params) {
- zend_hash_destroy(intern->bound_params);
- FREE_HASHTABLE(intern->bound_params);
- intern->bound_params = NULL;
- }
- if (intern->initialised) {
- zend_llist_del_element(&(intern->db_obj->free_list), intern->stmt,
- (int (*)(void *, void *)) php_sqlite3_compare_stmt_free);
- }
- if (!Z_ISUNDEF(intern->db_obj_zval)) {
- zval_ptr_dtor(&intern->db_obj_zval);
- }
- zend_object_std_dtor(&intern->zo);
- }
- /* }}} */
- static void php_sqlite3_result_object_free_storage(zend_object *object) /* {{{ */
- {
- php_sqlite3_result *intern = php_sqlite3_result_from_obj(object);
- if (!intern) {
- return;
- }
- if (!Z_ISNULL(intern->stmt_obj_zval)) {
- if (intern->stmt_obj && intern->stmt_obj->initialised) {
- sqlite3_reset(intern->stmt_obj->stmt);
- }
- zval_ptr_dtor(&intern->stmt_obj_zval);
- }
- zend_object_std_dtor(&intern->zo);
- }
- /* }}} */
- static zend_object *php_sqlite3_object_new(zend_class_entry *class_type) /* {{{ */
- {
- php_sqlite3_db_object *intern;
- /* Allocate memory for it */
- intern = zend_object_alloc(sizeof(php_sqlite3_db_object), class_type);
- /* Need to keep track of things to free */
- zend_llist_init(&(intern->free_list), sizeof(php_sqlite3_free_list *), (llist_dtor_func_t)php_sqlite3_free_list_dtor, 0);
- zend_object_std_init(&intern->zo, class_type);
- object_properties_init(&intern->zo, class_type);
- intern->zo.handlers = &sqlite3_object_handlers;
- return &intern->zo;
- }
- /* }}} */
- static zend_object *php_sqlite3_stmt_object_new(zend_class_entry *class_type) /* {{{ */
- {
- php_sqlite3_stmt *intern;
- /* Allocate memory for it */
- intern = zend_object_alloc(sizeof(php_sqlite3_stmt), class_type);
- zend_object_std_init(&intern->zo, class_type);
- object_properties_init(&intern->zo, class_type);
- intern->zo.handlers = &sqlite3_stmt_object_handlers;
- return &intern->zo;
- }
- /* }}} */
- static zend_object *php_sqlite3_result_object_new(zend_class_entry *class_type) /* {{{ */
- {
- php_sqlite3_result *intern;
- /* Allocate memory for it */
- intern = zend_object_alloc(sizeof(php_sqlite3_result), class_type);
- zend_object_std_init(&intern->zo, class_type);
- object_properties_init(&intern->zo, class_type);
- intern->zo.handlers = &sqlite3_result_object_handlers;
- return &intern->zo;
- }
- /* }}} */
- static void sqlite3_param_dtor(zval *data) /* {{{ */
- {
- struct php_sqlite3_bound_param *param = (struct php_sqlite3_bound_param*)Z_PTR_P(data);
- if (param->name) {
- zend_string_release_ex(param->name, 0);
- }
- if (!Z_ISNULL(param->parameter)) {
- zval_ptr_dtor(&(param->parameter));
- ZVAL_UNDEF(¶m->parameter);
- }
- efree(param);
- }
- /* }}} */
- /* {{{ PHP_MINIT_FUNCTION */
- PHP_MINIT_FUNCTION(sqlite3)
- {
- #ifdef ZTS
- /* Refuse to load if this wasn't a threasafe library loaded */
- if (!sqlite3_threadsafe()) {
- php_error_docref(NULL, E_WARNING, "A thread safe version of SQLite is required when using a thread safe version of PHP.");
- return FAILURE;
- }
- #endif
- memcpy(&sqlite3_object_handlers, &std_object_handlers, sizeof(zend_object_handlers));
- memcpy(&sqlite3_stmt_object_handlers, &std_object_handlers, sizeof(zend_object_handlers));
- memcpy(&sqlite3_result_object_handlers, &std_object_handlers, sizeof(zend_object_handlers));
- /* Register SQLite 3 Class */
- sqlite3_object_handlers.offset = XtOffsetOf(php_sqlite3_db_object, zo);
- sqlite3_object_handlers.clone_obj = NULL;
- sqlite3_object_handlers.free_obj = php_sqlite3_object_free_storage;
- php_sqlite3_sc_entry = register_class_SQLite3();
- php_sqlite3_sc_entry->create_object = php_sqlite3_object_new;
- /* Register SQLite 3 Prepared Statement Class */
- sqlite3_stmt_object_handlers.offset = XtOffsetOf(php_sqlite3_stmt, zo);
- sqlite3_stmt_object_handlers.clone_obj = NULL;
- sqlite3_stmt_object_handlers.free_obj = php_sqlite3_stmt_object_free_storage;
- php_sqlite3_stmt_entry = register_class_SQLite3Stmt();
- php_sqlite3_stmt_entry->create_object = php_sqlite3_stmt_object_new;
- /* Register SQLite 3 Result Class */
- sqlite3_result_object_handlers.offset = XtOffsetOf(php_sqlite3_result, zo);
- sqlite3_result_object_handlers.clone_obj = NULL;
- sqlite3_result_object_handlers.free_obj = php_sqlite3_result_object_free_storage;
- php_sqlite3_result_entry = register_class_SQLite3Result();
- php_sqlite3_result_entry->create_object = php_sqlite3_result_object_new;
- REGISTER_INI_ENTRIES();
- REGISTER_LONG_CONSTANT("SQLITE3_ASSOC", PHP_SQLITE3_ASSOC, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_NUM", PHP_SQLITE3_NUM, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_BOTH", PHP_SQLITE3_BOTH, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_INTEGER", SQLITE_INTEGER, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_FLOAT", SQLITE_FLOAT, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_TEXT", SQLITE3_TEXT, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_BLOB", SQLITE_BLOB, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_NULL", SQLITE_NULL, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_OPEN_READONLY", SQLITE_OPEN_READONLY, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_OPEN_READWRITE", SQLITE_OPEN_READWRITE, CONST_CS | CONST_PERSISTENT);
- REGISTER_LONG_CONSTANT("SQLITE3_OPEN_CREATE", SQLITE_OPEN_CREATE, CONST_CS | CONST_PERSISTENT);
- /* Class constants */
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "OK", sizeof("OK") - 1, SQLITE_OK);
- /* Constants for authorizer return */
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DENY", sizeof("DENY") - 1, SQLITE_DENY);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "IGNORE", sizeof("IGNORE") - 1, SQLITE_IGNORE);
- /* Constants for authorizer actions */
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_INDEX", sizeof("CREATE_INDEX") - 1, SQLITE_CREATE_INDEX);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_TABLE", sizeof("CREATE_TABLE") - 1, SQLITE_CREATE_TABLE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_TEMP_INDEX", sizeof("CREATE_TEMP_INDEX") - 1, SQLITE_CREATE_TEMP_INDEX);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_TEMP_TABLE", sizeof("CREATE_TEMP_TABLE") - 1, SQLITE_CREATE_TEMP_TABLE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_TEMP_TRIGGER", sizeof("CREATE_TEMP_TRIGGER") - 1, SQLITE_CREATE_TEMP_TRIGGER);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_TEMP_VIEW", sizeof("CREATE_TEMP_VIEW") - 1, SQLITE_CREATE_TEMP_VIEW);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_TRIGGER", sizeof("CREATE_TRIGGER") - 1, SQLITE_CREATE_TRIGGER);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_VIEW", sizeof("CREATE_VIEW") - 1, SQLITE_CREATE_VIEW);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DELETE", sizeof("DELETE") - 1, SQLITE_DELETE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_INDEX", sizeof("DROP_INDEX") - 1, SQLITE_DROP_INDEX);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_TABLE", sizeof("DROP_TABLE") - 1, SQLITE_DROP_TABLE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_TEMP_INDEX", sizeof("DROP_TEMP_INDEX") - 1, SQLITE_DROP_TEMP_INDEX);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_TEMP_TABLE", sizeof("DROP_TEMP_TABLE") - 1, SQLITE_DROP_TEMP_TABLE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_TEMP_TRIGGER", sizeof("DROP_TEMP_TRIGGER") - 1, SQLITE_DROP_TEMP_TRIGGER);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_TEMP_VIEW", sizeof("DROP_TEMP_VIEW") - 1, SQLITE_DROP_TEMP_VIEW);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_TRIGGER", sizeof("DROP_TRIGGER") - 1, SQLITE_DROP_TRIGGER);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_VIEW", sizeof("DROP_VIEW") - 1, SQLITE_DROP_VIEW);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "INSERT", sizeof("INSERT") - 1, SQLITE_INSERT);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "PRAGMA", sizeof("PRAGMA") - 1, SQLITE_PRAGMA);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "READ", sizeof("READ") - 1, SQLITE_READ);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "SELECT", sizeof("SELECT") - 1, SQLITE_SELECT);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "TRANSACTION", sizeof("TRANSACTION") - 1, SQLITE_TRANSACTION);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "UPDATE", sizeof("UPDATE") - 1, SQLITE_UPDATE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "ATTACH", sizeof("ATTACH") - 1, SQLITE_ATTACH);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DETACH", sizeof("DETACH") - 1, SQLITE_DETACH);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "ALTER_TABLE", sizeof("ALTER_TABLE") - 1, SQLITE_ALTER_TABLE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "REINDEX", sizeof("REINDEX") - 1, SQLITE_REINDEX);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "ANALYZE", sizeof("ANALYZE") - 1, SQLITE_ANALYZE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "CREATE_VTABLE", sizeof("CREATE_VTABLE") - 1, SQLITE_CREATE_VTABLE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "DROP_VTABLE", sizeof("DROP_VTABLE") - 1, SQLITE_DROP_VTABLE);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "FUNCTION", sizeof("FUNCTION") - 1, SQLITE_FUNCTION);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "SAVEPOINT", sizeof("SAVEPOINT") - 1, SQLITE_SAVEPOINT);
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "COPY", sizeof("COPY") - 1, SQLITE_COPY);
- #ifdef SQLITE_RECURSIVE
- zend_declare_class_constant_long(php_sqlite3_sc_entry, "RECURSIVE", sizeof("RECURSIVE") - 1, SQLITE_RECURSIVE);
- #endif
- #ifdef SQLITE_DETERMINISTIC
- REGISTER_LONG_CONSTANT("SQLITE3_DETERMINISTIC", SQLITE_DETERMINISTIC, CONST_CS | CONST_PERSISTENT);
- #endif
- return SUCCESS;
- }
- /* }}} */
- /* {{{ PHP_MSHUTDOWN_FUNCTION */
- PHP_MSHUTDOWN_FUNCTION(sqlite3)
- {
- UNREGISTER_INI_ENTRIES();
- return SUCCESS;
- }
- /* }}} */
- /* {{{ PHP_MINFO_FUNCTION */
- PHP_MINFO_FUNCTION(sqlite3)
- {
- php_info_print_table_start();
- php_info_print_table_header(2, "SQLite3 support", "enabled");
- php_info_print_table_row(2, "SQLite Library", sqlite3_libversion());
- php_info_print_table_end();
- DISPLAY_INI_ENTRIES();
- }
- /* }}} */
- /* {{{ PHP_GINIT_FUNCTION */
- static PHP_GINIT_FUNCTION(sqlite3)
- {
- #if defined(COMPILE_DL_SQLITE3) && defined(ZTS)
- ZEND_TSRMLS_CACHE_UPDATE();
- #endif
- memset(sqlite3_globals, 0, sizeof(*sqlite3_globals));
- }
- /* }}} */
- /* {{{ sqlite3_module_entry */
- zend_module_entry sqlite3_module_entry = {
- STANDARD_MODULE_HEADER,
- "sqlite3",
- NULL,
- PHP_MINIT(sqlite3),
- PHP_MSHUTDOWN(sqlite3),
- NULL,
- NULL,
- PHP_MINFO(sqlite3),
- PHP_SQLITE3_VERSION,
- PHP_MODULE_GLOBALS(sqlite3),
- PHP_GINIT(sqlite3),
- NULL,
- NULL,
- STANDARD_MODULE_PROPERTIES_EX
- };
- /* }}} */
- #ifdef COMPILE_DL_SQLITE3
- #ifdef ZTS
- ZEND_TSRMLS_CACHE_DEFINE()
- #endif
- ZEND_GET_MODULE(sqlite3)
- #endif
|