Home Reference Source

src/SQLite3/SQLite3.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. import SQLite3Schema from "./SQLite3Schema.js";
  14. import SQLite3IF from "./SQLite3IF.js";
  15.  
  16. /**
  17. * SQL のタイムアウト設定
  18. */
  19. const SQL_TIME_OUT = ".timeout 1000\n";
  20.  
  21. /**
  22. * WSH で SQLite3 を使用するためのライブラリ
  23. * @requires SenkoWSH
  24. */
  25. export default class SQLite3 {
  26. /**
  27. * 利用するデータベースをセット
  28. * @param {SFile} db_file DBファイル
  29. * @returns {Object<string, SQLite3IF>} テーブル操作用データ
  30. */
  31. static use(db_file) {
  32. /**
  33. * @type {Object<string, SQLite3IF>}
  34. */
  35. const output = {};
  36.  
  37. // テーブル名のリストを作成
  38. const table_name = SQLite3.execSQL(db_file, ".tables", "-readonly");
  39. if(!table_name) {
  40. console.log("Error : tables " + db_file);
  41. return null;
  42. }
  43. const table_name_list = table_name.trim().split(/\s+/);
  44.  
  45. // テーブル内の列データを作成する
  46. /**
  47. * 列名
  48. * @type {string[]}
  49. */
  50. const table_info_sql = [];
  51. for(let i = 0; i < table_name_list.length; i++) {
  52. const key = table_name_list[i];
  53. table_info_sql.push("pragma table_info(" + key + ");");
  54. }
  55.  
  56. // テーブル内の列データを全て取得する
  57. const table_info_data = SQLite3.execSQL(db_file, table_info_sql.join(""), "-readonly -json");
  58. if(!table_info_data) {
  59. console.log("Error : table_info " + db_file);
  60. return null;
  61. }
  62. // []で括られた1テーブルごとのJSON情報から、1テーブルずつ抜き出して、データを格納する
  63.  
  64. /**
  65. * 列情報
  66. * @type {SQLite3Schema[]}
  67. */
  68. const type_obj_list = [];
  69.  
  70. /**
  71. * @param {string} table_info_text
  72. * @returns {string}
  73. * @private
  74. */
  75. const create_table_info = function(table_info_text) {
  76. type_obj_list.push(SQLite3Schema.create(table_info_text));
  77. return "";
  78. }
  79. table_info_data.replace(/(\[[^\]]+\])/g, create_table_info);
  80.  
  81. // IFを作成していく
  82. for(let i = 0; i < table_name_list.length; i++) {
  83. const key = table_name_list[i];
  84. const data = new SQLite3IF(db_file, key, type_obj_list[i]);
  85. output[key] = data;
  86. }
  87. return output;
  88. }
  89.  
  90. /**
  91. * データベースをバキュームする
  92. * @param {SFile} db_file DBファイル
  93. * @returns {boolean}
  94. */
  95. static vaccum(db_file) {
  96. return SQLite3.execSQL(db_file, "vacuum;") !== null;
  97. }
  98.  
  99. /**
  100. * SQL文を実行する
  101. * @param {SFile} db_file DBファイル
  102. * @param {string} sql SQL文
  103. * @param {string} [option] 実行時のオプション
  104. * @returns {string | null} SQL実行結果
  105. */
  106. static execSQL(db_file, sql, option) {
  107. if(!SQLite3.sqlite3) {
  108. console.log("Error : execSQL " + db_file + "," + sql);
  109. return null;
  110. }
  111. const option_ = option !== undefined ? option : "";
  112. const sql_file = SFile.createTempFile();
  113. const sqt_text = SQL_TIME_OUT + sql;
  114. sql_file.writeString(sqt_text, "utf-8", "\r\n", false);
  115. const script = "\"" + SQLite3.sqlite3.getAbsolutePath() + "\" \"" + db_file.getAbsolutePath() + "\" " + option_ + " < \"" + sql_file.getAbsolutePath() + "\"";
  116. const output = System.execBatchScript(script, "utf-8");
  117. sql_file.remove();
  118. if(output.exit_code) {
  119. console.log(output.error);
  120. return null;
  121. }
  122. return output.out;
  123. }
  124.  
  125. /**
  126. * ツールをインストールする
  127. * - `./lib/sqlite3.exe` を利用する
  128. * - ファイルが存在しない場合は自動でダウンロードする
  129. * - 本ライブラリを使用する場合は、`setSQLite3` でツールを設定するか、`install` でダウンロードする必要がある
  130. *
  131. * @returns {boolean}
  132. */
  133. static install() {
  134. if(SQLite3.sqlite3) {
  135. return true;
  136. }
  137. const exe_path = "./lib/sqlite3.exe";
  138. const exe_file = new SFile(exe_path);
  139. if(exe_file.isFile()) {
  140. // ファイルがある場合はそれを設定する
  141. return SQLite3.setSQLite3(exe_file);
  142. }
  143. else {
  144. // ファイルがない場合は、ダウンロードしてくれる
  145. new SFile("./lib").mkdirs();
  146. const download_url = "https://www.sqlite.org/download.html";
  147. const download_text = new SFile(download_url).readString();
  148. if(download_text) {
  149. const match_data = download_text.match(/'[^']+\/sqlite-tools-win32[^']+'/);
  150. if(match_data) {
  151. const match_text = match_data[0].toString();
  152. const zip_url = "https://www.sqlite.org/" + match_text.substr(1, match_text.length - 2);
  153. const zip_binary = new SFile(zip_url).readBinary();
  154. if(zip_binary.length !== 0) {
  155. const temp_file_1 = SFile.createTempFile();
  156. temp_file_1.mkdirs();
  157. const zip_file = new SFile(temp_file_1 + "\\" + new SFile(zip_url).getName());
  158. zip_file.writeBinary(zip_binary);
  159. const temp_file_2 = SFile.createTempFile();
  160. SFile.extract(zip_file, temp_file_2);
  161. /**
  162. * @type {SFile}
  163. */
  164. const sqlite3_file = SFile.findFile(temp_file_2, /\\sqlite3.exe$/i);
  165. if(sqlite3_file) {
  166. sqlite3_file.move("./lib");
  167. }
  168. temp_file_1.remove();
  169. temp_file_2.remove();
  170. return SQLite3.setSQLite3(sqlite3_file);
  171. }
  172. }
  173. }
  174. }
  175. return false;
  176. }
  177.  
  178. /**
  179. * ツールを設定する
  180. * - `*.db` を操作するための `sqlite-tools-win32` に含まれる `sqlite3.exe` を設定する
  181. * @param {SFile|string} sqlite_tool_path
  182. * @returns {boolean}
  183. */
  184. static setSQLite3(sqlite_tool_path) {
  185. const exe_file = new SFile(sqlite_tool_path);
  186. if(!exe_file.isFile() || !/sqlite3.exe$/i.test(exe_file.getAbsolutePath()) ) {
  187. return false;
  188. }
  189. SQLite3.sqlite3 = exe_file;
  190. return true;
  191. }
  192. }
  193.  
  194. /**
  195. * `*.db` を操作するための `sqlite-tools-win32` に含まれる `sqlite3.exe`
  196. * @type {SFile}
  197. */
  198. SQLite3.sqlite3 = null;