/* * Copyright (C)2005-2012 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; private class SqliteConnection implements Connection { var c : Dynamic; var e : Dynamic; public function new( file : String ) { c = untyped __call__("sqlite_open", file, 0666, e); } public function close() { untyped __call__("sqlite_close", c); untyped __call__("unset", c); } public function request( s : String ) : ResultSet { var h = untyped __call__("sqlite_query", c, s, __php__("SQLITE_BOTH"), e); if(untyped __physeq__(h, false)) throw "Error while executing "+s+" ("+e+")"; return new SqliteResultSet(cast h); } public function escape( s : String ) { return untyped __call__("sqlite_escape_string", s); } public function quote( s : String ) { if( s.indexOf("\000") >= 0 ) return "x'"+base16_encode(s)+"'"; return "'" + untyped __call__("sqlite_escape_string", s) + "'"; } public function addValue( s : StringBuf, v : Dynamic ) { if( untyped __call__("is_int", v) || __call__("is_null", v)) s.add(v); else if( untyped __call__("is_bool", v) ) s.add(if( v ) 1 else 0); else s.add(quote(Std.string(v))); } public function lastInsertId() { return untyped __call__("sqlite_last_insert_rowid", c); } public function dbName() { return "SQLite"; } public function startTransaction() { request("BEGIN TRANSACTION"); } public function commit() { request("COMMIT"); startTransaction(); // match mysql usage } public function rollback() { request("ROLLBACK"); startTransaction(); // match mysql usage } function base16_encode(str : String) { str = untyped __call__("unpack", "H"+(2 * str.length), str); str = untyped __call__("chunk_split", untyped str[1]); return str; } } private class SqliteResultSet implements ResultSet { public var length(get,null) : Int; public var nfields(get,null) : Int; var r : Dynamic; var cache : Dynamic; public function new( r ) { this.r = r; } private function get_length() { if(untyped __physeq__(r, true)) return untyped __call__("sqlite_changes", r); else if (untyped __physeq__(r, false)) return 0; return untyped __call__("sqlite_num_rows", r); } private var _nfields : Int; private function get_nfields() { if(_nfields == null) _nfields = untyped __call__("sqlite_num_fields", r); return _nfields; } public function hasNext() { if( cache == null ) cache = next(); return (cache != null); } private var cRow : ArrayAccess; private function fetchRow() : Bool { cRow = untyped __call__("sqlite_fetch_array", r, __php__("SQLITE_ASSOC")); return ! untyped __physeq__(cRow, false); } public function next() : Dynamic { if( cache != null ) { var t = cache; cache = null; return t; } if(!fetchRow()) return null; return untyped __call__("_hx_anonymous", cRow); } public function results() : List { var l = new List(); while( true ) { var c = next(); if( c == null ) break; l.add(c); } return l; } public function getResult( n : Int ) : String { if(cRow == null && !fetchRow()) return null; return untyped __call__("array_values", cRow)[n]; } public function getIntResult( n : Int ) : Int { return untyped __call__("intval", getResult(n)); } public function getFloatResult( n : Int ) : Float { return untyped __call__("floatval", getResult(n)); } public function getFieldsNames() : Array { return throw "Not implemented"; } } @:coreApi class Sqlite { public static function open( file : String ) : Connection { return new SqliteConnection(file); } }