Home Reference Source

src/SQLite3/SQLite3Type.js

  1. /**
  2. * The script is part of toolbox-wsh.
  3. *
  4. * AUTHOR:
  5. * natade (http://twitter.com/natadea)
  6. *
  7. * LICENSE:
  8. * The MIT license https://opensource.org/licenses/MIT
  9. */
  10.  
  11. /// <reference path="../../include/SenkoWSH.d.ts" />
  12.  
  13. /**
  14. * データベースのレコードの列情報
  15. * @typedef {Object} SQLite3TypeData
  16. * @property {number} cid 列番号
  17. * @property {string} name 列名
  18. * @property {string} type 型名
  19. * @property {number} size 型のサイズ
  20. * @property {string|null} dflt_value 未設定は`null`, 設定されている場合は文字列
  21. * @property {boolean} is_not_null `null` を許してよいか
  22. */
  23.  
  24. /**
  25. * データベース内のテーブルの列情報
  26. * @requires SenkoWSH
  27. */
  28. export default class SQLite3Type {
  29.  
  30. /**
  31. * データベース内のテーブルの列情報を初期設定する
  32. * - `create` を使用して作成すること
  33. *
  34. * @param {SQLite3TypeData} info_data
  35. * @private
  36. */
  37. constructor(info_data) {
  38.  
  39. /**
  40. * 列情報
  41. */
  42. this.info = info_data;
  43.  
  44. // SQLite には扱える型名が複数あるため正規化した型名を作成
  45. {
  46. let type = "undefined";
  47. // https://www.sqlite.org/datatype3.html
  48. // TEXT, VARCHAR
  49. if(/char|clob|text|decimal/i.test(this.info.type)) {
  50. type = "string";
  51. }
  52. // NUMERIC
  53. else if(/numeric/i.test(this.info.type)) {
  54. type = "numeric";
  55. }
  56. // INTEGER
  57. else if(/int/i.test(this.info.type)) {
  58. type = "int";
  59. }
  60. // REAL
  61. else if(/double|float|real/i.test(this.info.type)) {
  62. type = "real";
  63. }
  64. // BLOB
  65. else if(/blob/i.test(this.info.type)) {
  66. type = "blob";
  67. }
  68. // BOOLEAN(NUMERIC)
  69. else if(/boolean/i.test(this.info.type)) {
  70. type = "boolean";
  71. }
  72. // DATETIME(NUMERIC)
  73. else if(/date|datetime/i.test(this.info.type)) {
  74. type = "datetime";
  75. }
  76. // NONE(NUMERIC)
  77. else if(/none/i.test(this.info.type)) {
  78. type = "none";
  79. }
  80.  
  81. /**
  82. * 正規化した型名
  83. */
  84. this.normalized_type = type;
  85. }
  86. }
  87.  
  88. /**
  89. * `SQLite3Type` を作成する
  90. * `-json` で `pragma table_info(x);` で取得した1レコードデータを引数に取る
  91. *
  92. * @param {import("./SQLite3Schema").SQLite3TableInfo} table_info_record
  93. */
  94. static create(table_info_record) {
  95. const cid = table_info_record.cid;
  96. const name = table_info_record.name;
  97. // type_data が未入力の場合はNONE として扱う
  98. const type_data = table_info_record.type.match(/[^(]+/);
  99. const type = type_data ? type_data[0] : "NONE";
  100. // 未定義なら null 存在する場合は '0' のように ' で囲まれている。
  101. let default_value = table_info_record.dflt_value;
  102. if(default_value !== null) {
  103. // 除去する
  104. if(/^'.*'$/.test(default_value)) {
  105. default_value = default_value.match(/^'(.*)'$/)[1];
  106. }
  107. }
  108. const dflt_value = default_value;
  109. const is_not_null = table_info_record.notnull !== 0;
  110. const size_data = table_info_record.type.match(/\(([0-9]+)\)/);
  111. const size = size_data ? Number.parseInt(size_data[1], 10) : -1;
  112. return new SQLite3Type({
  113. cid : cid,
  114. name : name,
  115. type : type,
  116. size : size,
  117. dflt_value : dflt_value,
  118. is_not_null : is_not_null
  119. });
  120. }
  121.  
  122. /**
  123. * 型情報を取得する
  124. *
  125. * @returns {SQLite3TypeData}
  126. */
  127. getType() {
  128. return {
  129. cid : this.info.cid,
  130. name : this.info.name,
  131. type : this.info.type,
  132. size : this.info.size,
  133. dflt_value : this.info.dflt_value,
  134. is_not_null : this.info.is_not_null
  135. };
  136. }
  137.  
  138. /**
  139. * JavaScript用のデータをSQL文で使用できる文字列へ変換
  140. * - `SQL` の型情報を元に `SQL` 内への記載用データへ変換
  141. * - 文字列データはシングルクォーテーションを付けた文字列を返す
  142. * - 数値データなどはシングルクォーテーション無しの文字列型を返す
  143. *
  144. * @param {any} x
  145. * @returns {string}
  146. */
  147. toSQLDataFromJSData(x) {
  148. const td = this.info;
  149. const js_type = System.typeOf(x);
  150. if(this.normalized_type === "string") {
  151. /**
  152. * @type {string}
  153. */
  154. const str = x.toString();
  155. if(td.size === -1) {
  156. return "'" + str + "'";
  157. }
  158. else {
  159. return "'" + str.slice(0, td.size) + "'";
  160. }
  161. }
  162. else if((this.normalized_type === "numeric") || (this.normalized_type === "none")) {
  163. if(js_type !== "number") {
  164. return "'" + x + "'";
  165. }
  166. return x.toString();
  167. }
  168. else if(this.normalized_type === "int") {
  169. let num = 0;
  170. if(js_type !== "number") {
  171. num = Number.parseFloat(x);
  172. }
  173. else {
  174. num = x;
  175. }
  176. if(!isFinite(num)) {
  177. num = 0;
  178. }
  179. return Math.trunc(num).toString();
  180. }
  181. else if(this.normalized_type === "real") {
  182. if(js_type !== "number") {
  183. return Number.parseFloat(x).toString();
  184. }
  185. return x.toString();
  186. }
  187. else if(this.normalized_type === "blob") {
  188. return "null";
  189. }
  190. else if(this.normalized_type === "boolean") {
  191. return !!x ? "1" : "0";
  192. }
  193. else if(this.normalized_type === "datetime") {
  194. const date = new Date(x);
  195. const date_text = Format.textf("%04d-%02d-%02d %02d:%02d:%02d",
  196. date.getUTCFullYear(),
  197. date.getUTCMonth() + 1,
  198. date.getUTCDate(),
  199. date.getUTCHours(),
  200. date.getUTCMinutes(),
  201. date.getUTCSeconds()
  202. );
  203. return "'" + date_text + "'";
  204. }
  205. console.log("Error : toSQLDataFromJSData " + x);
  206. return "null";
  207. }
  208.  
  209. /**
  210. * SQLで取得したデータをJavaScript用のデータへ変換
  211. * - 「`-json` で取得し `eval` で変換したデータ」から `SQL` の型情報を元に `JavaScript` の型へ変換
  212. *
  213. * @param {any} x
  214. * @returns {any}
  215. */
  216. toJSDataFromSQLData(x) {
  217. const js_type = System.typeOf(x);
  218. if(js_type === "null") {
  219. return null;
  220. }
  221. else if(this.normalized_type === "string") {
  222. if(js_type === "string") {
  223. return x;
  224. }
  225. else if(js_type === "object") {
  226. return null;
  227. }
  228. else {
  229. return x.toString();
  230. }
  231. }
  232. else if((this.normalized_type === "numeric") || (this.normalized_type === "none")) {
  233. if(js_type === "object") {
  234. return null;
  235. }
  236. const number = Number.parseFloat(x);
  237. if(Number.isFinite(number)) {
  238. return number;
  239. }
  240. if(x.toString() === number.toString()) {
  241. return number;
  242. }
  243. return x.toString();
  244. }
  245. else if(this.normalized_type === "int") {
  246. if(js_type === "object") {
  247. return null;
  248. }
  249. return Number.parseInt(x, 10);
  250. }
  251. else if(this.normalized_type === "real") {
  252. if(js_type === "object") {
  253. return null;
  254. }
  255. return Number.parseFloat(x);
  256. }
  257. else if(this.normalized_type === "blob") {
  258. return {};
  259. }
  260. else if(this.normalized_type === "boolean") {
  261. const number = Number.parseFloat(x);
  262. return number !== 0;
  263. }
  264. else if(this.normalized_type === "datetime") {
  265. /**
  266. * @type {string}
  267. */
  268. const date = x.toString();
  269. return new Date(date.replace(/\-/g, "/"));
  270. }
  271. return null;
  272. }
  273.  
  274. }