123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218 |
- /*
- * Copyright (C)2005-2019 Haxe Foundation
- *
- * Permission is hereby granted, free of charge, to any person obtaining a
- * copy of this software and associated documentation files (the "Software"),
- * to deal in the Software without restriction, including without limitation
- * the rights to use, copy, modify, merge, publish, distribute, sublicense,
- * and/or sell copies of the Software, and to permit persons to whom the
- * Software is furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
- * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
- * DEALINGS IN THE SOFTWARE.
- */
- package sys.db;
- import php.*;
- import php.Global.*;
- import php.db.*;
- import sys.db.*;
- @:coreApi class Sqlite {
- public static function open(file:String):Connection {
- return new SQLiteConnection(file);
- }
- }
- private class SQLiteConnection implements Connection {
- var db:SQLite3;
- public function new(file:String) {
- db = new SQLite3(file);
- db.enableExceptions(true);
- }
- public function request(s:String):ResultSet {
- var result = db.query(s);
- return new SQLiteResultSet(result);
- }
- public function close():Void {
- db.close();
- }
- public function escape(s:String):String {
- return SQLite3.escapeString(s);
- }
- public function quote(s:String):String {
- if (s.indexOf("\000") >= 0)
- return "x'" + Global.bin2hex(s) + "'";
- return "'" + SQLite3.escapeString(s) + "'";
- }
- public function addValue(s:StringBuf, v:Dynamic):Void {
- if (Global.is_int(v) || Global.is_null(v)) {
- s.add(v);
- } else if (Global.is_bool(v)) {
- s.add(v ? 1 : 0);
- } else {
- s.add(quote(Std.string(v)));
- }
- }
- public function lastInsertId():Int {
- return Syntax.int(db.lastInsertRowID());
- }
- public function dbName():String {
- return 'SQLite';
- }
- public function startTransaction():Void {
- db.query('BEGIN TRANSACTION');
- }
- public function commit():Void {
- db.query('COMMIT');
- }
- public function rollback():Void {
- db.query('ROLLBACK');
- }
- }
- private class SQLiteResultSet implements ResultSet {
- public var length(get, null):Int;
- public var nfields(get, null):Int;
- var cache = new NativeIndexedArray<{}>();
- var result:SQLite3Result;
- var resultIsDepleted = false;
- var fieldsInfo:NativeAssocArray<Int>;
- public function new(result:SQLite3Result) {
- this.result = result;
- }
- public function hasNext():Bool {
- return switch next() {
- case null: false;
- case row:
- array_unshift(cache, row);
- row;
- }
- }
- public function next():Dynamic {
- return switch array_shift(cache) {
- case null: fetchNext();
- case row: row;
- }
- }
- function fetchNext():Null<{}> {
- return resultIsDepleted ? null : switch result.fetchArray(Const.SQLITE3_ASSOC) {
- case false:
- resultIsDepleted = true;
- result.finalize();
- null;
- case row:
- Boot.createAnon(correctArrayTypes(row));
- }
- }
- public function cacheAll():NativeIndexedArray<{}> {
- var row = fetchNext();
- while(row != null) {
- cache.push(row);
- row = fetchNext();
- }
- return cache;
- }
- public function results():List<Dynamic> {
- var list = new List();
- for(row in cacheAll()) {
- list.add(row);
- }
- return list;
- }
- function getColumn(n:Int):Any {
- return array_values(Syntax.array(current()))[n];
- }
- public function getResult(n:Int):String {
- return Syntax.string(getColumn(n));
- }
- public function getIntResult(n:Int):Int {
- return Syntax.int(getColumn(n));
- }
- public function getFloatResult(n:Int):Float {
- return Syntax.float(getColumn(n));
- }
- public function getFieldsNames():Null<Array<String>> {
- var fieldsInfo = getFieldsInfo();
- return Global.array_keys(fieldsInfo);
- }
- function current():Null<{}> {
- return switch reset(cache) {
- case false:
- switch next() {
- case null: null;
- case row:
- cache.push(row);
- row;
- }
- case row: row;
- }
- }
- function correctArrayTypes(row:NativeAssocArray<String>):NativeAssocArray<Scalar> {
- var fieldsInfo = getFieldsInfo();
- Syntax.foreach(row, function(field:String, value:String) {
- row[field] = correctType(value, fieldsInfo[field]);
- });
- return cast row;
- }
- inline function getFieldsInfo():NativeAssocArray<Int> {
- if (fieldsInfo == null) {
- fieldsInfo = cast Syntax.arrayDecl();
- for (i in 0...nfields) {
- fieldsInfo[result.columnName(i)] = result.columnType(i);
- }
- }
- return fieldsInfo;
- }
- function correctType(value:String, type:Int):Scalar {
- if (value == null)
- return null;
- if (type == Const.SQLITE3_INTEGER)
- return Syntax.int(value);
- if (type == Const.SQLITE3_FLOAT)
- return Syntax.float(value);
- return value;
- }
- function get_length():Int
- return count(cacheAll());
- function get_nfields():Int
- return result.numColumns();
- }
|