ds4/src/migration/migrationHelpers.js

163 lines
5 KiB
JavaScript
Raw Normal View History

// SPDX-FileCopyrightText: 2021 Johannes Loher
//
// SPDX-License-Identifier: MIT
import { DS4Actor } from "../documents/actor/actor.js";
import { DS4Item } from "../documents/item/item.js";
import { logger } from "../utils/logger.js";
import { getGame } from "../utils/utils.js";
2021-07-08 02:32:25 +02:00
/**
* @template T
* @typedef {(document: T) => Promise<void>} Migrator
*/
2021-07-08 02:32:25 +02:00
/**
* Migrate a collection.
* @template T
* @param {WorldCollection} collection
* @param {Migrator<T>} migrateDocument
* @returns {Promise<void>} A promise that resolves once the migration is complete
*/
export async function migrateCollection(collection, migrateDocument) {
const { documentName } = collection.constructor;
for (const document of collection) {
logger.info(`Migrating ${documentName} document ${document.name} (${document.id})`);
2021-07-08 02:32:25 +02:00
try {
await migrateDocument(document);
2021-07-08 02:32:25 +02:00
} catch (err) {
logger.error(
`Error during migration of ${documentName} document ${document.name} (${document.id}), continuing anyways.`,
err,
);
2021-07-08 02:32:25 +02:00
}
}
}
/**
* @param {Migrator<ActiveEffect>} [migrateActiveEffect]
* @returns {Migrator<Scene>}
*/
export function getItemMigrator(migrateActiveEffect) {
/**
* @param {Item} item
*/
return async (item) => {
if (migrateActiveEffect) {
for (const effect of item.effects) {
await migrateActiveEffect(effect);
2021-07-08 02:32:25 +02:00
}
}
};
2021-07-08 02:32:25 +02:00
}
/**
* @param {Migrator<Item>} [migrateItem]
* @param {Migrator<ActiveEffect>} [migrateActiveEffect]
* @returns {Migrator<Scene>}
*/
export function getActorMigrator(migrateItem, migrateActiveEffect) {
/**
* @param {Actor} actor
*/
return async (actor) => {
if (migrateItem) {
for (const item of actor.items) {
await migrateItem(item);
}
}
if (migrateActiveEffect) {
for (const effect of actor.effects) {
await migrateActiveEffect(effect);
}
}
};
}
2021-07-08 02:32:25 +02:00
/**
* @param {Migrator<Actor>} [migrateActor]
* @returns {Migrator<Scene>}
*/
export function getSceneMigrator(migrateActor) {
/**
* @param {Scene} scene
*/
return async (scene) => {
if (migrateActor) {
for (const token of scene.tokens) {
if (!token.actorLink && token.actor) {
await migrateActor(token.actor);
}
2021-07-08 02:32:25 +02:00
}
}
};
2021-07-08 02:32:25 +02:00
}
2022-11-21 03:00:46 +01:00
/** @typedef {(pack: CompendiumCollection) => Promise<void>} CompendiumMigrator */
2021-07-08 02:32:25 +02:00
/**
* Migrate world compendium packs.
* @param {CompendiumMigrator} migrateCompendium A function for migrating a single compendium pack
* @returns {Promise<void>} A promise that resolves once the migration is complete
*/
export async function migrateCompendiums(migrateCompendium) {
2021-07-08 02:32:25 +02:00
for (const compendium of getGame().packs ?? []) {
if (compendium.metadata.package !== "world") continue;
if (!["Actor", "Item", "Scene"].includes(compendium.metadata.type)) continue;
2021-07-08 02:32:25 +02:00
await migrateCompendium(compendium);
}
}
/**
* @typedef {object} Migrators
* @property {Migrator<Item>} [migrateItem]
* @property {Migrator<Actor>} [migrateActor]
* @property {Migrator<Scene>} [migrateScene]
*/
/**
* Get a compendium migrator for the given migrators.
* @param {Migrators} [migrators={}] The functions to use for getting update data
* @param {{migrateToTemplateEarly?: boolean}} [options={}] Additional options for the compendium migrator
* @returns {CompendiumMigrator} The resulting compendium migrator
*/
2021-07-08 02:32:25 +02:00
export function getCompendiumMigrator(
{ migrateItem, migrateActor, migrateScene } = {},
2021-07-08 02:32:25 +02:00
{ migrateToTemplateEarly = true } = {},
) {
return async (pack) => {
const type = pack.metadata.type;
if (!["Actor", "Item", "Scene"].includes(type)) return;
const wasLocked = pack.locked;
await pack.configure({ locked: false });
2021-07-08 02:32:25 +02:00
if (migrateToTemplateEarly) {
await pack.migrate();
2021-07-08 02:32:25 +02:00
}
const documents = await pack.getDocuments();
2021-07-08 02:32:25 +02:00
for (const doc of documents) {
try {
logger.info(`Migrating document ${doc.name} (${doc.id}) in compendium ${pack.collection}`);
if (doc instanceof DS4Item && migrateItem) {
await migrateItem(doc);
} else if (doc instanceof DS4Actor && migrateActor) {
await migrateActor(doc);
} else if (doc instanceof Scene && migrateScene) {
await migrateScene(doc);
2021-07-08 02:32:25 +02:00
}
} catch (err) {
2021-09-12 17:48:14 +02:00
logger.error(
`Error during migration of document ${doc.name} (${doc.id}) in compendium ${pack.collection}, continuing anyways.`,
2021-09-12 17:48:14 +02:00
err,
);
2021-07-08 02:32:25 +02:00
}
}
if (!migrateToTemplateEarly) {
await pack.migrate();
2021-07-08 02:32:25 +02:00
}
await pack.configure({ locked: wasLocked });
2021-07-08 02:32:25 +02:00
};
}