Jdbc.hx 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. package java.db;
  2. import java.util.concurrent.atomic.AtomicInteger;
  3. import haxe.io.Bytes;
  4. import java.sql.Types;
  5. @:native('haxe.java.db.Jdbc')
  6. class Jdbc
  7. {
  8. public static function create(cnx:java.sql.Connection):sys.db.Connection
  9. {
  10. return new JdbcConnection(cnx);
  11. }
  12. }
  13. @:native('haxe.java.db.JdbcConnection')
  14. private class JdbcConnection implements sys.db.Connection
  15. {
  16. private static var ids = new AtomicInteger(0);
  17. private var id:Int;
  18. private var cnx:java.sql.Connection;
  19. private var _lastInsertId:Int;
  20. //escape handling
  21. private var escapeRegex:EReg;
  22. private var escapes:Array<Dynamic>;
  23. public function new(cnx)
  24. {
  25. this.id = ids.getAndIncrement();
  26. this.cnx = cnx;
  27. this.escapes = [];
  28. this.escapeRegex = ~/@@HX_ESCAPE(\d+)_(\d+)@@/;
  29. }
  30. public function close()
  31. {
  32. try
  33. this.cnx.close()
  34. catch(e:Dynamic) throw e;
  35. }
  36. public function escape(s:String):String
  37. {
  38. return "@@HX_ESCAPE" + id + "_" +escapes.push(s) + "@@";
  39. }
  40. public function quote(s:String):String
  41. {
  42. return "@@HX_ESCAPE" + id + "_" +escapes.push(s) + "@@";
  43. }
  44. public function addValue(s:StringBuf, v:Dynamic)
  45. {
  46. if (Std.is(v, Date))
  47. {
  48. if (dbName() == "SQLite")
  49. {
  50. v = Std.string(v);
  51. } else {
  52. var d:Date = v;
  53. v = new java.sql.Date(cast(d.getTime(), haxe.Int64));
  54. }
  55. } else if (Std.is(v, Bytes)) {
  56. var bt:Bytes = v;
  57. v = bt.getData();
  58. }
  59. s.add("@@HX_ESCAPE");
  60. s.add(id);
  61. s.add("_");
  62. s.add(escapes.push(v));
  63. s.add("@@");
  64. }
  65. public function lastInsertId():Int
  66. {
  67. return _lastInsertId;
  68. }
  69. public function dbName():String
  70. {
  71. try {
  72. var ret = cnx.getMetaData().getDriverName();
  73. var retc = ret.toLowerCase();
  74. if (retc.indexOf("mysql") != -1)
  75. return "MySQL";
  76. else if (retc.indexOf("sqlite") != -1)
  77. return "SQLite";
  78. return ret;
  79. } catch(e:Dynamic) { throw e; }
  80. }
  81. public function startTransaction()
  82. {
  83. try
  84. {
  85. cnx.setAutoCommit(false);
  86. }
  87. catch(e:Dynamic) throw e;
  88. }
  89. public function commit()
  90. {
  91. try
  92. {
  93. cnx.commit();
  94. }
  95. catch(e:Dynamic)
  96. {
  97. throw e;
  98. }
  99. }
  100. public function rollback()
  101. {
  102. try
  103. cnx.rollback()
  104. catch(e:Dynamic) throw e;
  105. }
  106. public function request(s:String):sys.db.ResultSet
  107. {
  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. {
  115. while (r.match(s))
  116. {
  117. var id = r.matched(1);
  118. if (id != myid) throw "Request quotes are only valid for one single request; They can't be cached.";
  119. newst.add(r.matchedLeft());
  120. var eid = Std.parseInt(r.matched(2));
  121. if (eid == null || eid > elen)
  122. throw "Invalid request quote ID " + eid;
  123. sentArray.push(escapes[eid - 1]);
  124. newst.add("?");
  125. s = r.matchedRight();
  126. }
  127. newst.add(s);
  128. var stmt = cnx.prepareStatement(newst.toString());
  129. for (i in 0...sentArray.length)
  130. {
  131. stmt.setObject(i + 1, sentArray[i]);
  132. }
  133. var ret = null, dbName = dbName();
  134. if (stmt.execute())
  135. {
  136. //is a result set
  137. var rs = stmt.getResultSet();
  138. ret = new JdbcResultSet(rs, dbName, stmt.getMetaData());
  139. } else {
  140. //is an update
  141. var affected = stmt.getUpdateCount();
  142. if (affected == 1)
  143. {
  144. var autogen = stmt.getGeneratedKeys();
  145. if (autogen.next())
  146. {
  147. this._lastInsertId = autogen.getInt(1);
  148. }
  149. }
  150. ret = new JdbcResultSet(null, dbName,null);
  151. }
  152. if (escapes.length != 0)
  153. escapes = [];
  154. this.id = ids.getAndIncrement();
  155. return ret;
  156. }
  157. catch(e:Dynamic)
  158. {
  159. if (escapes.length != 0)
  160. escapes = [];
  161. this.id = ids.getAndIncrement();
  162. throw e;
  163. }
  164. }
  165. }
  166. @:native('haxe.java.db.JdbcResultSet')
  167. private class JdbcResultSet implements sys.db.ResultSet
  168. {
  169. @:isVar public var length(get,null) : Int;
  170. public var nfields(get,null) : Int;
  171. private var rs:java.sql.ResultSet;
  172. private var names:Array<String>;
  173. private var types:java.NativeArray<Int>;
  174. private var dbName:String;
  175. public function new(rs, dbName, meta:java.sql.ResultSetMetaData)
  176. {
  177. this.dbName = dbName;
  178. this.rs = rs;
  179. if (meta != null)
  180. {
  181. try {
  182. var count = meta.getColumnCount();
  183. var names = [], types = new NativeArray(count);
  184. for (i in 0...count)
  185. {
  186. names.push(meta.getColumnName(i+1));
  187. types[i] = meta.getColumnType(i+1);
  188. }
  189. this.types = types;
  190. this.names = names;
  191. } catch(e:Dynamic) throw e;
  192. }
  193. }
  194. private function get_length():Int
  195. {
  196. if (length == 0)
  197. {
  198. try
  199. {
  200. var cur = rs.getRow();
  201. rs.last();
  202. this.length = rs.getRow();
  203. rs.absolute(cur);
  204. } catch(e:Dynamic) throw e;
  205. }
  206. return length;
  207. }
  208. private function get_nfields():Int
  209. {
  210. return names == null ? 0 : names.length;
  211. }
  212. public function hasNext() : Bool
  213. {
  214. try
  215. {
  216. return rs != null && rs.next();
  217. }
  218. catch(e:Dynamic) { return throw e; }
  219. }
  220. public function next() : Dynamic
  221. {
  222. try {
  223. if (rs == null) return null;
  224. var ret = {}, names = names, types = types;
  225. for (i in 0...names.length)
  226. {
  227. var name = names[i], t = types[i], val:Dynamic = null;
  228. if (t == Types.FLOAT)
  229. {
  230. val = rs.getDouble(i+1);
  231. } else if (t == Types.DATE || t == Types.TIME) {
  232. if (dbName == "SQLite")
  233. {
  234. var str = rs.getString(i+1);
  235. if (str != null)
  236. {
  237. var d:Date = Date.fromString(str);
  238. val = d;
  239. }
  240. } else {
  241. var d:java.sql.Date = rs.getDate(i+1);
  242. if (d != null)
  243. val = Date.fromTime(cast d.getTime());
  244. }
  245. } else if (t == Types.LONGVARBINARY || t == Types.VARBINARY || t == Types.BINARY || t == Types.BLOB) {
  246. var b = rs.getBytes(i+1);
  247. if (b != null)
  248. val = Bytes.ofData(b);
  249. } else {
  250. untyped __java__("val = rs.getObject(i + 1)"); //type parameter constraint + overloads
  251. }
  252. Reflect.setField(ret, name, val);
  253. }
  254. return ret;
  255. } catch(e:Dynamic) throw e;
  256. }
  257. public function results() : List<Dynamic>
  258. {
  259. var l = new List();
  260. if (rs == null) return l;
  261. try
  262. {
  263. while(rs.next())
  264. l.add(next());
  265. } catch(e:Dynamic) throw e;
  266. return l;
  267. }
  268. public function getResult( n : Int ) : String
  269. {
  270. try
  271. {
  272. return rs.getString(n);
  273. } catch(e:Dynamic) throw e;
  274. return null;
  275. }
  276. public function getIntResult( n : Int ) : Int
  277. {
  278. try
  279. {
  280. return rs.getInt(n);
  281. }
  282. catch(e:Dynamic) { return throw e; };
  283. }
  284. public function getFloatResult( n : Int ) : Float
  285. {
  286. try
  287. {
  288. return rs.getFloat(n);
  289. } catch(e:Dynamic) { return throw e; };
  290. }
  291. public function getFieldsNames() : Null<Array<String>>
  292. {
  293. return this.names;
  294. }
  295. }