import { uuid } from 'lib/uuid.js'; import { promiseChain } from 'lib/promise-utils.js'; import { time } from 'lib/time-utils.js' import { Database } from 'lib/database.js' const structureSql = ` CREATE TABLE folders ( id TEXT PRIMARY KEY, title TEXT NOT NULL DEFAULT "", created_time INT NOT NULL, updated_time INT NOT NULL ); CREATE INDEX folders_title ON folders (title); CREATE INDEX folders_updated_time ON folders (updated_time); CREATE TABLE notes ( id TEXT PRIMARY KEY, parent_id TEXT NOT NULL DEFAULT "", title TEXT NOT NULL DEFAULT "", body TEXT NOT NULL DEFAULT "", created_time INT NOT NULL, updated_time INT NOT NULL, is_conflict INT NOT NULL DEFAULT 0, latitude NUMERIC NOT NULL DEFAULT 0, longitude NUMERIC NOT NULL DEFAULT 0, altitude NUMERIC NOT NULL DEFAULT 0, author TEXT NOT NULL DEFAULT "", source_url TEXT NOT NULL DEFAULT "", is_todo INT NOT NULL DEFAULT 0, todo_due INT NOT NULL DEFAULT 0, todo_completed INT NOT NULL DEFAULT 0, source TEXT NOT NULL DEFAULT "", source_application TEXT NOT NULL DEFAULT "", application_data TEXT NOT NULL DEFAULT "", \`order\` INT NOT NULL DEFAULT 0 ); CREATE INDEX notes_title ON notes (title); CREATE INDEX notes_updated_time ON notes (updated_time); CREATE INDEX notes_is_conflict ON notes (is_conflict); CREATE INDEX notes_is_todo ON notes (is_todo); CREATE INDEX notes_order ON notes (\`order\`); CREATE TABLE tags ( id TEXT PRIMARY KEY, title TEXT NOT NULL DEFAULT "", created_time INT NOT NULL, updated_time INT NOT NULL ); CREATE INDEX tags_title ON tags (title); CREATE INDEX tags_updated_time ON tags (updated_time); CREATE TABLE note_tags ( id TEXT PRIMARY KEY, note_id TEXT NOT NULL, tag_id TEXT NOT NULL, created_time INT NOT NULL, updated_time INT NOT NULL ); CREATE INDEX note_tags_note_id ON note_tags (note_id); CREATE INDEX note_tags_tag_id ON note_tags (tag_id); CREATE INDEX note_tags_updated_time ON note_tags (updated_time); CREATE TABLE resources ( id TEXT PRIMARY KEY, title TEXT NOT NULL DEFAULT "", mime TEXT NOT NULL, filename TEXT NOT NULL DEFAULT "", created_time INT NOT NULL, updated_time INT NOT NULL ); CREATE INDEX resources_title ON resources (title); CREATE INDEX resources_updated_time ON resources (updated_time); CREATE TABLE settings ( \`key\` TEXT PRIMARY KEY, \`value\` TEXT, \`type\` INT NOT NULL ); CREATE TABLE table_fields ( id INTEGER PRIMARY KEY, table_name TEXT NOT NULL, field_name TEXT NOT NULL, field_type INT NOT NULL, field_default TEXT ); CREATE TABLE sync_items ( id INTEGER PRIMARY KEY, sync_target INT NOT NULL, sync_time INT NOT NULL DEFAULT 0, item_type INT NOT NULL, item_id TEXT NOT NULL ); CREATE INDEX sync_items_sync_time ON sync_items (sync_time); CREATE INDEX sync_items_sync_target ON sync_items (sync_target); CREATE INDEX sync_items_item_type ON sync_items (item_type); CREATE INDEX sync_items_item_id ON sync_items (item_id); CREATE TABLE deleted_items ( id INTEGER PRIMARY KEY, item_type INT NOT NULL, item_id TEXT NOT NULL, deleted_time INT NOT NULL ); CREATE TABLE version ( version INT NOT NULL ); INSERT INTO version (version) VALUES (1); `; class JoplinDatabase extends Database { constructor(driver) { super(driver); this.initialized_ = false; this.tableFields_ = null; } initialized() { return this.initialized_; } async open(options) { await super.open(options); return this.initialize(); } tableFieldNames(tableName) { let tf = this.tableFields(tableName); let output = []; for (let i = 0; i < tf.length; i++) { output.push(tf[i].name); } return output; } tableFields(tableName) { if (!this.tableFields_) throw new Error('Fields have not been loaded yet'); if (!this.tableFields_[tableName]) throw new Error('Unknown table: ' + tableName); return this.tableFields_[tableName]; } refreshTableFields() { this.logger().info('Initializing tables...'); let queries = []; queries.push(this.wrapQuery('DELETE FROM table_fields')); return this.selectAll('SELECT name FROM sqlite_master WHERE type="table"').then((tableRows) => { let chain = []; for (let i = 0; i < tableRows.length; i++) { let tableName = tableRows[i].name; if (tableName == 'android_metadata') continue; if (tableName == 'table_fields') continue; chain.push(() => { return this.selectAll('PRAGMA table_info("' + tableName + '")').then((pragmas) => { for (let i = 0; i < pragmas.length; i++) { let item = pragmas[i]; // In SQLite, if the default value is a string it has double quotes around it, so remove them here let defaultValue = item.dflt_value; if (typeof defaultValue == 'string' && defaultValue.length >= 2 && defaultValue[0] == '"' && defaultValue[defaultValue.length - 1] == '"') { defaultValue = defaultValue.substr(1, defaultValue.length - 2); } let q = Database.insertQuery('table_fields', { table_name: tableName, field_name: item.name, field_type: Database.enumId('fieldType', item.type), field_default: defaultValue, }); queries.push(q); } }); }); } return promiseChain(chain); }).then(() => { return this.transactionExecBatch(queries); }); } async upgradeDatabase(fromVersion) { // INSTRUCTIONS TO UPGRADE THE DATABASE: // // 1. Add the new version number to the existingDatabaseVersions array // 2. Add the upgrade logic to the "switch (targetVersion)" statement below const existingDatabaseVersions = [0, 1, 2, 3]; let currentVersionIndex = existingDatabaseVersions.indexOf(fromVersion); if (currentVersionIndex == existingDatabaseVersions.length - 1) return false; while (currentVersionIndex < existingDatabaseVersions.length - 1) { const targetVersion = existingDatabaseVersions[currentVersionIndex + 1]; this.logger().info("Converting database to version " + targetVersion); let queries = []; if (targetVersion == 1) { queries = this.wrapQueries(this.sqlStringToLines(structureSql)); } if (targetVersion == 2) { const newTableSql = ` CREATE TABLE deleted_items ( id INTEGER PRIMARY KEY, item_type INT NOT NULL, item_id TEXT NOT NULL, deleted_time INT NOT NULL, sync_target INT NOT NULL ); `; queries.push({ sql: 'DROP TABLE deleted_items' }); queries.push({ sql: this.sqlStringToLines(newTableSql)[0] }); queries.push({ sql: "CREATE INDEX deleted_items_sync_target ON deleted_items (sync_target)" }); } if (targetVersion == 3) { queries = this.alterColumnQueries('settings', { key: 'TEXT PRIMARY KEY', value: 'TEXT' }); } queries.push({ sql: 'UPDATE version SET version = ?', params: [targetVersion] }); await this.transactionExecBatch(queries); currentVersionIndex++; } return true; } async initialize() { this.logger().info('Checking for database schema update...'); let versionRow = null; try { // Will throw if the database has not been created yet, but this is handled below versionRow = await this.selectOne('SELECT * FROM version LIMIT 1'); } catch (error) { console.info(error); } const version = !versionRow ? 0 : versionRow.version; this.logger().info('Current database version', version); const upgraded = await this.upgradeDatabase(version); if (upgraded) await this.refreshTableFields(); this.tableFields_ = {}; let rows = await this.selectAll('SELECT * FROM table_fields'); for (let i = 0; i < rows.length; i++) { let row = rows[i]; if (!this.tableFields_[row.table_name]) this.tableFields_[row.table_name] = []; this.tableFields_[row.table_name].push({ name: row.field_name, type: row.field_type, default: Database.formatValue(row.field_type, row.field_default), }); } } } Database.TYPE_INT = 1; Database.TYPE_TEXT = 2; Database.TYPE_NUMERIC = 3; export { JoplinDatabase };