diff --git a/packages/server/schema.sqlite b/packages/server/schema.sqlite index 79fb1e7e2aa..c4350389278 100644 Binary files a/packages/server/schema.sqlite and b/packages/server/schema.sqlite differ diff --git a/packages/server/src/app.ts b/packages/server/src/app.ts index 5e1bac201c1..d3d25f30d11 100644 --- a/packages/server/src/app.ts +++ b/packages/server/src/app.ts @@ -5,7 +5,7 @@ import * as Koa from 'koa'; import * as fs from 'fs-extra'; import Logger, { LoggerWrapper, TargetType } from '@joplin/lib/Logger'; import config, { initConfig, runningInDocker } from './config'; -import { migrateLatest, waitForConnection, sqliteDefaultDir, latestMigration, DbConnection } from './db'; +import { migrateLatest, waitForConnection, sqliteDefaultDir, latestMigration } from './db'; import { AppContext, Env, KoaNext } from './utils/types'; import FsDriverNode from '@joplin/lib/fs-driver-node'; import routeHandler from './middleware/routeHandler'; @@ -17,11 +17,10 @@ import startServices from './utils/startServices'; import { credentialFile } from './utils/testing/testUtils'; import apiVersionHandler from './middleware/apiVersionHandler'; import clickJackingHandler from './middleware/clickJackingHandler'; -import newModelFactory, { Options } from './models/factory'; +import newModelFactory from './models/factory'; import setupCommands from './utils/setupCommands'; import { RouteResponseFormat, routeResponseFormat } from './utils/routeUtils'; import { parseEnv } from './env'; -import storageDriverFromConfig from './models/items/storage/storageDriverFromConfig'; interface Argv { env?: Env; @@ -222,13 +221,6 @@ async function main() { fs.writeFileSync(pidFile, `${process.pid}`); } - const newModelFactoryOptions = async (db: DbConnection): Promise => { - return { - storageDriver: await storageDriverFromConfig(config().storageDriver, db, { assignDriverId: env !== 'buildTypes' }), - storageDriverFallback: await storageDriverFromConfig(config().storageDriverFallback, db, { assignDriverId: env !== 'buildTypes' }), - }; - }; - let runCommandAndExitApp = true; if (selectedCommand) { @@ -245,7 +237,7 @@ async function main() { }); } else { const connectionCheck = await waitForConnection(config().database); - const models = newModelFactory(connectionCheck.connection, config(), await newModelFactoryOptions(connectionCheck.connection)); + const models = newModelFactory(connectionCheck.connection, config()); await selectedCommand.run(commandArgv, { db: connectionCheck.connection, @@ -275,7 +267,7 @@ async function main() { appLogger().info('Connection check:', connectionCheckLogInfo); const ctx = app.context as AppContext; - await setupAppContext(ctx, env, connectionCheck.connection, appLogger, await newModelFactoryOptions(connectionCheck.connection)); + await setupAppContext(ctx, env, connectionCheck.connection, appLogger); await initializeJoplinUtils(config(), ctx.joplinBase.models, ctx.joplinBase.services.mustache); diff --git a/packages/server/src/migrations/20211105183559_storage.ts b/packages/server/src/migrations/20211105183559_storage.ts index 94f6a05fa1d..c2367f44a36 100644 --- a/packages/server/src/migrations/20211105183559_storage.ts +++ b/packages/server/src/migrations/20211105183559_storage.ts @@ -5,10 +5,16 @@ export async function up(db: DbConnection): Promise { await db.schema.createTable('storages', (table: Knex.CreateTableBuilder) => { table.increments('id').unique().primary().notNullable(); table.text('connection_string').notNullable(); + table.bigInteger('updated_time').notNullable(); + table.bigInteger('created_time').notNullable(); }); + const now = Date.now(); + await db('storages').insert({ connection_string: 'Type=Database', + updated_time: now, + created_time: now, }); // First we create the column and set a default so as to populate the @@ -21,6 +27,10 @@ export async function up(db: DbConnection): Promise { await db.schema.alterTable('items', (table: Knex.CreateTableBuilder) => { table.integer('content_storage_id').notNullable().alter(); }); + + await db.schema.alterTable('storages', (table: Knex.CreateTableBuilder) => { + table.unique(['connection_string']); + }); } export async function down(db: DbConnection): Promise { diff --git a/packages/server/src/models/ItemModel.ts b/packages/server/src/models/ItemModel.ts index d532532cb35..87680b83dba 100644 --- a/packages/server/src/models/ItemModel.ts +++ b/packages/server/src/models/ItemModel.ts @@ -9,8 +9,9 @@ import { ChangePreviousItem } from './ChangeModel'; import { unique } from '../utils/array'; import StorageDriverBase, { Context } from './items/storage/StorageDriverBase'; import { DbConnection } from '../db'; -import { Config, StorageDriverMode } from '../utils/types'; -import { NewModelFactoryHandler, Options } from './factory'; +import { Config, StorageDriverConfig, StorageDriverMode } from '../utils/types'; +import { NewModelFactoryHandler } from './factory'; +import storageDriverFromConfig from './items/storage/storageDriverFromConfig'; const mimeUtils = require('@joplin/lib/mime-utils.js').mime; @@ -49,14 +50,16 @@ export interface ItemLoadOptions extends LoadOptions { export default class ItemModel extends BaseModel { private updatingTotalSizes_: boolean = false; - private storageDriver_: StorageDriverBase = null; - private storageDriverFallback_: StorageDriverBase = null; + private storageDriverConfig_: StorageDriverConfig; + private storageDriverConfigFallback_: StorageDriverConfig; - public constructor(db: DbConnection, modelFactory: NewModelFactoryHandler, config: Config, options: Options) { + private static storageDrivers_: Map = new Map(); + + public constructor(db: DbConnection, modelFactory: NewModelFactoryHandler, config: Config) { super(db, modelFactory, config); - this.storageDriver_ = options.storageDriver; - this.storageDriverFallback_ = options.storageDriverFallback; + this.storageDriverConfig_ = config.storageDriver; + this.storageDriverConfigFallback_ = config.storageDriverFallback; } protected get tableName(): string { @@ -75,6 +78,26 @@ export default class ItemModel extends BaseModel { return Object.keys(databaseSchema[this.tableName]).filter(f => f !== 'content'); } + private async storageDriverFromConfig(config: StorageDriverConfig): Promise { + let driver = ItemModel.storageDrivers_.get(config); + + if (!driver) { + driver = await storageDriverFromConfig(config, this.db); + ItemModel.storageDrivers_.set(config, driver); + } + + return driver; + } + + public async storageDriver(): Promise { + return this.storageDriverFromConfig(this.storageDriverConfig_); + } + + public async storageDriverFallback(): Promise { + if (!this.storageDriverConfigFallback_) return null; + return this.storageDriverFromConfig(this.storageDriverConfigFallback_); + } + public async checkIfAllowed(user: User, action: AclAction, resource: Item = null): Promise { if (action === AclAction.Create) { if (!(await this.models().shareUser().isShareParticipant(resource.jop_share_id, user.id))) throw new ErrorForbidden('user has no access to this share'); @@ -136,25 +159,31 @@ export default class ItemModel extends BaseModel { } private async storageDriverWrite(itemId: Uuid, content: Buffer, context: Context) { - await this.storageDriver_.write(itemId, content, context); + const storageDriver = await this.storageDriver(); + const storageDriverFallback = await this.storageDriverFallback(); + + await storageDriver.write(itemId, content, context); - if (this.storageDriverFallback_) { - if (this.storageDriverFallback_.mode === StorageDriverMode.ReadWrite) { - await this.storageDriverFallback_.write(itemId, content, context); - } else if (this.storageDriverFallback_.mode === StorageDriverMode.ReadOnly) { - await this.storageDriverFallback_.write(itemId, Buffer.from(''), context); + if (storageDriverFallback) { + if (storageDriverFallback.mode === StorageDriverMode.ReadWrite) { + await storageDriverFallback.write(itemId, content, context); + } else if (storageDriverFallback.mode === StorageDriverMode.ReadOnly) { + await storageDriverFallback.write(itemId, Buffer.from(''), context); } else { - throw new Error(`Unsupported fallback mode: ${this.storageDriverFallback_.mode}`); + throw new Error(`Unsupported fallback mode: ${storageDriverFallback.mode}`); } } } private async storageDriverRead(itemId: Uuid, context: Context) { - if (await this.storageDriver_.exists(itemId, context)) { - return this.storageDriver_.read(itemId, context); + const storageDriver = await this.storageDriver(); + const storageDriverFallback = await this.storageDriverFallback(); + + if (await storageDriver.exists(itemId, context)) { + return storageDriver.read(itemId, context); } else { - if (!this.storageDriverFallback_) throw new Error(`Content does not exist but fallback content driver is not defined: ${itemId}`); - return this.storageDriverFallback_.read(itemId, context); + if (!storageDriverFallback) throw new Error(`Content does not exist but fallback content driver is not defined: ${itemId}`); + return storageDriverFallback.read(itemId, context); } } @@ -417,7 +446,8 @@ export default class ItemModel extends BaseModel { try { const content = itemToSave.content; delete itemToSave.content; - itemToSave.content_storage_id = this.storageDriver_.storageId; + + itemToSave.content_storage_id = (await this.storageDriver()).storageId; itemToSave.content_size = content ? content.byteLength : 0; @@ -624,14 +654,17 @@ export default class ItemModel extends BaseModel { const ids = typeof id === 'string' ? [id] : id; if (!ids.length) return; + const storageDriver = await this.storageDriver(); + const storageDriverFallback = await this.storageDriverFallback(); + const shares = await this.models().share().byItemIds(ids); await this.withTransaction(async () => { await this.models().share().delete(shares.map(s => s.id)); await this.models().userItem().deleteByItemIds(ids); await this.models().itemResource().deleteByItemIds(ids); - await this.storageDriver_.delete(ids, { models: this.models() }); - if (this.storageDriverFallback_) await this.storageDriverFallback_.delete(ids, { models: this.models() }); + await storageDriver.delete(ids, { models: this.models() }); + if (storageDriverFallback) await storageDriverFallback.delete(ids, { models: this.models() }); await super.delete(ids, options); }, 'ItemModel::delete'); @@ -679,7 +712,7 @@ export default class ItemModel extends BaseModel { let previousItem: ChangePreviousItem = null; if (item.content && !item.content_storage_id) { - item.content_storage_id = this.storageDriver_.storageId; + item.content_storage_id = (await this.storageDriver()).storageId; } if (isNew) { diff --git a/packages/server/src/models/factory.ts b/packages/server/src/models/factory.ts index 581394400ce..4e410d81075 100644 --- a/packages/server/src/models/factory.ts +++ b/packages/server/src/models/factory.ts @@ -72,39 +72,29 @@ import SubscriptionModel from './SubscriptionModel'; import UserFlagModel from './UserFlagModel'; import EventModel from './EventModel'; import { Config } from '../utils/types'; -import StorageDriverBase from './items/storage/StorageDriverBase'; import LockModel from './LockModel'; import StorageModel from './StorageModel'; -export interface Options { - storageDriver: StorageDriverBase; - storageDriverFallback?: StorageDriverBase; -} - export type NewModelFactoryHandler = (db: DbConnection)=> Models; export class Models { private db_: DbConnection; private config_: Config; - private options_: Options; - public constructor(db: DbConnection, config: Config, options: Options) { + public constructor(db: DbConnection, config: Config) { this.db_ = db; this.config_ = config; - this.options_ = options; - - // if (!options.storageDriver) throw new Error('StorageDriver is required'); this.newModelFactory = this.newModelFactory.bind(this); } private newModelFactory(db: DbConnection) { - return new Models(db, this.config_, this.options_); + return new Models(db, this.config_); } public item() { - return new ItemModel(this.db_, this.newModelFactory, this.config_, this.options_); + return new ItemModel(this.db_, this.newModelFactory, this.config_); } public user() { @@ -177,6 +167,6 @@ export class Models { } -export default function newModelFactory(db: DbConnection, config: Config, options: Options): Models { - return new Models(db, config, options); +export default function newModelFactory(db: DbConnection, config: Config): Models { + return new Models(db, config); } diff --git a/packages/server/src/models/items/storage/StorageDriverDatabase.test.ts b/packages/server/src/models/items/storage/StorageDriverDatabase.test.ts index 026d9fb7810..f4aeaab642d 100644 --- a/packages/server/src/models/items/storage/StorageDriverDatabase.test.ts +++ b/packages/server/src/models/items/storage/StorageDriverDatabase.test.ts @@ -1,8 +1,7 @@ import { clientType } from '../../../db'; import { afterAllTests, beforeAllDb, beforeEachDb, db, expectNotThrow, expectThrow, models } from '../../../utils/testing/testUtils'; -import { StorageDriverMode } from '../../../utils/types'; +import { StorageDriverConfig, StorageDriverMode, StorageDriverType } from '../../../utils/types'; import StorageDriverDatabase from './StorageDriverDatabase'; -import StorageDriverMemory from './StorageDriverMemory'; import { shouldDeleteContent, shouldNotCreateItemIfContentNotSaved, shouldNotUpdateItemIfContentNotSaved, shouldSupportFallbackDriver, shouldSupportFallbackDriverInReadWriteMode, shouldUpdateContentStorageIdAfterSwitchingDriver, shouldWriteToContentAndReadItBack } from './testUtils'; const newDriver = () => { @@ -11,6 +10,12 @@ const newDriver = () => { }); }; +const newConfig = (): StorageDriverConfig => { + return { + type: StorageDriverType.Database, + }; +}; + describe('StorageDriverDatabase', function() { beforeAll(async () => { @@ -26,23 +31,19 @@ describe('StorageDriverDatabase', function() { }); test('should write to content and read it back', async function() { - const driver = newDriver(); - await shouldWriteToContentAndReadItBack(driver); + await shouldWriteToContentAndReadItBack(newConfig()); }); test('should delete the content', async function() { - const driver = newDriver(); - await shouldDeleteContent(driver); + await shouldDeleteContent(newConfig()); }); test('should not create the item if the content cannot be saved', async function() { - const driver = newDriver(); - await shouldNotCreateItemIfContentNotSaved(driver); + await shouldNotCreateItemIfContentNotSaved(newConfig()); }); test('should not update the item if the content cannot be saved', async function() { - const driver = newDriver(); - await shouldNotUpdateItemIfContentNotSaved(driver); + await shouldNotUpdateItemIfContentNotSaved(newConfig()); }); test('should fail if the item row does not exist', async function() { @@ -56,15 +57,15 @@ describe('StorageDriverDatabase', function() { }); test('should support fallback content drivers', async function() { - await shouldSupportFallbackDriver(newDriver(), new StorageDriverMemory(2)); + await shouldSupportFallbackDriver(newConfig(), { type: StorageDriverType.Memory }); }); test('should support fallback content drivers in rw mode', async function() { - await shouldSupportFallbackDriverInReadWriteMode(newDriver(), new StorageDriverMemory(2, { mode: StorageDriverMode.ReadWrite })); + await shouldSupportFallbackDriverInReadWriteMode(newConfig(), { type: StorageDriverType.Memory, mode: StorageDriverMode.ReadWrite }); }); test('should update content storage ID after switching driver', async function() { - await shouldUpdateContentStorageIdAfterSwitchingDriver(newDriver(), new StorageDriverMemory(2)); + await shouldUpdateContentStorageIdAfterSwitchingDriver(newConfig(), { type: StorageDriverType.Memory }); }); }); diff --git a/packages/server/src/models/items/storage/StorageDriverFs.test.ts b/packages/server/src/models/items/storage/StorageDriverFs.test.ts index 74ecac39e22..5bbab1748c0 100644 --- a/packages/server/src/models/items/storage/StorageDriverFs.test.ts +++ b/packages/server/src/models/items/storage/StorageDriverFs.test.ts @@ -1,5 +1,6 @@ import { pathExists, remove } from 'fs-extra'; import { afterAllTests, beforeAllDb, beforeEachDb, expectNotThrow, expectThrow, tempDirPath } from '../../../utils/testing/testUtils'; +import { StorageDriverConfig, StorageDriverType } from '../../../utils/types'; import StorageDriverFs from './StorageDriverFs'; import { shouldDeleteContent, shouldNotCreateItemIfContentNotSaved, shouldNotUpdateItemIfContentNotSaved, shouldWriteToContentAndReadItBack } from './testUtils'; @@ -9,6 +10,13 @@ const newDriver = () => { return new StorageDriverFs(1, { path: basePath_ }); }; +const newConfig = (): StorageDriverConfig => { + return { + type: StorageDriverType.Filesystem, + path: basePath_, + }; +}; + describe('StorageDriverFs', function() { beforeAll(async () => { @@ -30,23 +38,19 @@ describe('StorageDriverFs', function() { }); test('should write to content and read it back', async function() { - const driver = newDriver(); - await shouldWriteToContentAndReadItBack(driver); + await shouldWriteToContentAndReadItBack(newConfig()); }); test('should delete the content', async function() { - const driver = newDriver(); - await shouldDeleteContent(driver); + await shouldDeleteContent(newConfig()); }); test('should not create the item if the content cannot be saved', async function() { - const driver = newDriver(); - await shouldNotCreateItemIfContentNotSaved(driver); + await shouldNotCreateItemIfContentNotSaved(newConfig()); }); test('should not update the item if the content cannot be saved', async function() { - const driver = newDriver(); - await shouldNotUpdateItemIfContentNotSaved(driver); + await shouldNotUpdateItemIfContentNotSaved(newConfig()); }); test('should write to a file and read it back', async function() { diff --git a/packages/server/src/models/items/storage/StorageDriverMemory.test.ts b/packages/server/src/models/items/storage/StorageDriverMemory.test.ts index d382f76b933..7a8bccef65f 100644 --- a/packages/server/src/models/items/storage/StorageDriverMemory.test.ts +++ b/packages/server/src/models/items/storage/StorageDriverMemory.test.ts @@ -1,7 +1,13 @@ import { afterAllTests, beforeAllDb, beforeEachDb } from '../../../utils/testing/testUtils'; -import StorageDriverMemory from './StorageDriverMemory'; +import { StorageDriverConfig, StorageDriverType } from '../../../utils/types'; import { shouldDeleteContent, shouldNotCreateItemIfContentNotSaved, shouldNotUpdateItemIfContentNotSaved, shouldWriteToContentAndReadItBack } from './testUtils'; +const newConfig = (): StorageDriverConfig => { + return { + type: StorageDriverType.Memory, + }; +}; + describe('StorageDriverMemory', function() { beforeAll(async () => { @@ -17,23 +23,19 @@ describe('StorageDriverMemory', function() { }); test('should write to content and read it back', async function() { - const driver = new StorageDriverMemory(1); - await shouldWriteToContentAndReadItBack(driver); + await shouldWriteToContentAndReadItBack(newConfig()); }); test('should delete the content', async function() { - const driver = new StorageDriverMemory(1); - await shouldDeleteContent(driver); + await shouldDeleteContent(newConfig()); }); test('should not create the item if the content cannot be saved', async function() { - const driver = new StorageDriverMemory(1); - await shouldNotCreateItemIfContentNotSaved(driver); + await shouldNotCreateItemIfContentNotSaved(newConfig()); }); test('should not update the item if the content cannot be saved', async function() { - const driver = new StorageDriverMemory(1); - await shouldNotUpdateItemIfContentNotSaved(driver); + await shouldNotUpdateItemIfContentNotSaved(newConfig()); }); }); diff --git a/packages/server/src/models/items/storage/storageDriverFromConfig.ts b/packages/server/src/models/items/storage/storageDriverFromConfig.ts index c6db08f688b..0c168a8feb4 100644 --- a/packages/server/src/models/items/storage/storageDriverFromConfig.ts +++ b/packages/server/src/models/items/storage/storageDriverFromConfig.ts @@ -23,19 +23,19 @@ export default async function(config: StorageDriverConfig, db: DbConnection, opt let storageId: number = 0; if (options.assignDriverId) { - const models = newModelFactory(db, globalConfig(), { storageDriver: null }); + const models = newModelFactory(db, globalConfig()); const connectionString = serializeStorageConfig(config); - const existingStorage = await models.storage().byConnectionString(connectionString); + let storage = await models.storage().byConnectionString(connectionString); - if (existingStorage) { - storageId = existingStorage.id; - } else { - const storage = await models.storage().save({ + if (!storage) { + await models.storage().save({ connection_string: connectionString, }); - storageId = storage.id; + storage = await models.storage().byConnectionString(connectionString); } + + storageId = storage.id; } if (config.type === StorageDriverType.Database) { diff --git a/packages/server/src/models/items/storage/testUtils.ts b/packages/server/src/models/items/storage/testUtils.ts index 8f72f589833..55224e8664a 100644 --- a/packages/server/src/models/items/storage/testUtils.ts +++ b/packages/server/src/models/items/storage/testUtils.ts @@ -1,20 +1,30 @@ +import config from '../../../config'; import { Item } from '../../../services/database/types'; -import { createUserAndSession, makeNoteSerializedBody, models } from '../../../utils/testing/testUtils'; -import { StorageDriverMode } from '../../../utils/types'; -import StorageDriverBase, { Context } from './StorageDriverBase'; - -const testModels = (driver: StorageDriverBase) => { - return models({ storageDriver: driver }); +import { createUserAndSession, db, makeNoteSerializedBody, models } from '../../../utils/testing/testUtils'; +import { Config, StorageDriverConfig, StorageDriverMode } from '../../../utils/types'; +import newModelFactory from '../../factory'; +import { Context } from './StorageDriverBase'; + +const newTestModels = (driverConfig: StorageDriverConfig, driverConfigFallback: StorageDriverConfig = null) => { + const newConfig: Config = { + ...config(), + storageDriver: driverConfig, + storageDriverFallback: driverConfigFallback, + }; + return newModelFactory(db(), newConfig); }; -export async function shouldWriteToContentAndReadItBack(driver: StorageDriverBase) { +export async function shouldWriteToContentAndReadItBack(driverConfig: StorageDriverConfig) { const { user } = await createUserAndSession(1); const noteBody = makeNoteSerializedBody({ id: '00000000000000000000000000000001', title: 'testing driver', }); - const output = await testModels(driver).item().saveFromRawContent(user, [{ + const testModels = newTestModels(driverConfig); + const driver = await testModels.item().storageDriver(); + + const output = await testModels.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', body: Buffer.from(noteBody), }]); @@ -22,38 +32,43 @@ export async function shouldWriteToContentAndReadItBack(driver: StorageDriverBas const result = output['00000000000000000000000000000001.md']; expect(result.error).toBeFalsy(); - const item = await testModels(driver).item().loadWithContent(result.item.id); + const item = await testModels.item().loadWithContent(result.item.id); expect(item.content.byteLength).toBe(item.content_size); expect(item.content_storage_id).toBe(driver.storageId); const rawContent = await driver.read(item.id, { models: models() }); expect(rawContent.byteLength).toBe(item.content_size); - const jopItem = testModels(driver).item().itemToJoplinItem(item); + const jopItem = testModels.item().itemToJoplinItem(item); expect(jopItem.id).toBe('00000000000000000000000000000001'); expect(jopItem.title).toBe('testing driver'); } -export async function shouldDeleteContent(driver: StorageDriverBase) { +export async function shouldDeleteContent(driverConfig: StorageDriverConfig) { const { user } = await createUserAndSession(1); const noteBody = makeNoteSerializedBody({ id: '00000000000000000000000000000001', title: 'testing driver', }); - const output = await testModels(driver).item().saveFromRawContent(user, [{ + const testModels = newTestModels(driverConfig); + + const output = await testModels.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', body: Buffer.from(noteBody), }]); const item: Item = output['00000000000000000000000000000001.md'].item; - expect((await testModels(driver).item().all()).length).toBe(1); - await testModels(driver).item().delete(item.id); - expect((await testModels(driver).item().all()).length).toBe(0); + expect((await testModels.item().all()).length).toBe(1); + await testModels.item().delete(item.id); + expect((await testModels.item().all()).length).toBe(0); } -export async function shouldNotCreateItemIfContentNotSaved(driver: StorageDriverBase) { +export async function shouldNotCreateItemIfContentNotSaved(driverConfig: StorageDriverConfig) { + const testModels = newTestModels(driverConfig); + const driver = await testModels.item().storageDriver(); + const previousWrite = driver.write; driver.write = () => { throw new Error('not working!'); }; @@ -64,26 +79,29 @@ export async function shouldNotCreateItemIfContentNotSaved(driver: StorageDriver title: 'testing driver', }); - const output = await testModels(driver).item().saveFromRawContent(user, [{ + const output = await testModels.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', body: Buffer.from(noteBody), }]); expect(output['00000000000000000000000000000001.md'].error.message).toBe('not working!'); - expect((await testModels(driver).item().all()).length).toBe(0); + expect((await testModels.item().all()).length).toBe(0); } finally { driver.write = previousWrite; } } -export async function shouldNotUpdateItemIfContentNotSaved(driver: StorageDriverBase) { +export async function shouldNotUpdateItemIfContentNotSaved(driverConfig: StorageDriverConfig) { const { user } = await createUserAndSession(1); const noteBody = makeNoteSerializedBody({ id: '00000000000000000000000000000001', title: 'testing driver', }); - await testModels(driver).item().saveFromRawContent(user, [{ + const testModels = newTestModels(driverConfig); + const driver = await testModels.item().storageDriver(); + + await testModels.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', body: Buffer.from(noteBody), }]); @@ -93,12 +111,12 @@ export async function shouldNotUpdateItemIfContentNotSaved(driver: StorageDriver title: 'updated 1', }); - await testModels(driver).item().saveFromRawContent(user, [{ + await testModels.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', body: Buffer.from(noteBodyMod1), }]); - const itemMod1 = testModels(driver).item().itemToJoplinItem(await testModels(driver).item().loadByJopId(user.id, '00000000000000000000000000000001', { withContent: true })); + const itemMod1 = testModels.item().itemToJoplinItem(await testModels.item().loadByJopId(user.id, '00000000000000000000000000000001', { withContent: true })); expect(itemMod1.title).toBe('updated 1'); const noteBodyMod2 = makeNoteSerializedBody({ @@ -110,23 +128,26 @@ export async function shouldNotUpdateItemIfContentNotSaved(driver: StorageDriver driver.write = () => { throw new Error('not working!'); }; try { - const output = await testModels(driver).item().saveFromRawContent(user, [{ + const output = await testModels.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', body: Buffer.from(noteBodyMod2), }]); expect(output['00000000000000000000000000000001.md'].error.message).toBe('not working!'); - const itemMod2 = testModels(driver).item().itemToJoplinItem(await testModels(driver).item().loadByJopId(user.id, '00000000000000000000000000000001', { withContent: true })); + const itemMod2 = testModels.item().itemToJoplinItem(await testModels.item().loadByJopId(user.id, '00000000000000000000000000000001', { withContent: true })); expect(itemMod2.title).toBe('updated 1'); // Check it has not been updated } finally { driver.write = previousWrite; } } -export async function shouldSupportFallbackDriver(driver: StorageDriverBase, fallbackDriver: StorageDriverBase) { +export async function shouldSupportFallbackDriver(driverConfig: StorageDriverConfig, fallbackDriverConfig: StorageDriverConfig) { const { user } = await createUserAndSession(1); - const output = await testModels(driver).item().saveFromRawContent(user, [{ + const testModels = newTestModels(driverConfig); + const driver = await testModels.item().storageDriver(); + + const output = await testModels.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', body: Buffer.from(makeNoteSerializedBody({ id: '00000000000000000000000000000001', @@ -144,10 +165,7 @@ export async function shouldSupportFallbackDriver(driver: StorageDriverBase, fal previousByteLength = content.byteLength; } - const testModelWithFallback = models({ - storageDriver: driver, - storageDriverFallback: fallbackDriver, - }); + const testModelWithFallback = newTestModels(driverConfig, fallbackDriverConfig); // If the item content is not on the main content driver, it should get // it from the fallback one. @@ -165,6 +183,8 @@ export async function shouldSupportFallbackDriver(driver: StorageDriverBase, fal }]); { + const fallbackDriver = await testModelWithFallback.item().storageDriverFallback(); + // Check that it has cleared the fallback driver content const context: Context = { models: models() }; const fallbackContent = await fallbackDriver.read(itemId, context); @@ -176,15 +196,12 @@ export async function shouldSupportFallbackDriver(driver: StorageDriverBase, fal } } -export async function shouldSupportFallbackDriverInReadWriteMode(driver: StorageDriverBase, fallbackDriver: StorageDriverBase) { - if (fallbackDriver.mode !== StorageDriverMode.ReadWrite) throw new Error('Content driver must be configured in RW mode for this test'); +export async function shouldSupportFallbackDriverInReadWriteMode(driverConfig: StorageDriverConfig, fallbackDriverConfig: StorageDriverConfig) { + if (fallbackDriverConfig.mode !== StorageDriverMode.ReadWrite) throw new Error('Content driver must be configured in RW mode for this test'); const { user } = await createUserAndSession(1); - const testModelWithFallback = models({ - storageDriver: driver, - storageDriverFallback: fallbackDriver, - }); + const testModelWithFallback = newTestModels(driverConfig, fallbackDriverConfig); const output = await testModelWithFallback.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', @@ -197,6 +214,9 @@ export async function shouldSupportFallbackDriverInReadWriteMode(driver: Storage const itemId = output['00000000000000000000000000000001.md'].item.id; { + const driver = await testModelWithFallback.item().storageDriver(); + const fallbackDriver = await testModelWithFallback.item().storageDriverFallback(); + // Check that it has written the content to both drivers const context: Context = { models: models() }; const fallbackContent = await fallbackDriver.read(itemId, context); @@ -207,18 +227,15 @@ export async function shouldSupportFallbackDriverInReadWriteMode(driver: Storage } } -export async function shouldUpdateContentStorageIdAfterSwitchingDriver(oldDriver: StorageDriverBase, newDriver: StorageDriverBase) { - if (oldDriver.storageId === newDriver.storageId) throw new Error('Drivers must be different for this test'); +export async function shouldUpdateContentStorageIdAfterSwitchingDriver(oldDriverConfig: StorageDriverConfig, newDriverConfig: StorageDriverConfig) { + if (oldDriverConfig.type === newDriverConfig.type) throw new Error('Drivers must be different for this test'); const { user } = await createUserAndSession(1); - const oldDriverModel = models({ - storageDriver: oldDriver, - }); - - const newDriverModel = models({ - storageDriver: newDriver, - }); + const oldDriverModel = newTestModels(oldDriverConfig); + const newDriverModel = newTestModels(newDriverConfig); + const oldDriver = await oldDriverModel.item().storageDriver(); + const newDriver = await newDriverModel.item().storageDriver(); const output = await oldDriverModel.item().saveFromRawContent(user, [{ name: '00000000000000000000000000000001.md', diff --git a/packages/server/src/services/database/types.ts b/packages/server/src/services/database/types.ts index f9e25ad5feb..e7109a13159 100644 --- a/packages/server/src/services/database/types.ts +++ b/packages/server/src/services/database/types.ts @@ -249,6 +249,8 @@ export interface Event extends WithUuid { export interface Storage { id?: number; connection_string?: string; + updated_time?: string; + created_time?: string; } export interface Item extends WithDates, WithUuid { @@ -427,6 +429,8 @@ export const databaseSchema: DatabaseTables = { storages: { id: { type: 'number' }, connection_string: { type: 'string' }, + updated_time: { type: 'string' }, + created_time: { type: 'string' }, }, items: { id: { type: 'string' }, diff --git a/packages/server/src/tools/debugTools.ts b/packages/server/src/tools/debugTools.ts index bfc8f71ec5a..6737e5a691c 100644 --- a/packages/server/src/tools/debugTools.ts +++ b/packages/server/src/tools/debugTools.ts @@ -1,7 +1,6 @@ import time from '@joplin/lib/time'; import { DbConnection, dropTables, migrateLatest } from '../db'; import newModelFactory from '../models/factory'; -import storageDriverFromConfig from '../models/items/storage/storageDriverFromConfig'; import { AccountType } from '../models/UserModel'; import { User, UserFlagType } from '../services/database/types'; import { Config } from '../utils/types'; @@ -35,10 +34,7 @@ export async function createTestUsers(db: DbConnection, config: Config, options: const password = 'hunter1hunter2hunter3'; - const models = newModelFactory(db, config, { - // storageDriver: new StorageDriverDatabase(1, { dbClientType: clientType(db) }), - storageDriver: await storageDriverFromConfig(config.storageDriver, db), // new StorageDriverDatabase(1, { dbClientType: clientType(db) }), - }); + const models = newModelFactory(db, config); if (options.count) { const users: User[] = []; diff --git a/packages/server/src/utils/setupAppContext.ts b/packages/server/src/utils/setupAppContext.ts index f53c9a4a117..d3dce8e6550 100644 --- a/packages/server/src/utils/setupAppContext.ts +++ b/packages/server/src/utils/setupAppContext.ts @@ -1,7 +1,7 @@ import { LoggerWrapper } from '@joplin/lib/Logger'; import config from '../config'; import { DbConnection } from '../db'; -import newModelFactory, { Models, Options as ModelFactoryOptions } from '../models/factory'; +import newModelFactory, { Models } from '../models/factory'; import { AppContext, Config, Env } from './types'; import routes from '../routes/routes'; import ShareService from '../services/ShareService'; @@ -23,8 +23,8 @@ async function setupServices(env: Env, models: Models, config: Config): Promise< return output; } -export default async function(appContext: AppContext, env: Env, dbConnection: DbConnection, appLogger: ()=> LoggerWrapper, options: ModelFactoryOptions): Promise { - const models = newModelFactory(dbConnection, config(), options); +export default async function(appContext: AppContext, env: Env, dbConnection: DbConnection, appLogger: ()=> LoggerWrapper): Promise { + const models = newModelFactory(dbConnection, config()); // The joplinBase object is immutable because it is shared by all requests. // Then a "joplin" context property is created from it per request, which diff --git a/packages/server/src/utils/testing/testUtils.ts b/packages/server/src/utils/testing/testUtils.ts index c6f936c8770..0f227cbb215 100644 --- a/packages/server/src/utils/testing/testUtils.ts +++ b/packages/server/src/utils/testing/testUtils.ts @@ -1,7 +1,7 @@ import { DbConnection, connectDb, disconnectDb, truncateTables } from '../../db'; import { User, Session, Item, Uuid } from '../../services/database/types'; import { createDb, CreateDbOptions } from '../../tools/dbTools'; -import modelFactory, { Options as ModelFactoryOptions } from '../../models/factory'; +import modelFactory from '../../models/factory'; import { AppContext, Env } from '../types'; import config, { initConfig } from '../../config'; import Logger from '@joplin/lib/Logger'; @@ -23,7 +23,6 @@ import MustacheService from '../../services/MustacheService'; import uuidgen from '../uuidgen'; import { createCsrfToken } from '../csrf'; import { cookieSet } from '../cookies'; -import StorageDriverMemory from '../../models/items/storage/StorageDriverMemory'; import { parseEnv } from '../../env'; // Takes into account the fact that this file will be inside the /dist directory @@ -195,7 +194,7 @@ export async function koaAppContext(options: AppContextTestOptions = null): Prom const appLogger = Logger.create('AppTest'); - const baseAppContext = await setupAppContext({} as any, Env.Dev, db_, () => appLogger, { storageDriver: new StorageDriverMemory(1) }); + const baseAppContext = await setupAppContext({} as any, Env.Dev, db_, () => appLogger); // Set type to "any" because the Koa context has many properties and we // don't need to mock all of them. @@ -243,16 +242,8 @@ export function db() { return db_; } -const storageDriverMemory = new StorageDriverMemory(1); - -export function models(options: ModelFactoryOptions = null) { - options = { - storageDriver: storageDriverMemory, - storageDriverFallback: null, - ...options, - }; - - return modelFactory(db(), config(), options); +export function models() { + return modelFactory(db(), config()); } export function parseHtml(html: string): Document {