From cb3a8b7fec7baf25b5dd26c57962452520c59ea8 Mon Sep 17 00:00:00 2001 From: David Boulard Date: Wed, 14 Oct 2020 17:48:17 +0200 Subject: [PATCH] Adding soft lock feature --- README.md | 6 +++ lib/actions/down.js | 17 +++++- lib/actions/up.js | 17 +++++- lib/utils/lock.js | 42 +++++++++++++++ samples/migrate-mongo-config.js | 6 +++ test/actions/down.test.js | 92 ++++++++++++++++++++++++++++++++- test/actions/up.test.js | 92 ++++++++++++++++++++++++++++++++- 7 files changed, 264 insertions(+), 8 deletions(-) create mode 100644 lib/utils/lock.js diff --git a/README.md b/README.md index 4a697f1..09c7ac5 100644 --- a/README.md +++ b/README.md @@ -83,6 +83,12 @@ module.exports = { // Enable the algorithm to create a checksum of the file contents and use that in the comparison to determin // if the file should be run. Requires that scripts are coded to be run multiple times. useFileHash: false + + // The mongodb collection where the lock will be created. + lockCollectionName: "changelog_lock", + + // The value in seconds for the TTL index that will be used for the lock. Value of 0 will disable the feature. + lockTtl: 0 }; ```` diff --git a/lib/actions/down.js b/lib/actions/down.js index e952192..914286e 100644 --- a/lib/actions/down.js +++ b/lib/actions/down.js @@ -1,11 +1,12 @@ const _ = require("lodash"); const { promisify } = require("util"); -const fnArgs = require('fn-args'); +const fnArgs = require("fn-args"); const status = require("./status"); const config = require("../env/config"); const migrationsDir = require("../env/migrationsDir"); -const hasCallback = require('../utils/has-callback'); +const hasCallback = require("../utils/has-callback"); +const lock = require("../utils/lock"); module.exports = async (db, client) => { const downgraded = []; @@ -13,6 +14,16 @@ module.exports = async (db, client) => { const appliedItems = statusItems.filter(item => item.appliedAt !== "PENDING"); const lastAppliedItem = _.last(appliedItems); + if (await lock.exist(db)) { + throw new Error("Could not migrate down, a lock is in place."); + } + + try { + await lock.activate(db); + } catch(err) { + throw new Error(`Could not create a lock: ${err.message}`); + } + if (lastAppliedItem) { try { const migration = await migrationsDir.loadMigration(lastAppliedItem.fileName); @@ -26,6 +37,7 @@ module.exports = async (db, client) => { } } catch (err) { + await lock.clear(db); throw new Error( `Could not migrate down ${lastAppliedItem.fileName}: ${err.message}` ); @@ -40,5 +52,6 @@ module.exports = async (db, client) => { } } + await lock.clear(db); return downgraded; }; diff --git a/lib/actions/up.js b/lib/actions/up.js index 4862573..4230ea2 100644 --- a/lib/actions/up.js +++ b/lib/actions/up.js @@ -1,18 +1,29 @@ const _ = require("lodash"); const pEachSeries = require("p-each-series"); const { promisify } = require("util"); -const fnArgs = require('fn-args'); +const fnArgs = require("fn-args"); const status = require("./status"); const config = require("../env/config"); const migrationsDir = require("../env/migrationsDir"); -const hasCallback = require('../utils/has-callback'); +const hasCallback = require("../utils/has-callback"); +const lock = require("../utils/lock"); module.exports = async (db, client) => { const statusItems = await status(db); const pendingItems = _.filter(statusItems, { appliedAt: "PENDING" }); const migrated = []; + if (await lock.exist(db)) { + throw new Error("Could not migrate up, a lock is in place."); + } + + try { + await lock.activate(db); + } catch(err) { + throw new Error(`Could not create a lock: ${err.message}`); + } + const migrateItem = async item => { try { const migration = await migrationsDir.loadMigration(item.fileName); @@ -31,6 +42,7 @@ module.exports = async (db, client) => { ); error.stack = err.stack; error.migrated = migrated; + await lock.clear(db); throw error; } @@ -49,5 +61,6 @@ module.exports = async (db, client) => { }; await pEachSeries(pendingItems, migrateItem); + await lock.clear(db); return migrated; }; diff --git a/lib/utils/lock.js b/lib/utils/lock.js new file mode 100644 index 0000000..db4dbec --- /dev/null +++ b/lib/utils/lock.js @@ -0,0 +1,42 @@ +const config = require('../env/config'); + +async function getLockCollection(db) { + const { lockCollectionName, lockTtl } = await config.read(); + if (lockTtl <= 0) { + return null; + } + + const lockCollection = db.collection(lockCollectionName); + lockCollection.createIndex({ createdAt: 1 }, { expireAfterSeconds: lockTtl }); + return lockCollection; +} + +async function exist(db) { + const lockCollection = await getLockCollection(db); + if (!lockCollection) { + return false; + } + const foundLocks = await lockCollection.find({}).toArray(); + + return foundLocks.length > 0; +} + +async function activate(db) { + const lockCollection = await getLockCollection(db); + if (lockCollection) { + await lockCollection.insertOne({ createdAt: new Date() }); + } +} + +async function clear(db) { + const lockCollection = await getLockCollection(db); + if (lockCollection) { + await lockCollection.deleteMany({}); + } +} + +module.exports = { + exist, + activate, + clear, +} diff --git a/samples/migrate-mongo-config.js b/samples/migrate-mongo-config.js index 7993812..32ea434 100644 --- a/samples/migrate-mongo-config.js +++ b/samples/migrate-mongo-config.js @@ -22,6 +22,12 @@ const config = { // The mongodb collection where the applied changes are stored. Only edit this when really necessary. changelogCollectionName: "changelog", + // The mongodb collection where the lock will be created. + lockCollectionName: "changelog_lock", + + // The value in seconds for the TTL index that will be used for the lock. Value of 0 will disable the feature. + lockTtl: 0, + // The file extension to create migrations and search for in migration dir migrationFileExtension: ".js", diff --git a/test/actions/down.test.js b/test/actions/down.test.js index 5cbe891..a82e868 100644 --- a/test/actions/down.test.js +++ b/test/actions/down.test.js @@ -7,11 +7,13 @@ describe("down", () => { let down; let status; let config; + let lock; let migrationsDir; let db; let client; let migration; let changelogCollection; + let changelogLockCollection; function mockStatus() { return sinon.stub().returns( @@ -31,7 +33,11 @@ describe("down", () => { function mockConfig() { return { shouldExist: sinon.stub().returns(Promise.resolve()), - read: sinon.stub().returns({ changelogCollectionName: "changelog" }) + read: sinon.stub().returns({ + changelogCollectionName: "changelog", + lockCollectionName: "changelog_lock", + lockTtl: 10 + }) }; } @@ -45,6 +51,7 @@ describe("down", () => { const mock = {}; mock.collection = sinon.stub(); mock.collection.withArgs("changelog").returns(changelogCollection); + mock.collection.withArgs("changelog_lock").returns(changelogLockCollection); return mock; } @@ -66,17 +73,40 @@ describe("down", () => { }; } + function mockChangelogLockCollection() { + const findStub = { + toArray: () => { + return []; + } + } + + return { + insertOne: sinon.stub().returns(Promise.resolve()), + createIndex: sinon.stub().returns(Promise.resolve()), + find: sinon.stub().returns(findStub), + deleteMany: sinon.stub().returns(Promise.resolve()), + } + } + function loadDownWithInjectedMocks() { return proxyquire("../../lib/actions/down", { "./status": status, "../env/config": config, - "../env/migrationsDir": migrationsDir + "../env/migrationsDir": migrationsDir, + "../utils/lock": lock, + }); + } + + function loadLockWithInjectedMocks() { + return proxyquire("../../lib/utils/lock", { + "../env/config": config }); } beforeEach(() => { migration = mockMigration(); changelogCollection = mockChangelogCollection(); + changelogLockCollection = mockChangelogLockCollection(); status = mockStatus(); config = mockConfig(); @@ -84,6 +114,7 @@ describe("down", () => { db = mockDb(); client = mockClient(); + lock = loadLockWithInjectedMocks(); down = loadDownWithInjectedMocks(); }); @@ -179,4 +210,61 @@ describe("down", () => { const items = await down(db); expect(items).to.deep.equal(["20160609113225-last_migration.js"]); }); + + it("should lock if feature is enabled", async() => { + await down(db); + expect(changelogLockCollection.createIndex.called).to.equal(true); + expect(changelogLockCollection.find.called).to.equal(true); + expect(changelogLockCollection.insertOne.called).to.equal(true); + expect(changelogLockCollection.deleteMany.called).to.equal(true); + }); + + it("should ignore lock if feature is disabled", async() => { + config.read = sinon.stub().returns({ + changelogCollectionName: "changelog", + lockCollectionName: "changelog_lock", + lockTtl: 0 + }); + const findStub = { + toArray: () => { + return [{ createdAt: new Date() }]; + } + } + changelogLockCollection.find.returns(findStub); + + await down(db); + expect(changelogLockCollection.createIndex.called).to.equal(false); + expect(changelogLockCollection.find.called).to.equal(false); + }); + + it("should yield an error when unable to create a lock", async() => { + changelogLockCollection.insertOne.returns(Promise.reject(new Error("Kernel panic"))); + + try { + await down(db); + expect.fail("Error was not thrown"); + } catch (err) { + expect(err.message).to.deep.equal( + "Could not create a lock: Kernel panic" + ); + } + }); + + it("should yield an error when changelog is locked", async() => { + const findStub = { + toArray: () => { + return [{ createdAt: new Date() }]; + } + } + changelogLockCollection.find.returns(findStub); + + try { + await down(db); + expect.fail("Error was not thrown"); + } catch (err) { + expect(err.message).to.deep.equal( + "Could not migrate down, a lock is in place." + ); + } + }); }); diff --git a/test/actions/up.test.js b/test/actions/up.test.js index 8508122..517e3f5 100644 --- a/test/actions/up.test.js +++ b/test/actions/up.test.js @@ -7,6 +7,7 @@ describe("up", () => { let up; let status; let config; + let lock; let migrationsDir; let db; let client; @@ -14,6 +15,7 @@ describe("up", () => { let firstPendingMigration; let secondPendingMigration; let changelogCollection; + let changelogLockCollection; function mockStatus() { return sinon.stub().returns( @@ -42,7 +44,9 @@ describe("up", () => { return { shouldExist: sinon.stub().returns(Promise.resolve()), read: sinon.stub().returns({ - changelogCollectionName: "changelog" + changelogCollectionName: "changelog", + lockCollectionName: "changelog_lock", + lockTtl: 10 }) }; } @@ -63,6 +67,7 @@ describe("up", () => { const mock = {}; mock.collection = sinon.stub(); mock.collection.withArgs("changelog").returns(changelogCollection); + mock.collection.withArgs("changelog_lock").returns(changelogLockCollection); return mock; } @@ -84,11 +89,33 @@ describe("up", () => { }; } + function mockChangelogLockCollection() { + const findStub = { + toArray: () => { + return []; + } + } + + return { + insertOne: sinon.stub().returns(Promise.resolve()), + createIndex: sinon.stub().returns(Promise.resolve()), + find: sinon.stub().returns(findStub), + deleteMany: sinon.stub().returns(Promise.resolve()), + } + } + function loadUpWithInjectedMocks() { return proxyquire("../../lib/actions/up", { "./status": status, "../env/config": config, - "../env/migrationsDir": migrationsDir + "../env/migrationsDir": migrationsDir, + "../utils/lock": lock + }); + } + + function loadLockWithInjectedMocks() { + return proxyquire("../../lib/utils/lock", { + "../env/config": config }); } @@ -96,6 +123,7 @@ describe("up", () => { firstPendingMigration = mockMigration(); secondPendingMigration = mockMigration(); changelogCollection = mockChangelogCollection(); + changelogLockCollection = mockChangelogLockCollection(); status = mockStatus(); config = mockConfig(); @@ -103,6 +131,7 @@ describe("up", () => { db = mockDb(); client = mockClient(); + lock = loadLockWithInjectedMocks(); up = loadUpWithInjectedMocks(); }); @@ -200,4 +229,63 @@ describe("up", () => { ); } }); + + it("should lock if feature is enabled", async() => { + await up(db); + expect(changelogLockCollection.createIndex.called).to.equal(true); + expect(changelogLockCollection.find.called).to.equal(true); + expect(changelogLockCollection.insertOne.called).to.equal(true); + expect(changelogLockCollection.deleteMany.called).to.equal(true); + }); + + it("should ignore lock if feature is disabled", async() => { + config.read = sinon.stub().returns({ + changelogCollectionName: "changelog", + lockCollectionName: "changelog_lock", + lockTtl: 0 + }); + const findStub = { + toArray: () => { + return [{ createdAt: new Date() }]; + } + } + changelogLockCollection.find.returns(findStub); + + await up(db); + expect(changelogLockCollection.createIndex.called).to.equal(false); + expect(changelogLockCollection.find.called).to.equal(false); + expect(changelogLockCollection.insertOne.called).to.equal(false); + expect(changelogLockCollection.deleteMany.called).to.equal(false); + }); + + it("should yield an error when unable to create a lock", async() => { + changelogLockCollection.insertOne.returns(Promise.reject(new Error("Kernel panic"))); + + try { + await up(db); + expect.fail("Error was not thrown"); + } catch (err) { + expect(err.message).to.deep.equal( + "Could not create a lock: Kernel panic" + ); + } + }); + + it("should yield an error when changelog is locked", async() => { + const findStub = { + toArray: () => { + return [{ createdAt: new Date() }]; + } + } + changelogLockCollection.find.returns(findStub); + + try { + await up(db); + expect.fail("Error was not thrown"); + } catch (err) { + expect(err.message).to.deep.equal( + "Could not migrate up, a lock is in place." + ); + } + }); });