Jdbc.hx 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. /*
  2. * Copyright (C)2005-2019 Haxe Foundation
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a
  5. * copy of this software and associated documentation files (the "Software"),
  6. * to deal in the Software without restriction, including without limitation
  7. * the rights to use, copy, modify, merge, publish, distribute, sublicense,
  8. * and/or sell copies of the Software, and to permit persons to whom the
  9. * Software is furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
  20. * DEALINGS IN THE SOFTWARE.
  21. */
  22. package java.db;
  23. import java.util.concurrent.atomic.AtomicInteger;
  24. import haxe.io.Bytes;
  25. import java.sql.Types;
  26. @:native('haxe.java.db.Jdbc')
  27. class Jdbc {
  28. public static function create(cnx:java.sql.Connection):sys.db.Connection {
  29. return new JdbcConnection(cnx);
  30. }
  31. }
  32. @:native('haxe.java.db.JdbcConnection')
  33. private class JdbcConnection implements sys.db.Connection {
  34. private static var ids = new AtomicInteger(0);
  35. private var id:Int;
  36. private var cnx:java.sql.Connection;
  37. private var _lastInsertId:Int;
  38. // escape handling
  39. private var escapeRegex:EReg;
  40. private var escapes:Array<Dynamic>;
  41. public function new(cnx) {
  42. this.id = ids.getAndIncrement();
  43. this.cnx = cnx;
  44. this.escapes = [];
  45. this.escapeRegex = ~/@@HX_ESCAPE(\d+)_(\d+)@@/;
  46. }
  47. public function close() {
  48. try
  49. this.cnx.close()
  50. catch (e:Dynamic)
  51. throw e;
  52. }
  53. public function escape(s:String):String {
  54. return "@@HX_ESCAPE" + id + "_" + escapes.push(s) + "@@";
  55. }
  56. public function quote(s:String):String {
  57. return "@@HX_ESCAPE" + id + "_" + escapes.push(s) + "@@";
  58. }
  59. public function addValue(s:StringBuf, v:Dynamic) {
  60. if (Std.is(v, Date)) {
  61. v = Std.string(v);
  62. } else if (Std.is(v, Bytes)) {
  63. var bt:Bytes = v;
  64. v = bt.getData();
  65. }
  66. s.add("@@HX_ESCAPE");
  67. s.add(id);
  68. s.add("_");
  69. s.add(escapes.push(v));
  70. s.add("@@");
  71. }
  72. public function lastInsertId():Int {
  73. return _lastInsertId;
  74. }
  75. public function dbName():String {
  76. try {
  77. var ret = cnx.getMetaData().getDriverName();
  78. var retc = ret.toLowerCase();
  79. if (retc.indexOf("mysql") != -1)
  80. return "MySQL";
  81. else if (retc.indexOf("sqlite") != -1)
  82. return "SQLite";
  83. return ret;
  84. } catch (e:Dynamic) {
  85. throw e;
  86. }
  87. }
  88. public function startTransaction() {
  89. try {
  90. cnx.setAutoCommit(false);
  91. } catch (e:Dynamic)
  92. throw e;
  93. }
  94. public function commit() {
  95. try {
  96. cnx.commit();
  97. } catch (e:Dynamic) {
  98. throw e;
  99. }
  100. }
  101. public function rollback() {
  102. try
  103. cnx.rollback()
  104. catch (e:Dynamic)
  105. throw e;
  106. }
  107. public function request(s:String):sys.db.ResultSet {
  108. var newst = new StringBuf();
  109. var sentArray = [];
  110. // cycle through the request string, adding any @@HX_ESCAPE@@ reference to the sentArray
  111. var r = escapeRegex;
  112. var myid = id + "", escapes = escapes, elen = escapes.length;
  113. try {
  114. while (r.match(s)) {
  115. var id = r.matched(1);
  116. if (id != myid)
  117. throw "Request quotes are only valid for one single request; They can't be cached.";
  118. newst.add(r.matchedLeft());
  119. var eid = Std.parseInt(r.matched(2));
  120. if (eid == null || eid > elen)
  121. throw "Invalid request quote ID " + eid;
  122. sentArray.push(escapes[eid - 1]);
  123. newst.add("?");
  124. s = r.matchedRight();
  125. }
  126. newst.add(s);
  127. var stmt = cnx.prepareStatement(newst.toString(), java.sql.Statement.Statement_Statics.RETURN_GENERATED_KEYS);
  128. for (i in 0...sentArray.length) {
  129. stmt.setObject(i + 1, sentArray[i]);
  130. }
  131. var ret = null, dbName = dbName();
  132. if (stmt.execute()) {
  133. // is a result set
  134. var rs = stmt.getResultSet();
  135. ret = new JdbcResultSet(rs, dbName, stmt.getMetaData());
  136. } else {
  137. // is an update
  138. var affected = stmt.getUpdateCount();
  139. if (affected == 1) {
  140. var autogen = stmt.getGeneratedKeys();
  141. if (autogen.next()) {
  142. this._lastInsertId = autogen.getInt(1);
  143. }
  144. }
  145. ret = new JdbcResultSet(null, dbName, null);
  146. }
  147. if (escapes.length != 0)
  148. escapes = [];
  149. this.id = ids.getAndIncrement();
  150. return ret;
  151. } catch (e:Dynamic) {
  152. if (escapes.length != 0)
  153. escapes = [];
  154. this.id = ids.getAndIncrement();
  155. throw e;
  156. }
  157. }
  158. }
  159. @:native('haxe.java.db.JdbcResultSet')
  160. private class JdbcResultSet implements sys.db.ResultSet {
  161. @:isVar public var length(get, null):Int;
  162. public var nfields(get, null):Int;
  163. private var rs:java.sql.ResultSet;
  164. private var names:Array<String>;
  165. private var types:java.NativeArray<Int>;
  166. private var dbName:String;
  167. private var didNext:Bool;
  168. public function new(rs, dbName, meta:java.sql.ResultSetMetaData) {
  169. this.dbName = dbName;
  170. this.rs = rs;
  171. if (meta != null) {
  172. try {
  173. var count = meta.getColumnCount();
  174. var names = [], types = new NativeArray(count);
  175. for (i in 0...count) {
  176. names.push(meta.getColumnName(i + 1));
  177. types[i] = meta.getColumnType(i + 1);
  178. }
  179. this.types = types;
  180. this.names = names;
  181. } catch (e:Dynamic)
  182. throw e;
  183. }
  184. }
  185. private function get_length():Int {
  186. if (length == 0) {
  187. try {
  188. var cur = rs.getRow();
  189. rs.last();
  190. this.length = rs.getRow();
  191. rs.absolute(cur);
  192. } catch (e:Dynamic)
  193. throw e;
  194. }
  195. return length;
  196. }
  197. private function get_nfields():Int {
  198. return names == null ? 0 : names.length;
  199. }
  200. public function hasNext():Bool {
  201. try {
  202. didNext = true;
  203. return rs != null && rs.next();
  204. } catch (e:Dynamic) {
  205. return throw e;
  206. }
  207. }
  208. public function next():Dynamic {
  209. try {
  210. if (rs == null)
  211. return null;
  212. if (didNext) {
  213. didNext = false;
  214. } else {
  215. if (!rs.next()) {
  216. return null;
  217. }
  218. }
  219. var ret = {}, names = names, types = types;
  220. for (i in 0...names.length) {
  221. var name = names[i], t = types[i], val:Dynamic = null;
  222. if (t == Types.FLOAT) {
  223. val = rs.getDouble(i + 1);
  224. } else if (t == Types.DATE || t == Types.TIME) {
  225. if (dbName == "SQLite") {
  226. var str = rs.getString(i + 1);
  227. if (str != null) {
  228. var d:Date = Date.fromString(str);
  229. val = d;
  230. }
  231. } else {
  232. var d:java.sql.Date = rs.getDate(i + 1);
  233. if (d != null)
  234. val = Date.fromTime(cast d.getTime());
  235. }
  236. } else if (t == Types.LONGVARBINARY || t == Types.VARBINARY || t == Types.BINARY || t == Types.BLOB) {
  237. var b = rs.getBytes(i + 1);
  238. if (b != null)
  239. val = Bytes.ofData(b);
  240. } else {
  241. untyped __java__("val = rs.getObject(i + 1)"); // type parameter constraint + overloads
  242. }
  243. Reflect.setField(ret, name, val);
  244. }
  245. return ret;
  246. } catch (e:Dynamic)
  247. throw e;
  248. }
  249. public function results():List<Dynamic> {
  250. var l = new List();
  251. if (rs == null)
  252. return l;
  253. try {
  254. while (hasNext())
  255. l.add(next());
  256. } catch (e:Dynamic)
  257. throw e;
  258. return l;
  259. }
  260. public function getResult(n:Int):String {
  261. try {
  262. return rs.getString(n);
  263. } catch (e:Dynamic)
  264. throw e;
  265. }
  266. public function getIntResult(n:Int):Int {
  267. try {
  268. return rs.getInt(n);
  269. } catch (e:Dynamic) {
  270. return throw e;
  271. };
  272. }
  273. public function getFloatResult(n:Int):Float {
  274. try {
  275. return rs.getFloat(n);
  276. } catch (e:Dynamic) {
  277. return throw e;
  278. };
  279. }
  280. public function getFieldsNames():Null<Array<String>> {
  281. return this.names;
  282. }
  283. }