Added ability for import to automatically upgrade older db versions
This commit is contained in:
parent
631d232a09
commit
650084ac84
194
src/database.js
194
src/database.js
@ -1,22 +1,48 @@
|
|||||||
import Dexie from "dexie";
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
import Dexie, { Version, DexieOptions } from "dexie";
|
||||||
|
|
||||||
import blobToBuffer from "./helpers/blobToBuffer";
|
import blobToBuffer from "./helpers/blobToBuffer";
|
||||||
import { getGridDefaultInset } from "./helpers/grid";
|
import { getGridDefaultInset } from "./helpers/grid";
|
||||||
import { convertOldActionsToShapes } from "./actions";
|
import { convertOldActionsToShapes } from "./actions";
|
||||||
import { createThumbnail } from "./helpers/image";
|
import { createThumbnail } from "./helpers/image";
|
||||||
|
|
||||||
function loadVersions(db) {
|
// Helper to create a thumbnail for a file in a db
|
||||||
|
async function createDataThumbnail(data) {
|
||||||
|
const url = URL.createObjectURL(new Blob([data.file]));
|
||||||
|
return await Dexie.waitFor(
|
||||||
|
new Promise((resolve) => {
|
||||||
|
let image = new Image();
|
||||||
|
image.onload = async () => {
|
||||||
|
const thumbnail = await createThumbnail(image);
|
||||||
|
resolve(thumbnail);
|
||||||
|
};
|
||||||
|
image.src = url;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @callback VersionCallback
|
||||||
|
* @param {Version} version
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Mapping of version number to their upgrade function
|
||||||
|
* @type {Object.<number, VersionCallback>}
|
||||||
|
*/
|
||||||
|
const versions = {
|
||||||
// v1.2.0
|
// v1.2.0
|
||||||
db.version(1).stores({
|
1(v) {
|
||||||
|
v.stores({
|
||||||
maps: "id, owner",
|
maps: "id, owner",
|
||||||
states: "mapId",
|
states: "mapId",
|
||||||
tokens: "id, owner",
|
tokens: "id, owner",
|
||||||
user: "key",
|
user: "key",
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.2.1 - Move from blob files to array buffers
|
// v1.2.1 - Move from blob files to array buffers
|
||||||
db.version(2)
|
2(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade(async (tx) => {
|
||||||
.upgrade(async (tx) => {
|
|
||||||
const maps = await Dexie.waitFor(tx.table("maps").toArray());
|
const maps = await Dexie.waitFor(tx.table("maps").toArray());
|
||||||
let mapBuffers = {};
|
let mapBuffers = {};
|
||||||
for (let map of maps) {
|
for (let map of maps) {
|
||||||
@ -29,10 +55,10 @@ function loadVersions(db) {
|
|||||||
map.file = mapBuffers[map.id];
|
map.file = mapBuffers[map.id];
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.3.0 - Added new default tokens
|
// v1.3.0 - Added new default tokens
|
||||||
db.version(3)
|
3(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("states")
|
.table("states")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -92,10 +118,10 @@ function loadVersions(db) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.3.1 - Added show grid option
|
// v1.3.1 - Added show grid option
|
||||||
db.version(4)
|
4(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("maps")
|
.table("maps")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -103,10 +129,10 @@ function loadVersions(db) {
|
|||||||
map.showGrid = false;
|
map.showGrid = false;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.4.0 - Added fog subtraction
|
// v1.4.0 - Added fog subtraction
|
||||||
db.version(5)
|
5(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("states")
|
.table("states")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -120,10 +146,10 @@ function loadVersions(db) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.4.2 - Added map resolutions
|
// v1.4.2 - Added map resolutions
|
||||||
db.version(6)
|
6(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("maps")
|
.table("maps")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -132,10 +158,10 @@ function loadVersions(db) {
|
|||||||
map.quality = "original";
|
map.quality = "original";
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.5.0 - Fixed default token rogue spelling
|
// v1.5.0 - Fixed default token rogue spelling
|
||||||
db.version(7)
|
7(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("states")
|
.table("states")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -147,10 +173,10 @@ function loadVersions(db) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.5.0 - Added map snap to grid option
|
// v1.5.0 - Added map snap to grid option
|
||||||
db.version(8)
|
8(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("maps")
|
.table("maps")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -158,10 +184,10 @@ function loadVersions(db) {
|
|||||||
map.snapToGrid = true;
|
map.snapToGrid = true;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.5.1 - Added lock, visibility and modified to tokens
|
// v1.5.1 - Added lock, visibility and modified to tokens
|
||||||
db.version(9)
|
9(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("states")
|
.table("states")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -175,10 +201,10 @@ function loadVersions(db) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.5.1 - Added token prop category and remove isVehicle bool
|
// v1.5.1 - Added token prop category and remove isVehicle bool
|
||||||
db.version(10)
|
10(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("tokens")
|
.table("tokens")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -187,10 +213,10 @@ function loadVersions(db) {
|
|||||||
delete token.isVehicle;
|
delete token.isVehicle;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.5.2 - Added automatic cache invalidation to maps
|
// v1.5.2 - Added automatic cache invalidation to maps
|
||||||
db.version(11)
|
11(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("maps")
|
.table("maps")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -198,10 +224,10 @@ function loadVersions(db) {
|
|||||||
map.lastUsed = map.lastModified;
|
map.lastUsed = map.lastModified;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.5.2 - Added automatic cache invalidation to tokens
|
// v1.5.2 - Added automatic cache invalidation to tokens
|
||||||
db.version(12)
|
12(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("tokens")
|
.table("tokens")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -209,10 +235,10 @@ function loadVersions(db) {
|
|||||||
token.lastUsed = token.lastModified;
|
token.lastUsed = token.lastModified;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.6.0 - Added map grouping and grid scale and offset
|
// v1.6.0 - Added map grouping and grid scale and offset
|
||||||
db.version(13)
|
13(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("maps")
|
.table("maps")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -232,10 +258,10 @@ function loadVersions(db) {
|
|||||||
delete map.gridType;
|
delete map.gridType;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.6.0 - Added token grouping
|
// v1.6.0 - Added token grouping
|
||||||
db.version(14)
|
14(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("tokens")
|
.table("tokens")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -243,10 +269,10 @@ function loadVersions(db) {
|
|||||||
token.group = "";
|
token.group = "";
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.6.1 - Added width and height to tokens
|
// v1.6.1 - Added width and height to tokens
|
||||||
db.version(15)
|
15(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade(async (tx) => {
|
||||||
.upgrade(async (tx) => {
|
|
||||||
const tokens = await Dexie.waitFor(tx.table("tokens").toArray());
|
const tokens = await Dexie.waitFor(tx.table("tokens").toArray());
|
||||||
let tokenSizes = {};
|
let tokenSizes = {};
|
||||||
for (let token of tokens) {
|
for (let token of tokens) {
|
||||||
@ -269,10 +295,10 @@ function loadVersions(db) {
|
|||||||
token.height = tokenSizes[token.id].height;
|
token.height = tokenSizes[token.id].height;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// v1.7.0 - Added note tool
|
// v1.7.0 - Added note tool
|
||||||
db.version(16)
|
16(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("states")
|
.table("states")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -281,11 +307,10 @@ function loadVersions(db) {
|
|||||||
state.editFlags = [...state.editFlags, "notes"];
|
state.editFlags = [...state.editFlags, "notes"];
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// 1.7.0 (hotfix) - Optimized fog shape edits to only include needed data
|
// 1.7.0 (hotfix) - Optimized fog shape edits to only include needed data
|
||||||
db.version(17)
|
17(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("states")
|
.table("states")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -305,11 +330,10 @@ function loadVersions(db) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// 1.8.0 - Added note text only mode, converted draw and fog representations
|
// 1.8.0 - Added note text only mode, converted draw and fog representations
|
||||||
db.version(18)
|
18(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade((tx) => {
|
||||||
.upgrade((tx) => {
|
|
||||||
return tx
|
return tx
|
||||||
.table("states")
|
.table("states")
|
||||||
.toCollection()
|
.toCollection()
|
||||||
@ -333,25 +357,10 @@ function loadVersions(db) {
|
|||||||
delete state.fogDrawActionIndex;
|
delete state.fogDrawActionIndex;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
async function createDataThumbnail(data) {
|
|
||||||
const url = URL.createObjectURL(new Blob([data.file]));
|
|
||||||
return await Dexie.waitFor(
|
|
||||||
new Promise((resolve) => {
|
|
||||||
let image = new Image();
|
|
||||||
image.onload = async () => {
|
|
||||||
const thumbnail = await createThumbnail(image);
|
|
||||||
resolve(thumbnail);
|
|
||||||
};
|
|
||||||
image.src = url;
|
|
||||||
})
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 1.8.0 - Add thumbnail to maps and add measurement to grid
|
// 1.8.0 - Add thumbnail to maps and add measurement to grid
|
||||||
db.version(19)
|
19(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade(async (tx) => {
|
||||||
.upgrade(async (tx) => {
|
|
||||||
const maps = await Dexie.waitFor(tx.table("maps").toArray());
|
const maps = await Dexie.waitFor(tx.table("maps").toArray());
|
||||||
const thumbnails = {};
|
const thumbnails = {};
|
||||||
for (let map of maps) {
|
for (let map of maps) {
|
||||||
@ -365,11 +374,10 @@ function loadVersions(db) {
|
|||||||
map.grid.measurement = { type: "chebyshev", scale: "5ft" };
|
map.grid.measurement = { type: "chebyshev", scale: "5ft" };
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
// 1.8.0 - Add thumbnail to tokens
|
// 1.8.0 - Add thumbnail to tokens
|
||||||
db.version(20)
|
20(v) {
|
||||||
.stores({})
|
v.stores({}).upgrade(async (tx) => {
|
||||||
.upgrade(async (tx) => {
|
|
||||||
const tokens = await Dexie.waitFor(tx.table("tokens").toArray());
|
const tokens = await Dexie.waitFor(tx.table("tokens").toArray());
|
||||||
const thumbnails = {};
|
const thumbnails = {};
|
||||||
for (let token of tokens) {
|
for (let token of tokens) {
|
||||||
@ -382,11 +390,35 @@ function loadVersions(db) {
|
|||||||
token.thumbnail = thumbnails[token.id];
|
token.thumbnail = thumbnails[token.id];
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const latestVersion = 20;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load versions onto a database up to a specific version number
|
||||||
|
* @param {Dexie} db
|
||||||
|
* @param {number=} upTo version number to load up to, latest version if undefined
|
||||||
|
*/
|
||||||
|
export function loadVersions(db, upTo = latestVersion) {
|
||||||
|
for (let versionNumber = 1; versionNumber <= upTo; versionNumber++) {
|
||||||
|
versions[versionNumber](db.version(versionNumber));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the dexie database used in DatabaseContext
|
/**
|
||||||
export function getDatabase(options, name="OwlbearRodeoDB") {
|
* Get a Dexie database with a name and versions applied
|
||||||
|
* @param {DexieOptions} options
|
||||||
|
* @param {string=} name
|
||||||
|
* @param {number=} versionNumber
|
||||||
|
* @returns {Dexie}
|
||||||
|
*/
|
||||||
|
export function getDatabase(
|
||||||
|
options,
|
||||||
|
name = "OwlbearRodeoDB",
|
||||||
|
versionNumber = latestVersion
|
||||||
|
) {
|
||||||
let db = new Dexie(name, options);
|
let db = new Dexie(name, options);
|
||||||
loadVersions(db);
|
loadVersions(db, versionNumber);
|
||||||
return db;
|
return db;
|
||||||
}
|
}
|
||||||
|
@ -48,14 +48,12 @@ function ImportExportModal({ isOpen, onRequestClose }) {
|
|||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
backgroundTaskRunningRef.current = true;
|
backgroundTaskRunningRef.current = true;
|
||||||
try {
|
try {
|
||||||
// Ensure import DB is cleared before importing new data
|
|
||||||
let importDB = getDatabase({}, importDBName);
|
|
||||||
importDB.delete();
|
|
||||||
await worker.importData(
|
await worker.importData(
|
||||||
file,
|
file,
|
||||||
importDBName,
|
importDBName,
|
||||||
Comlink.proxy(handleDBProgress)
|
Comlink.proxy(handleDBProgress)
|
||||||
);
|
);
|
||||||
|
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
setShowImportSelector(true);
|
setShowImportSelector(true);
|
||||||
backgroundTaskRunningRef.current = false;
|
backgroundTaskRunningRef.current = false;
|
||||||
@ -106,6 +104,7 @@ function ImportExportModal({ isOpen, onRequestClose }) {
|
|||||||
async function handleImportSelectorClose() {
|
async function handleImportSelectorClose() {
|
||||||
const importDB = getDatabase({}, importDBName);
|
const importDB = getDatabase({}, importDBName);
|
||||||
await importDB.delete();
|
await importDB.delete();
|
||||||
|
importDB.close();
|
||||||
setShowImportSelector(false);
|
setShowImportSelector(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -157,6 +156,8 @@ function ImportExportModal({ isOpen, onRequestClose }) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
await importDB.delete();
|
await importDB.delete();
|
||||||
|
importDB.close();
|
||||||
|
db.close();
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
backgroundTaskRunningRef.current = false;
|
backgroundTaskRunningRef.current = false;
|
||||||
window.location.reload();
|
window.location.reload();
|
||||||
|
@ -91,8 +91,19 @@ let service = {
|
|||||||
if (importMeta.data.databaseName !== db.name) {
|
if (importMeta.data.databaseName !== db.name) {
|
||||||
throw new Error("Unable to import database, name mismatch");
|
throw new Error("Unable to import database, name mismatch");
|
||||||
}
|
}
|
||||||
|
if (importMeta.data.databaseVersion > db.verno) {
|
||||||
|
throw new Error(
|
||||||
|
`Database version differs. Current database is in version ${db.verno} but export is ${importMeta.data.databaseVersion}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
let importDB = getDatabase({}, databaseName);
|
// Ensure import DB is cleared before importing new data
|
||||||
|
let importDB = getDatabase({}, databaseName, 0);
|
||||||
|
await importDB.delete();
|
||||||
|
importDB.close();
|
||||||
|
|
||||||
|
// Load import database up to it's desired version
|
||||||
|
importDB = getDatabase({}, databaseName, importMeta.data.databaseVersion);
|
||||||
await importInto(importDB, data, {
|
await importInto(importDB, data, {
|
||||||
progressCallback,
|
progressCallback,
|
||||||
acceptNameDiff: true,
|
acceptNameDiff: true,
|
||||||
@ -107,9 +118,10 @@ let service = {
|
|||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
|
acceptVersionDiff: true,
|
||||||
});
|
});
|
||||||
db.close();
|
|
||||||
importDB.close();
|
importDB.close();
|
||||||
|
db.close();
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user