coh-content-db 2.0.0-rc.1 → 2.0.0-rc.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.editorconfig +10 -11
- package/.github/workflows/release.yml +1 -1
- package/README.md +13 -23
- package/dist/coh-content-db.d.ts +32 -43
- package/dist/coh-content-db.js +175 -177
- package/dist/coh-content-db.js.map +1 -1
- package/dist/coh-content-db.mjs +175 -177
- package/dist/coh-content-db.mjs.map +1 -1
- package/package.json +1 -1
- package/src/main/api/{server-group-data.ts → content-bundle.ts} +8 -15
- package/src/main/changelog.ts +4 -2
- package/src/main/db/alternates.ts +10 -1
- package/src/main/db/bundle-metadata.ts +40 -0
- package/src/main/db/coh-content-database.ts +69 -16
- package/src/main/index.ts +2 -2
- package/src/test/api/badge-data.test.ts +1 -1
- package/src/test/api/{server-group-data.fixture.ts → content-bundle.fixture.ts} +3 -4
- package/src/test/api/{server-group-data.test.ts → content-bundle.test.ts} +4 -5
- package/src/test/changelog.test.ts +3 -3
- package/src/test/db/alternates.test.ts +21 -2
- package/src/test/db/badge.test.ts +1 -1
- package/src/test/db/bundle-metadata.test.ts +67 -0
- package/src/test/db/coh-content-database.test.ts +125 -24
- package/src/main/db/server-group.ts +0 -112
- package/src/test/db/server-group.test.ts +0 -124
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"coh-content-db.mjs","sources":["../src/main/api/alignment.ts","../src/main/api/badge-partial-type.ts","../src/main/api/badge-type.ts","../src/main/api/enhancement-category.ts","../src/main/api/plaque-type.ts","../src/main/api/sex.ts","../src/main/db/key.ts","../src/main/db/archetype.ts","../src/main/db/badge-partial.ts","../src/main/db/alternates.ts","../src/main/db/badge.ts","../src/main/db/vidiot-map-point-of-interest.ts","../src/main/db/vidiot-map.ts","../src/main/db/game-map.ts","../src/main/db/server-group.ts","../src/main/db/coh-content-database.ts","../src/main/changelog.ts","../src/main/util.ts"],"sourcesContent":["export const ALIGNMENT = ['H', 'V', 'P'] as const\n\nexport type Alignment = typeof ALIGNMENT[number]\n","export const BADGE_PARTIAL_TYPE = [\n 'PLAQUE',\n 'BADGE',\n 'INVENTION',\n 'INVENTION_PLUS_ONE', // Some invention badges require you to build x of two different invention levels, 'plus one of either level'.\n] as const\n\nexport type BadgePartialType = typeof BADGE_PARTIAL_TYPE[number]\n","export const BADGE_TYPE = [\n 'EXPLORATION',\n 'HISTORY',\n 'ACCOMPLISHMENT',\n 'ACHIEVEMENT',\n 'ACCOLADE',\n 'GLADIATOR',\n 'VETERAN',\n 'PVP',\n 'INVENTION',\n 'DEFEAT',\n 'EVENT',\n 'OUROBOROS',\n 'CONSIGNMENT',\n 'DAY_JOB',\n 'AE',\n] as const\n\nexport type BadgeType = typeof BADGE_TYPE[number]\n","export const ENHANCEMENT_CATEGORY = [\n 'DEFENSE_DEBUFF',\n 'TO_HIT_DEBUFF',\n 'TAUNT',\n 'CONFUSE',\n 'HEALING',\n 'DEFENSE_BUFF',\n 'RESIST_DAMAGE',\n 'INTANGIBILITY',\n 'SLEEP',\n 'SLOW',\n 'HOLD',\n 'STUN',\n 'IMMOBILIZE',\n 'FEAR',\n 'ENDURANCE_MODIFICATION',\n 'ENDURANCE_REDUCTION',\n 'RECHARGE_REDUCTION',\n 'INTERRUPT_DURATION',\n 'ACCURACY',\n 'TO_HIT_BUFF',\n 'DAMAGE',\n 'KNOCKBACK',\n 'RUN_SPEED',\n 'JUMP',\n 'FLY_SPEED',\n 'RANGE',\n] as const\n\nexport type EnhancementCategory = typeof ENHANCEMENT_CATEGORY[number]\n","export const PLAQUE_TYPE = [\n 'WALL_PLAQUE',\n 'MONUMENT',\n]\n\nexport type PlaqueType = typeof PLAQUE_TYPE[number]\n","export const SEX = ['M', 'F'] as const\n\nexport type Sex = typeof SEX[number]\n","const INVALID_KEY_PATTERN = /[^a-z0-9-]/\n\nexport class Key {\n readonly #value: string\n\n constructor(value: string) {\n this.#validateKey(value)\n this.#value = value\n }\n\n get value(): string {\n return this.#value\n }\n\n #validateKey(key: string): void {\n if (INVALID_KEY_PATTERN.test(key)) throw new Error(`Invalid key: [${key}]; Keys can only contain lowercase characters, numbers and dashes.`)\n }\n}\n","import { Key } from './key'\nimport { ArchetypeData } from '../api/archetype-data'\n\nexport class Archetype {\n readonly key: string\n readonly name: string\n readonly description?: string\n\n constructor(data: ArchetypeData) {\n this.key = new Key(data.key).value\n this.name = data.name\n this.description = data.description\n }\n}\n","import { BadgePartialData } from '../api/badge-partial-data'\nimport { PlaqueType } from '../api/plaque-type'\nimport { BadgePartialType } from '../api/badge-partial-type'\nimport { EnhancementCategory } from '../api/enhancement-category'\nimport { Key } from './key'\n\nexport class BadgePartial {\n readonly key: string\n readonly type: BadgePartialType | string\n readonly mapKey?: string\n readonly loc?: number[]\n readonly plaqueType?: PlaqueType | string\n readonly inscription?: string\n readonly vidiotMapKey?: string\n readonly badgeKey?: string\n readonly inventionLevel?: number\n readonly inventionTypes?: (EnhancementCategory | string)[]\n readonly inventionCount?: number\n readonly notes?: string\n\n constructor(data: BadgePartialData) {\n this.key = new Key(data.key).value\n this.type = data.type\n this.mapKey = data.mapKey\n this.loc = data.loc\n this.plaqueType = data.plaqueType\n this.inscription = data.inscription\n this.vidiotMapKey = data.vidiotMapKey\n this.badgeKey = data.badgeKey\n this.inventionLevel = data.inventionLevel\n this.inventionTypes = data.inventionTypes\n this.inventionCount = data.inventionCount\n this.notes = data.notes\n }\n}\n","import { AlternateData } from '../api/alternate-data'\nimport { Sex } from '../api/sex'\nimport { Alignment } from '../api/alignment'\n\nconst ALIGNMENT_SORT: Record<string, number> = { H: 2, V: 1, P: 0 }\nconst SEX_SORT: Record<string, number> = { M: 1, F: 0 }\n\nexport class Alternates<T> {\n readonly #sortedValues: AlternateData<T>[] = []\n\n constructor(values: AlternateData<T>[]) {\n this.#sortedValues = values.sort()\n this.#sortedValues.sort((a, b) => this.#compareAlternates(a, b))\n }\n\n getValue(alignment?: Alignment | string, sex?: Sex | string): T | undefined {\n for (let index = this.#sortedValues.length; index--;) {\n const entry = this.#sortedValues[index]\n if ((entry.alignment === undefined || entry.alignment === alignment)\n && (entry.sex === undefined || entry.sex === sex)\n ) return entry.value\n }\n return undefined\n }\n\n /**\n * Get the default value for this list of alternates, the value with the highest priority and lowest specificity.\n */\n get default(): T | undefined {\n return this.#sortedValues[0]?.value\n }\n\n /**\n * Get the list of alternates sorted in canonical order (alignment then sex, low to high specificity).\n */\n get canonical(): AlternateData<T>[] {\n return this.#sortedValues\n }\n\n #compareAlternates(a: AlternateData<T>, b: AlternateData<T>): number {\n const aSpecificity = (a.alignment ? 2 : 0) + (a.sex ? 1 : 0)\n const bSpecificity = (b.alignment ? 2 : 0) + (b.sex ? 1 : 0)\n if (aSpecificity !== bSpecificity) return aSpecificity - bSpecificity // Order first by least-specific\n\n const alignmentComparison = this.#compareAlignment(a.alignment, b.alignment) // Next by alignment\n if (alignmentComparison !== 0) return alignmentComparison\n\n const sexComparison = this.#compareSex(a.sex, b.sex) // Last by sex\n if (sexComparison !== 0) return sexComparison\n\n return String(a.value).localeCompare(String(b.value))\n }\n\n #compareAlignment(a: Alignment | string | undefined, b: Alignment | string | undefined): number {\n if (a === b) return 0\n if (a === undefined && b !== undefined) return -1\n if (b === undefined && a !== undefined) return 1\n\n const aSort = a === undefined ? -1 : ALIGNMENT_SORT[a] ?? -1 // Unknown values get -1 priority\n const bSort = b === undefined ? -1 : ALIGNMENT_SORT[b] ?? -1\n\n if (aSort !== bSort) return bSort - aSort\n\n // Unknown values (not in ALIGNMENT_SORT) are sorted alphabetically\n return a?.localeCompare(b ?? '') ?? 0\n }\n\n #compareSex(a?: Sex | string | undefined, b?: Sex | string | undefined): number {\n if (a === b) return 0\n if (a === undefined && b !== undefined) return -1\n if (b === undefined && a !== undefined) return 1\n\n const aSort = SEX_SORT[a ?? -1] ?? -1 // Unknown values get -1 priority\n const bSort = SEX_SORT[b ?? -1] ?? -1\n\n if (aSort !== bSort) return bSort - aSort\n\n // Unknown values (not in SEX_SORT) are sorted alphabetically\n return a?.localeCompare(b ?? '') ?? 0\n }\n}\n","import { BadgeType } from '../api/badge-type'\nimport { Alignment } from '../api/alignment'\nimport { Link } from '../api/link'\nimport { BadgeData } from '../api/badge-data'\nimport { BadgePartial } from './badge-partial'\nimport { Key } from './key'\nimport { Alternates } from './alternates'\n\nexport class Badge {\n readonly #partialsIndex: Record<string, BadgePartial> = {}\n\n /**\n * The database key for this badge.\n */\n readonly key: string\n\n /**\n * The type of badge.\n */\n readonly type: BadgeType | string\n\n /**\n * The name of this badge.\n *\n * May vary by character sex or alignment.\n */\n readonly name: Alternates<string>\n\n /**\n * The character alignments that this badge is available to.\n */\n readonly alignment: Alignment[]\n\n /**\n * The badge text as it appears in-game. May vary by character sex or alignment.\n */\n readonly badgeText: Alternates<string>\n\n /**\n * Description of how to acquire the badge.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly acquisition?: string\n\n /**\n * Absolute URL to this badge's icon.\n *\n * May vary by character sex or alignment.\n */\n readonly icon: Alternates<string>\n\n /**\n * Freeform notes or tips about the badge.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly notes?: string\n\n /**\n * List of external links for this Badge. Wiki, forums, etc.\n */\n readonly links?: Link[]\n\n /**\n * For exploration badges, the key of the {@link GameMap} that this badge is found on.\n */\n readonly mapKey?: string\n\n /**\n * For exploration badges, the `/loc` coordinates of the badge on the in-game map.\n */\n readonly loc?: [number, number, number]\n\n /**\n * For badges that appear on a Vidiot Map, the number or letter the badge appears as.\n */\n readonly vidiotMapKey?: string\n\n /**\n * ID used with the in-game `/settitle` command to apply the badge.\n */\n readonly setTitle?: {\n /**\n * `/settitle` id.\n */\n id?: number\n /**\n * `/settitle` id if different for praetorian characters.\n */\n praetorianId?: number\n }\n\n /**\n * A description of the effect the badge will have, such as a buff or granting a temporary power.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly effect?: string\n\n /**\n * A list of requirements for badges that have partial fulfilment steps, such as visiting plaques for history badges, or collecting other badges for meta-badges like accolades.\n */\n readonly partials?: BadgePartial[]\n\n /**\n * Some badges are not included in the badge total count... such as Flames of Prometheus, which can be removed by redeeming it for a Notice of the Well.\n */\n readonly ignoreInTotals: boolean\n\n constructor(data: BadgeData) {\n this.key = new Key(data.key).value\n this.type = data.type\n this.name = new Alternates(data.name)\n this.alignment = data.alignment\n this.badgeText = new Alternates(data.badgeText ?? [])\n this.acquisition = data.acquisition\n this.icon = new Alternates(data.icon ?? [])\n this.notes = data.notes\n this.links = data.links\n this.mapKey = data.mapKey\n this.loc = data.loc\n this.effect = data.effect\n this.vidiotMapKey = data.vidiotMapKey\n this.setTitle = data.setTitle\n this.ignoreInTotals = data.ignoreInTotals ?? false\n\n this.partials = data.partials?.map((data) => {\n if (this.#partialsIndex[data.key] !== undefined) throw new Error(`Duplicate badge partial key [${data.key}]`)\n const badge = new BadgePartial(data)\n this.#partialsIndex[badge.key] = badge\n return badge\n })\n }\n\n getPartial(key: string): BadgePartial {\n const result = this.#partialsIndex[key]\n if (result === undefined) throw new Error(`Unknown badge partial key [${key}]`)\n return result\n }\n}\n","import { VidiotMapPointOfInterestData } from '../api/vidiot-map-point-of-interest-data'\n\nexport class VidiotMapPointOfInterest {\n /**\n * The pixel-space position of the PoI on the map graphic.\n *\n * Screen-space, pixels from top-left `[0, 0]`.\n */\n readonly pos?: [number, number]\n\n /**\n * Freeform notes about the PoI.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly notes?: string\n\n /**\n * If the POI is a zone transfer, the map it transfers to.\n */\n readonly mapKey?: string\n\n /**\n * If the POI is a badge, the badge.\n */\n readonly badgeKey?: string\n\n /**\n * If the POI is a partial for a badge, the partial key.\n */\n readonly badgePartialKey?: string\n\n constructor(data: VidiotMapPointOfInterestData) {\n this.pos = data.pos\n this.notes = data.notes\n this.mapKey = data.mapKey\n this.badgeKey = data.badgeKey\n this.badgePartialKey = data.badgePartialKey\n }\n}\n","import { VidiotMapData } from '../api/vidiot-map-data'\nimport { VidiotMapPointOfInterest } from './vidiot-map-point-of-interest'\n\nexport class VidiotMap {\n /**\n * URL of the map image.\n */\n readonly imageUrl: string\n\n /**\n * Name to display for the Vidiot map.\n */\n readonly name?: string\n\n /**\n * List of Points of Interest labelled on the image.\n */\n readonly pointsOfInterest?: VidiotMapPointOfInterest[]\n\n constructor(data: VidiotMapData) {\n this.imageUrl = data.imageUrl\n this.name = data.name\n this.pointsOfInterest = data.pointsOfInterest?.map(data => new VidiotMapPointOfInterest(data))\n }\n}\n","import { VidiotMap } from './vidiot-map'\nimport { Link } from '../api/link'\nimport { GameMapData } from '../api/game-map-data'\nimport { Key } from './key'\n\nexport class GameMap {\n /**\n * The database key for this map.\n */\n readonly key: string\n\n /**\n * The name of the map as it appears in-game.\n */\n readonly name: string\n\n /**\n * List of external links for this Map. Wiki, forums, etc.\n */\n readonly links?: Link[]\n\n /**\n * List of Vidiot Map assets for this map.\n */\n readonly vidiotMaps?: VidiotMap[]\n\n constructor(data: GameMapData) {\n this.key = new Key(data.key).value\n this.name = data.name\n this.links = data.links\n this.vidiotMaps = data.vidiotMaps?.map(data => new VidiotMap(data))\n }\n}\n","import { Archetype } from './archetype'\nimport { Badge } from './badge'\nimport { ServerGroupData } from '../api/server-group-data'\nimport { Key } from './key'\nimport { Change } from '../api/change'\nimport { GameMap } from './game-map'\nimport { Link } from '../api/link'\n\nexport class ServerGroup {\n readonly #archetypeIndex: Record<string, Archetype> = {}\n readonly #mapIndex: Record<string, GameMap> = {}\n readonly #badgeIndex: Record<string, Badge> = {}\n\n /**\n * The database key for this server group.\n */\n readonly key: string\n\n /**\n * Name of the server group.\n */\n readonly name: string\n\n /**\n * Description of the server group.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly description?: string\n\n /**\n * Repository where the db content package is maintained.\n */\n readonly repository?: string\n\n /**\n * List of external links for this Server Group. Wiki, forums, etc.\n */\n readonly links?: Link[]\n\n /**\n * List of the game server names in this server group.\n * Torchbearer, Excelsior, etc.\n */\n readonly servers: string[]\n\n /**\n * List of archetypes available in this server group.\n */\n readonly archetypes: Archetype[]\n\n /**\n * List of game maps supported by this server group.\n */\n readonly maps: GameMap[]\n\n /**\n * List of badges available on this server group.\n */\n readonly badges: Badge[]\n\n /**\n * Change log for this data package.\n */\n readonly changelog?: Change[]\n\n constructor(data: ServerGroupData) {\n this.key = new Key(data.key).value\n this.name = data.name\n this.description = data.description\n this.repository = data.repository\n this.links = data.links\n this.servers = data.servers ?? []\n this.archetypes = data.archetypes?.map((data) => {\n if (this.#archetypeIndex[data.key] !== undefined) throw new Error(`Duplicate archetype key [${data.key}]`)\n const archetype = new Archetype(data)\n this.#archetypeIndex[archetype.key] = archetype\n return archetype\n }) ?? []\n this.maps = data.maps?.map((data) => {\n if (this.#mapIndex[data.key] !== undefined) throw new Error(`Duplicate map key [${data.key}]`)\n const map = new GameMap(data)\n this.#mapIndex[map.key] = map\n return map\n }) ?? []\n this.badges = data.badges?.map((data) => {\n if (this.#badgeIndex[data.key] !== undefined) throw new Error(`Duplicate badge key [${data.key}]`)\n const badge = new Badge(data)\n this.#badgeIndex[badge.key] = badge\n return badge\n }) ?? []\n this.changelog = data.changelog\n }\n\n getArchetype(key: string): Archetype {\n const result = this.#archetypeIndex[key]\n if (result === undefined) throw new Error(`Unknown archetype key [${key}]`)\n return result\n }\n\n getMap(key: string): GameMap {\n const result = this.#mapIndex[key]\n if (result === undefined) throw new Error(`Unknown map key [${key}]`)\n return result\n }\n\n getBadge(key: string): Badge {\n const result = this.#badgeIndex[key]\n if (result === undefined) throw new Error(`Unknown badge key [${key}]`)\n return result\n }\n}\n","import { ServerGroup } from './server-group'\nimport { ServerGroupData } from '../api/server-group-data'\n\nexport class CohContentDatabase {\n readonly #serverGroups: Record<string, ServerGroup> = {}\n\n /**\n * Load a server group data package into the database.\n * @param data The data to load.\n */\n loadServerGroupData(data: ServerGroupData): void {\n this.#serverGroups[data.key] = new ServerGroup(data)\n }\n\n /**\n * Get all the server groups currently loaded in the database.\n */\n listServerGroups(): ServerGroup[] {\n return Object.values(this.#serverGroups)\n }\n\n /**\n * get a server group by key.\n * @param serverGroupKey The key.\n */\n getServerGroup(serverGroupKey: string): ServerGroup | null {\n return this.#serverGroups[serverGroupKey]\n }\n}\n","import { Change } from './api/change'\n\nexport const CHANGELOG: Change[] = [\n {\n version: '2.0.0',\n date: new Date('2025-03-12'),\n description: ''\n + '* Replaced redundant interfaces with their concrete equivalents.\\n'\n + '* Replaced enums with extensible union types; Server groups with new badge types, enhancement types, etc. can now extend them locally.\\n'\n + '* Removed the `serverGroup` property from entities to simplify the object tree; Server group context will need to be managed separately.\\n'\n + '* Standardized pluralization of some field names (name, icon).\\n'\n + '* Combined `settitle` ids into a single tuple field.\\n'\n + '* Change from GNU to The Unlicense.\\n'\n + '* Removed dependency on lodash. There are now no third-party runtime dependencies.\\n'\n + '* Moved from webpack to rollup for packaging.\\n'\n + '* Add eslint for linting.\\n'\n + '* Add jest for unit tests.\\n'\n + '* Added GitHub Actions for CI.\\n',\n },\n]\n","/**\n * Create a reference string that can be used in most text strings to display a link to the given badge.\n * @param target The badge or badge key to target.\n */\nexport function createBadgeReference(target: string | { key: string }): string {\n const key = typeof target === 'string' ? target : target.key\n return `[badge:${key}]`\n}\n\n/**\n * Create a reference string that can be used in most text strings to display a link to the given map.\n * @param target The {@link GameMap} or map key to target.\n */\nexport function createMapReference(target: string | { key: string }): string {\n const key = typeof target === 'string' ? target : target.key\n return `[map:${key}]`\n}\n"],"names":["__privateAdd","__privateMethod","__privateSet","__privateGet","__publicField","data"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"coh-content-db.mjs","sources":["../src/main/api/alignment.ts","../src/main/api/badge-partial-type.ts","../src/main/api/badge-type.ts","../src/main/api/enhancement-category.ts","../src/main/api/plaque-type.ts","../src/main/api/sex.ts","../src/main/db/key.ts","../src/main/db/archetype.ts","../src/main/db/badge-partial.ts","../src/main/db/alternates.ts","../src/main/db/badge.ts","../src/main/db/vidiot-map-point-of-interest.ts","../src/main/db/vidiot-map.ts","../src/main/db/game-map.ts","../src/main/db/bundle-metadata.ts","../src/main/db/coh-content-database.ts","../src/main/changelog.ts","../src/main/util.ts"],"sourcesContent":["export const ALIGNMENT = ['H', 'V', 'P'] as const\n\nexport type Alignment = typeof ALIGNMENT[number]\n","export const BADGE_PARTIAL_TYPE = [\n 'PLAQUE',\n 'BADGE',\n 'INVENTION',\n 'INVENTION_PLUS_ONE', // Some invention badges require you to build x of two different invention levels, 'plus one of either level'.\n] as const\n\nexport type BadgePartialType = typeof BADGE_PARTIAL_TYPE[number]\n","export const BADGE_TYPE = [\n 'EXPLORATION',\n 'HISTORY',\n 'ACCOMPLISHMENT',\n 'ACHIEVEMENT',\n 'ACCOLADE',\n 'GLADIATOR',\n 'VETERAN',\n 'PVP',\n 'INVENTION',\n 'DEFEAT',\n 'EVENT',\n 'OUROBOROS',\n 'CONSIGNMENT',\n 'DAY_JOB',\n 'AE',\n] as const\n\nexport type BadgeType = typeof BADGE_TYPE[number]\n","export const ENHANCEMENT_CATEGORY = [\n 'DEFENSE_DEBUFF',\n 'TO_HIT_DEBUFF',\n 'TAUNT',\n 'CONFUSE',\n 'HEALING',\n 'DEFENSE_BUFF',\n 'RESIST_DAMAGE',\n 'INTANGIBILITY',\n 'SLEEP',\n 'SLOW',\n 'HOLD',\n 'STUN',\n 'IMMOBILIZE',\n 'FEAR',\n 'ENDURANCE_MODIFICATION',\n 'ENDURANCE_REDUCTION',\n 'RECHARGE_REDUCTION',\n 'INTERRUPT_DURATION',\n 'ACCURACY',\n 'TO_HIT_BUFF',\n 'DAMAGE',\n 'KNOCKBACK',\n 'RUN_SPEED',\n 'JUMP',\n 'FLY_SPEED',\n 'RANGE',\n] as const\n\nexport type EnhancementCategory = typeof ENHANCEMENT_CATEGORY[number]\n","export const PLAQUE_TYPE = [\n 'WALL_PLAQUE',\n 'MONUMENT',\n]\n\nexport type PlaqueType = typeof PLAQUE_TYPE[number]\n","export const SEX = ['M', 'F'] as const\n\nexport type Sex = typeof SEX[number]\n","const INVALID_KEY_PATTERN = /[^a-z0-9-]/\n\nexport class Key {\n readonly #value: string\n\n constructor(value: string) {\n this.#validateKey(value)\n this.#value = value\n }\n\n get value(): string {\n return this.#value\n }\n\n #validateKey(key: string): void {\n if (INVALID_KEY_PATTERN.test(key)) throw new Error(`Invalid key: [${key}]; Keys can only contain lowercase characters, numbers and dashes.`)\n }\n}\n","import { Key } from './key'\nimport { ArchetypeData } from '../api/archetype-data'\n\nexport class Archetype {\n readonly key: string\n readonly name: string\n readonly description?: string\n\n constructor(data: ArchetypeData) {\n this.key = new Key(data.key).value\n this.name = data.name\n this.description = data.description\n }\n}\n","import { BadgePartialData } from '../api/badge-partial-data'\nimport { PlaqueType } from '../api/plaque-type'\nimport { BadgePartialType } from '../api/badge-partial-type'\nimport { EnhancementCategory } from '../api/enhancement-category'\nimport { Key } from './key'\n\nexport class BadgePartial {\n readonly key: string\n readonly type: BadgePartialType | string\n readonly mapKey?: string\n readonly loc?: number[]\n readonly plaqueType?: PlaqueType | string\n readonly inscription?: string\n readonly vidiotMapKey?: string\n readonly badgeKey?: string\n readonly inventionLevel?: number\n readonly inventionTypes?: (EnhancementCategory | string)[]\n readonly inventionCount?: number\n readonly notes?: string\n\n constructor(data: BadgePartialData) {\n this.key = new Key(data.key).value\n this.type = data.type\n this.mapKey = data.mapKey\n this.loc = data.loc\n this.plaqueType = data.plaqueType\n this.inscription = data.inscription\n this.vidiotMapKey = data.vidiotMapKey\n this.badgeKey = data.badgeKey\n this.inventionLevel = data.inventionLevel\n this.inventionTypes = data.inventionTypes\n this.inventionCount = data.inventionCount\n this.notes = data.notes\n }\n}\n","import { AlternateData } from '../api/alternate-data'\nimport { Sex } from '../api/sex'\nimport { Alignment } from '../api/alignment'\n\nconst ALIGNMENT_SORT: Record<string, number> = { H: 2, V: 1, P: 0 }\nconst SEX_SORT: Record<string, number> = { M: 1, F: 0 }\n\nexport class Alternates<T> {\n readonly #sortedValues: AlternateData<T>[] = []\n\n constructor(values: AlternateData<T>[]) {\n this.#sortedValues = values.sort()\n this.#sortedValues.sort((a, b) => this.#compareAlternates(a, b))\n }\n\n getValue(alignment?: Alignment | string, sex?: Sex | string): T | undefined {\n for (let index = this.#sortedValues.length; index--;) {\n const entry = this.#sortedValues[index]\n if ((entry.alignment === undefined || entry.alignment === alignment)\n && (entry.sex === undefined || entry.sex === sex)\n ) return entry.value\n }\n\n return this.default\n }\n\n /**\n * Get the default value for this list of alternates, the value with the highest priority and lowest specificity.\n */\n get default(): T | undefined {\n return this.#sortedValues[0]?.value\n }\n\n /**\n * Get the list of alternates sorted in canonical order (alignment then sex, low to high specificity).\n */\n get canonical(): AlternateData<T>[] {\n return this.#sortedValues\n }\n\n /**\n * Create a joined string from the alternate values in canonical order.\n * @param separator Separator to use. Default is ' / '\n */\n toString(separator: string): string {\n return this.canonical.map(x => x.value).join(separator)\n }\n\n #compareAlternates(a: AlternateData<T>, b: AlternateData<T>): number {\n const aSpecificity = (a.alignment ? 2 : 0) + (a.sex ? 1 : 0)\n const bSpecificity = (b.alignment ? 2 : 0) + (b.sex ? 1 : 0)\n if (aSpecificity !== bSpecificity) return aSpecificity - bSpecificity // Order first by least-specific\n\n const alignmentComparison = this.#compareAlignment(a.alignment, b.alignment) // Next by alignment\n if (alignmentComparison !== 0) return alignmentComparison\n\n const sexComparison = this.#compareSex(a.sex, b.sex) // Last by sex\n if (sexComparison !== 0) return sexComparison\n\n return String(a.value).localeCompare(String(b.value))\n }\n\n #compareAlignment(a: Alignment | string | undefined, b: Alignment | string | undefined): number {\n if (a === b) return 0\n if (a === undefined && b !== undefined) return -1\n if (b === undefined && a !== undefined) return 1\n\n const aSort = a === undefined ? -1 : ALIGNMENT_SORT[a] ?? -1 // Unknown values get -1 priority\n const bSort = b === undefined ? -1 : ALIGNMENT_SORT[b] ?? -1\n\n if (aSort !== bSort) return bSort - aSort\n\n // Unknown values (not in ALIGNMENT_SORT) are sorted alphabetically\n return a?.localeCompare(b ?? '') ?? 0\n }\n\n #compareSex(a?: Sex | string | undefined, b?: Sex | string | undefined): number {\n if (a === b) return 0\n if (a === undefined && b !== undefined) return -1\n if (b === undefined && a !== undefined) return 1\n\n const aSort = SEX_SORT[a ?? -1] ?? -1 // Unknown values get -1 priority\n const bSort = SEX_SORT[b ?? -1] ?? -1\n\n if (aSort !== bSort) return bSort - aSort\n\n // Unknown values (not in SEX_SORT) are sorted alphabetically\n return a?.localeCompare(b ?? '') ?? 0\n }\n}\n","import { BadgeType } from '../api/badge-type'\nimport { Alignment } from '../api/alignment'\nimport { Link } from '../api/link'\nimport { BadgeData } from '../api/badge-data'\nimport { BadgePartial } from './badge-partial'\nimport { Key } from './key'\nimport { Alternates } from './alternates'\n\nexport class Badge {\n readonly #partialsIndex: Record<string, BadgePartial> = {}\n\n /**\n * The database key for this badge.\n */\n readonly key: string\n\n /**\n * The type of badge.\n */\n readonly type: BadgeType | string\n\n /**\n * The name of this badge.\n *\n * May vary by character sex or alignment.\n */\n readonly name: Alternates<string>\n\n /**\n * The character alignments that this badge is available to.\n */\n readonly alignment: Alignment[]\n\n /**\n * The badge text as it appears in-game. May vary by character sex or alignment.\n */\n readonly badgeText: Alternates<string>\n\n /**\n * Description of how to acquire the badge.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly acquisition?: string\n\n /**\n * Absolute URL to this badge's icon.\n *\n * May vary by character sex or alignment.\n */\n readonly icon: Alternates<string>\n\n /**\n * Freeform notes or tips about the badge.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly notes?: string\n\n /**\n * List of external links for this Badge. Wiki, forums, etc.\n */\n readonly links?: Link[]\n\n /**\n * For exploration badges, the key of the {@link GameMap} that this badge is found on.\n */\n readonly mapKey?: string\n\n /**\n * For exploration badges, the `/loc` coordinates of the badge on the in-game map.\n */\n readonly loc?: [number, number, number]\n\n /**\n * For badges that appear on a Vidiot Map, the number or letter the badge appears as.\n */\n readonly vidiotMapKey?: string\n\n /**\n * ID used with the in-game `/settitle` command to apply the badge.\n */\n readonly setTitle?: {\n /**\n * `/settitle` id.\n */\n id?: number\n /**\n * `/settitle` id if different for praetorian characters.\n */\n praetorianId?: number\n }\n\n /**\n * A description of the effect the badge will have, such as a buff or granting a temporary power.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly effect?: string\n\n /**\n * A list of requirements for badges that have partial fulfilment steps, such as visiting plaques for history badges, or collecting other badges for meta-badges like accolades.\n */\n readonly partials?: BadgePartial[]\n\n /**\n * Some badges are not included in the badge total count... such as Flames of Prometheus, which can be removed by redeeming it for a Notice of the Well.\n */\n readonly ignoreInTotals: boolean\n\n constructor(data: BadgeData) {\n this.key = new Key(data.key).value\n this.type = data.type\n this.name = new Alternates(data.name)\n this.alignment = data.alignment\n this.badgeText = new Alternates(data.badgeText ?? [])\n this.acquisition = data.acquisition\n this.icon = new Alternates(data.icon ?? [])\n this.notes = data.notes\n this.links = data.links\n this.mapKey = data.mapKey\n this.loc = data.loc\n this.effect = data.effect\n this.vidiotMapKey = data.vidiotMapKey\n this.setTitle = data.setTitle\n this.ignoreInTotals = data.ignoreInTotals ?? false\n\n this.partials = data.partials?.map((data) => {\n if (this.#partialsIndex[data.key] !== undefined) throw new Error(`Duplicate badge partial key [${data.key}]`)\n const badge = new BadgePartial(data)\n this.#partialsIndex[badge.key] = badge\n return badge\n })\n }\n\n getPartial(key: string): BadgePartial {\n const result = this.#partialsIndex[key]\n if (result === undefined) throw new Error(`Unknown badge partial key [${key}]`)\n return result\n }\n}\n","import { VidiotMapPointOfInterestData } from '../api/vidiot-map-point-of-interest-data'\n\nexport class VidiotMapPointOfInterest {\n /**\n * The pixel-space position of the PoI on the map graphic.\n *\n * Screen-space, pixels from top-left `[0, 0]`.\n */\n readonly pos?: [number, number]\n\n /**\n * Freeform notes about the PoI.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly notes?: string\n\n /**\n * If the POI is a zone transfer, the map it transfers to.\n */\n readonly mapKey?: string\n\n /**\n * If the POI is a badge, the badge.\n */\n readonly badgeKey?: string\n\n /**\n * If the POI is a partial for a badge, the partial key.\n */\n readonly badgePartialKey?: string\n\n constructor(data: VidiotMapPointOfInterestData) {\n this.pos = data.pos\n this.notes = data.notes\n this.mapKey = data.mapKey\n this.badgeKey = data.badgeKey\n this.badgePartialKey = data.badgePartialKey\n }\n}\n","import { VidiotMapData } from '../api/vidiot-map-data'\nimport { VidiotMapPointOfInterest } from './vidiot-map-point-of-interest'\n\nexport class VidiotMap {\n /**\n * URL of the map image.\n */\n readonly imageUrl: string\n\n /**\n * Name to display for the Vidiot map.\n */\n readonly name?: string\n\n /**\n * List of Points of Interest labelled on the image.\n */\n readonly pointsOfInterest?: VidiotMapPointOfInterest[]\n\n constructor(data: VidiotMapData) {\n this.imageUrl = data.imageUrl\n this.name = data.name\n this.pointsOfInterest = data.pointsOfInterest?.map(data => new VidiotMapPointOfInterest(data))\n }\n}\n","import { VidiotMap } from './vidiot-map'\nimport { Link } from '../api/link'\nimport { GameMapData } from '../api/game-map-data'\nimport { Key } from './key'\n\nexport class GameMap {\n /**\n * The database key for this map.\n */\n readonly key: string\n\n /**\n * The name of the map as it appears in-game.\n */\n readonly name: string\n\n /**\n * List of external links for this Map. Wiki, forums, etc.\n */\n readonly links?: Link[]\n\n /**\n * List of Vidiot Map assets for this map.\n */\n readonly vidiotMaps?: VidiotMap[]\n\n constructor(data: GameMapData) {\n this.key = new Key(data.key).value\n this.name = data.name\n this.links = data.links\n this.vidiotMaps = data.vidiotMaps?.map(data => new VidiotMap(data))\n }\n}\n","import { ContentBundle } from '../api/content-bundle'\nimport { Change } from '../api/change'\nimport { Link } from '../api/link'\n\nexport class BundleMetadata {\n /**\n * Name of the server group.\n */\n readonly name: string\n\n /**\n * Description of the server group.\n *\n * Supports {@link https://www.markdownguide.org/|Markdown} format.\n */\n readonly description?: string\n\n /**\n * Repository where the db content package is maintained.\n */\n readonly repository?: string\n\n /**\n * List of external links for this Server Group. Wiki, forums, etc.\n */\n readonly links: Link[]\n\n /**\n * Change log for this data package.\n */\n readonly changelog: Change[]\n\n constructor(bundle: ContentBundle) {\n this.name = bundle.name\n this.description = bundle.description\n this.repository = bundle.repository\n this.links = bundle.links ?? []\n this.changelog = bundle.changelog ?? []\n }\n}\n","import { ContentBundle } from '../api/content-bundle'\nimport { Archetype } from './archetype'\nimport { GameMap } from './game-map'\nimport { Badge } from './badge'\nimport { BundleMetadata } from './bundle-metadata'\n\nexport class CohContentDatabase {\n readonly #archetypeIndex: Record<string, Archetype> = {}\n readonly #mapIndex: Record<string, GameMap> = {}\n readonly #badgeIndex: Record<string, Badge> = {}\n\n /**\n * Metadata about the content bundle.\n */\n readonly metadata: BundleMetadata\n\n /**\n * List of the game server names in this server group.\n * Torchbearer, Excelsior, etc.\n */\n readonly servers: string[]\n\n /**\n * List of archetypes available in this server group.\n */\n readonly archetypes: Archetype[]\n\n /**\n * List of game maps supported by this server group.\n */\n readonly maps: GameMap[]\n\n /**\n * List of badges available on this server group.\n */\n readonly badges: Badge[]\n\n /**\n * Initialize the database with a content bundle.\n * @param bundle The data to load.\n */\n constructor(bundle: ContentBundle) {\n this.metadata = new BundleMetadata(bundle)\n this.servers = bundle.servers ?? []\n this.archetypes = bundle.archetypes?.map((data) => {\n if (this.#archetypeIndex[data.key] !== undefined) throw new Error(`Duplicate archetype key [${data.key}]`)\n const archetype = new Archetype(data)\n this.#archetypeIndex[archetype.key] = archetype\n return archetype\n }) ?? []\n this.maps = bundle.maps?.map((data) => {\n if (this.#mapIndex[data.key] !== undefined) throw new Error(`Duplicate map key [${data.key}]`)\n const map = new GameMap(data)\n this.#mapIndex[map.key] = map\n return map\n }) ?? []\n this.badges = bundle.badges?.map((data) => {\n if (this.#badgeIndex[data.key] !== undefined) throw new Error(`Duplicate badge key [${data.key}]`)\n const badge = new Badge(data)\n this.#badgeIndex[badge.key] = badge\n return badge\n }) ?? []\n }\n\n getArchetype(key: string): Archetype {\n const result = this.#archetypeIndex[key]\n if (result === undefined) throw new Error(`Unknown archetype key [${key}]`)\n return result\n }\n\n getMap(key: string): GameMap {\n const result = this.#mapIndex[key]\n if (result === undefined) throw new Error(`Unknown map key [${key}]`)\n return result\n }\n\n getBadge(key: string): Badge {\n const result = this.#badgeIndex[key]\n if (result === undefined) throw new Error(`Unknown badge key [${key}]`)\n return result\n }\n}\n","import { Change } from './api/change'\n\nexport const CHANGELOG: Change[] = [\n {\n version: '2.0.0',\n date: new Date('2025-03-12'),\n description: ''\n + '* Replaced redundant interfaces with their concrete equivalents.\\n'\n + `* Server groups are now referred to as 'forks'.\\n`\n + '* Replaced enums with extensible union types; Forks with new badge types, enhancement types, etc. can now extend them locally.\\n'\n + '* `IServerGroupData` is now `ContentBundle` and each database instance is now designed to accept only a single server group.\\n'\n + '* Removed the `serverGroup` property from entities to simplify the object tree given that only a single context can exist per db now.\\n'\n + '* Standardized pluralization of some field names (name, icon).\\n'\n + '* Combined `settitle` ids into a single tuple field.\\n'\n + '* Change from GNU to The Unlicense.\\n'\n + '* Removed dependency on lodash. There are now no third-party runtime dependencies.\\n'\n + '* Moved from webpack to rollup for packaging.\\n'\n + '* Add eslint for linting.\\n'\n + '* Add jest for unit tests.\\n'\n + '* Added GitHub Actions for CI.\\n',\n },\n]\n","/**\n * Create a reference string that can be used in most text strings to display a link to the given badge.\n * @param target The badge or badge key to target.\n */\nexport function createBadgeReference(target: string | { key: string }): string {\n const key = typeof target === 'string' ? target : target.key\n return `[badge:${key}]`\n}\n\n/**\n * Create a reference string that can be used in most text strings to display a link to the given map.\n * @param target The {@link GameMap} or map key to target.\n */\nexport function createMapReference(target: string | { key: string }): string {\n const key = typeof target === 'string' ? target : target.key\n return `[map:${key}]`\n}\n"],"names":["__privateAdd","__privateMethod","__privateSet","__privateGet","__publicField","data"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -5,23 +5,16 @@ import { Change } from './change'
|
|
|
5
5
|
import { Link } from './link'
|
|
6
6
|
|
|
7
7
|
/**
|
|
8
|
-
* A
|
|
8
|
+
* A content bundle holds the data that makes up one forked instance of the game since the original sunset, such as Homecoming (https://forums.homecomingservers.com/).
|
|
9
9
|
*/
|
|
10
|
-
export interface
|
|
10
|
+
export interface ContentBundle {
|
|
11
11
|
/**
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
* Keys can only contain lowercase letters, numbers and hyphens (`-`).
|
|
15
|
-
*/
|
|
16
|
-
readonly key: string
|
|
17
|
-
|
|
18
|
-
/**
|
|
19
|
-
* Name of the server group.
|
|
12
|
+
* Name of the fork this bundle contains data for.
|
|
20
13
|
*/
|
|
21
14
|
readonly name: string
|
|
22
15
|
|
|
23
16
|
/**
|
|
24
|
-
* Description of the
|
|
17
|
+
* Description of the fork.
|
|
25
18
|
*
|
|
26
19
|
* Supports {@link https://www.markdownguide.org/|Markdown} format.
|
|
27
20
|
*/
|
|
@@ -38,23 +31,23 @@ export interface ServerGroupData {
|
|
|
38
31
|
readonly links?: Link[]
|
|
39
32
|
|
|
40
33
|
/**
|
|
41
|
-
* List of the game server names in this
|
|
34
|
+
* List of the game server names in this fork.
|
|
42
35
|
* Torchbearer, Excelsior, etc.
|
|
43
36
|
*/
|
|
44
37
|
readonly servers?: string[]
|
|
45
38
|
|
|
46
39
|
/**
|
|
47
|
-
* List of archetypes available in this
|
|
40
|
+
* List of archetypes available in this fork.
|
|
48
41
|
*/
|
|
49
42
|
readonly archetypes?: ArchetypeData[]
|
|
50
43
|
|
|
51
44
|
/**
|
|
52
|
-
* List of game maps supported by this
|
|
45
|
+
* List of game maps supported by this fork.
|
|
53
46
|
*/
|
|
54
47
|
readonly maps?: GameMapData[]
|
|
55
48
|
|
|
56
49
|
/**
|
|
57
|
-
* List of badges available on this
|
|
50
|
+
* List of badges available on this fork.
|
|
58
51
|
*/
|
|
59
52
|
readonly badges?: BadgeData[]
|
|
60
53
|
|
package/src/main/changelog.ts
CHANGED
|
@@ -6,8 +6,10 @@ export const CHANGELOG: Change[] = [
|
|
|
6
6
|
date: new Date('2025-03-12'),
|
|
7
7
|
description: ''
|
|
8
8
|
+ '* Replaced redundant interfaces with their concrete equivalents.\n'
|
|
9
|
-
+
|
|
10
|
-
+ '*
|
|
9
|
+
+ `* Server groups are now referred to as 'forks'.\n`
|
|
10
|
+
+ '* Replaced enums with extensible union types; Forks with new badge types, enhancement types, etc. can now extend them locally.\n'
|
|
11
|
+
+ '* `IServerGroupData` is now `ContentBundle` and each database instance is now designed to accept only a single server group.\n'
|
|
12
|
+
+ '* Removed the `serverGroup` property from entities to simplify the object tree given that only a single context can exist per db now.\n'
|
|
11
13
|
+ '* Standardized pluralization of some field names (name, icon).\n'
|
|
12
14
|
+ '* Combined `settitle` ids into a single tuple field.\n'
|
|
13
15
|
+ '* Change from GNU to The Unlicense.\n'
|
|
@@ -20,7 +20,8 @@ export class Alternates<T> {
|
|
|
20
20
|
&& (entry.sex === undefined || entry.sex === sex)
|
|
21
21
|
) return entry.value
|
|
22
22
|
}
|
|
23
|
-
|
|
23
|
+
|
|
24
|
+
return this.default
|
|
24
25
|
}
|
|
25
26
|
|
|
26
27
|
/**
|
|
@@ -37,6 +38,14 @@ export class Alternates<T> {
|
|
|
37
38
|
return this.#sortedValues
|
|
38
39
|
}
|
|
39
40
|
|
|
41
|
+
/**
|
|
42
|
+
* Create a joined string from the alternate values in canonical order.
|
|
43
|
+
* @param separator Separator to use. Default is ' / '
|
|
44
|
+
*/
|
|
45
|
+
toString(separator: string): string {
|
|
46
|
+
return this.canonical.map(x => x.value).join(separator)
|
|
47
|
+
}
|
|
48
|
+
|
|
40
49
|
#compareAlternates(a: AlternateData<T>, b: AlternateData<T>): number {
|
|
41
50
|
const aSpecificity = (a.alignment ? 2 : 0) + (a.sex ? 1 : 0)
|
|
42
51
|
const bSpecificity = (b.alignment ? 2 : 0) + (b.sex ? 1 : 0)
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { ContentBundle } from '../api/content-bundle'
|
|
2
|
+
import { Change } from '../api/change'
|
|
3
|
+
import { Link } from '../api/link'
|
|
4
|
+
|
|
5
|
+
export class BundleMetadata {
|
|
6
|
+
/**
|
|
7
|
+
* Name of the server group.
|
|
8
|
+
*/
|
|
9
|
+
readonly name: string
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Description of the server group.
|
|
13
|
+
*
|
|
14
|
+
* Supports {@link https://www.markdownguide.org/|Markdown} format.
|
|
15
|
+
*/
|
|
16
|
+
readonly description?: string
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* Repository where the db content package is maintained.
|
|
20
|
+
*/
|
|
21
|
+
readonly repository?: string
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* List of external links for this Server Group. Wiki, forums, etc.
|
|
25
|
+
*/
|
|
26
|
+
readonly links: Link[]
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Change log for this data package.
|
|
30
|
+
*/
|
|
31
|
+
readonly changelog: Change[]
|
|
32
|
+
|
|
33
|
+
constructor(bundle: ContentBundle) {
|
|
34
|
+
this.name = bundle.name
|
|
35
|
+
this.description = bundle.description
|
|
36
|
+
this.repository = bundle.repository
|
|
37
|
+
this.links = bundle.links ?? []
|
|
38
|
+
this.changelog = bundle.changelog ?? []
|
|
39
|
+
}
|
|
40
|
+
}
|
|
@@ -1,29 +1,82 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
1
|
+
import { ContentBundle } from '../api/content-bundle'
|
|
2
|
+
import { Archetype } from './archetype'
|
|
3
|
+
import { GameMap } from './game-map'
|
|
4
|
+
import { Badge } from './badge'
|
|
5
|
+
import { BundleMetadata } from './bundle-metadata'
|
|
3
6
|
|
|
4
7
|
export class CohContentDatabase {
|
|
5
|
-
readonly #
|
|
8
|
+
readonly #archetypeIndex: Record<string, Archetype> = {}
|
|
9
|
+
readonly #mapIndex: Record<string, GameMap> = {}
|
|
10
|
+
readonly #badgeIndex: Record<string, Badge> = {}
|
|
6
11
|
|
|
7
12
|
/**
|
|
8
|
-
*
|
|
9
|
-
* @param data The data to load.
|
|
13
|
+
* Metadata about the content bundle.
|
|
10
14
|
*/
|
|
11
|
-
|
|
12
|
-
this.#serverGroups[data.key] = new ServerGroup(data)
|
|
13
|
-
}
|
|
15
|
+
readonly metadata: BundleMetadata
|
|
14
16
|
|
|
15
17
|
/**
|
|
16
|
-
*
|
|
18
|
+
* List of the game server names in this server group.
|
|
19
|
+
* Torchbearer, Excelsior, etc.
|
|
17
20
|
*/
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
+
readonly servers: string[]
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* List of archetypes available in this server group.
|
|
25
|
+
*/
|
|
26
|
+
readonly archetypes: Archetype[]
|
|
21
27
|
|
|
22
28
|
/**
|
|
23
|
-
*
|
|
24
|
-
* @param serverGroupKey The key.
|
|
29
|
+
* List of game maps supported by this server group.
|
|
25
30
|
*/
|
|
26
|
-
|
|
27
|
-
|
|
31
|
+
readonly maps: GameMap[]
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* List of badges available on this server group.
|
|
35
|
+
*/
|
|
36
|
+
readonly badges: Badge[]
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Initialize the database with a content bundle.
|
|
40
|
+
* @param bundle The data to load.
|
|
41
|
+
*/
|
|
42
|
+
constructor(bundle: ContentBundle) {
|
|
43
|
+
this.metadata = new BundleMetadata(bundle)
|
|
44
|
+
this.servers = bundle.servers ?? []
|
|
45
|
+
this.archetypes = bundle.archetypes?.map((data) => {
|
|
46
|
+
if (this.#archetypeIndex[data.key] !== undefined) throw new Error(`Duplicate archetype key [${data.key}]`)
|
|
47
|
+
const archetype = new Archetype(data)
|
|
48
|
+
this.#archetypeIndex[archetype.key] = archetype
|
|
49
|
+
return archetype
|
|
50
|
+
}) ?? []
|
|
51
|
+
this.maps = bundle.maps?.map((data) => {
|
|
52
|
+
if (this.#mapIndex[data.key] !== undefined) throw new Error(`Duplicate map key [${data.key}]`)
|
|
53
|
+
const map = new GameMap(data)
|
|
54
|
+
this.#mapIndex[map.key] = map
|
|
55
|
+
return map
|
|
56
|
+
}) ?? []
|
|
57
|
+
this.badges = bundle.badges?.map((data) => {
|
|
58
|
+
if (this.#badgeIndex[data.key] !== undefined) throw new Error(`Duplicate badge key [${data.key}]`)
|
|
59
|
+
const badge = new Badge(data)
|
|
60
|
+
this.#badgeIndex[badge.key] = badge
|
|
61
|
+
return badge
|
|
62
|
+
}) ?? []
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
getArchetype(key: string): Archetype {
|
|
66
|
+
const result = this.#archetypeIndex[key]
|
|
67
|
+
if (result === undefined) throw new Error(`Unknown archetype key [${key}]`)
|
|
68
|
+
return result
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
getMap(key: string): GameMap {
|
|
72
|
+
const result = this.#mapIndex[key]
|
|
73
|
+
if (result === undefined) throw new Error(`Unknown map key [${key}]`)
|
|
74
|
+
return result
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
getBadge(key: string): Badge {
|
|
78
|
+
const result = this.#badgeIndex[key]
|
|
79
|
+
if (result === undefined) throw new Error(`Unknown badge key [${key}]`)
|
|
80
|
+
return result
|
|
28
81
|
}
|
|
29
82
|
}
|
package/src/main/index.ts
CHANGED
|
@@ -11,7 +11,7 @@ export * from './api/enhancement-category'
|
|
|
11
11
|
export * from './api/game-map-data'
|
|
12
12
|
export * from './api/link'
|
|
13
13
|
export * from './api/plaque-type'
|
|
14
|
-
export * from './api/
|
|
14
|
+
export * from './api/content-bundle'
|
|
15
15
|
export * from './api/sex'
|
|
16
16
|
export * from './api/vidiot-map-data'
|
|
17
17
|
export * from './api/vidiot-map-point-of-interest-data'
|
|
@@ -23,7 +23,7 @@ export * from './db/badge-partial'
|
|
|
23
23
|
export * from './db/coh-content-database'
|
|
24
24
|
export * from './db/game-map'
|
|
25
25
|
export * from './db/key'
|
|
26
|
-
export * from './db/
|
|
26
|
+
export * from './db/bundle-metadata'
|
|
27
27
|
export * from './db/vidiot-map'
|
|
28
28
|
export * from './db/vidiot-map-point-of-interest'
|
|
29
29
|
|
|
@@ -1,12 +1,11 @@
|
|
|
1
1
|
import { defineFixture } from 'efate'
|
|
2
|
-
import { Change,
|
|
2
|
+
import { Change, ContentBundle } from '../../main'
|
|
3
3
|
import { archetypeDataFixture } from './archetype-data.fixture'
|
|
4
4
|
import { gameMapDataFixture } from './game-map-data.fixture'
|
|
5
5
|
import { badgeDataFixture } from './badge-data.fixture'
|
|
6
6
|
|
|
7
|
-
export const
|
|
8
|
-
t.
|
|
9
|
-
t.name.as(index => `Server Group ${index}`)
|
|
7
|
+
export const contentBundleFixture = defineFixture<ContentBundle>((t) => {
|
|
8
|
+
t.name.as(index => `Bundle ${index}`)
|
|
10
9
|
t.description?.asLoremIpsum()
|
|
11
10
|
t.repository?.as(index => `https://nouri.org?id=${index}`)
|
|
12
11
|
t.servers?.asArray()
|
|
@@ -1,15 +1,14 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { ContentBundle } from '../../main'
|
|
2
2
|
import { TEST_BADGE } from './badge-data.test'
|
|
3
3
|
|
|
4
4
|
// If you change this test, update the example in the README as well
|
|
5
|
-
export const
|
|
6
|
-
|
|
7
|
-
name: 'My Server Group',
|
|
5
|
+
export const TEST_BUNDLE: ContentBundle = {
|
|
6
|
+
name: 'My Content Bundle',
|
|
8
7
|
badges: [TEST_BADGE],
|
|
9
8
|
}
|
|
10
9
|
|
|
11
10
|
describe('ServerGroupData', () => {
|
|
12
11
|
test('should be a usable interface', () => {
|
|
13
|
-
expect(
|
|
12
|
+
expect(TEST_BUNDLE).not.toBeUndefined()
|
|
14
13
|
})
|
|
15
14
|
})
|
|
@@ -2,7 +2,7 @@ import { CHANGELOG } from '../main'
|
|
|
2
2
|
|
|
3
3
|
describe('CHANGELOG', () => {
|
|
4
4
|
test('should be extant', () => {
|
|
5
|
-
expect(CHANGELOG).not.
|
|
5
|
+
expect(CHANGELOG).not.toBeUndefined()
|
|
6
6
|
expect(CHANGELOG).not.toBeUndefined()
|
|
7
7
|
})
|
|
8
8
|
|
|
@@ -22,14 +22,14 @@ describe('CHANGELOG', () => {
|
|
|
22
22
|
test('should have dates', () => {
|
|
23
23
|
for (const change of CHANGELOG) {
|
|
24
24
|
const date = change.date
|
|
25
|
-
expect(date).not.
|
|
25
|
+
expect(date).not.toBeUndefined()
|
|
26
26
|
expect(date).not.toBeUndefined()
|
|
27
27
|
}
|
|
28
28
|
})
|
|
29
29
|
|
|
30
30
|
test('should have descriptions', () => {
|
|
31
31
|
for (const change of CHANGELOG) {
|
|
32
|
-
expect(change).not.
|
|
32
|
+
expect(change).not.toBeUndefined()
|
|
33
33
|
expect(change).not.toBeUndefined()
|
|
34
34
|
}
|
|
35
35
|
})
|
|
@@ -58,13 +58,22 @@ describe(Alternates.name, () => {
|
|
|
58
58
|
]).getValue('V', 'M')).toBe('Male Villain')
|
|
59
59
|
})
|
|
60
60
|
|
|
61
|
-
test('should return
|
|
61
|
+
test('should return the lowest canonical value if there is no default', () => {
|
|
62
62
|
expect(new Alternates([
|
|
63
|
+
{ alignment: 'H', value: 'Hero' },
|
|
63
64
|
{ sex: 'M', value: 'Male' },
|
|
65
|
+
{ alignment: 'P', sex: 'F', value: 'Praetorian Female' },
|
|
66
|
+
{ alignment: 'V', sex: 'M', value: 'Male Villain' },
|
|
67
|
+
]).getValue()).toBe('Male')
|
|
68
|
+
})
|
|
69
|
+
|
|
70
|
+
test('should return the lowest canonical value if a specific is requested that does not exist', () => {
|
|
71
|
+
expect(new Alternates([
|
|
64
72
|
{ alignment: 'H', value: 'Hero' },
|
|
73
|
+
{ alignment: 'V', value: 'Villain' },
|
|
65
74
|
{ alignment: 'V', sex: 'M', value: 'Male Villain' },
|
|
66
75
|
{ alignment: 'P', sex: 'F', value: 'Praetorian Female' },
|
|
67
|
-
]).getValue()).
|
|
76
|
+
]).getValue(undefined, 'F')).toBe('Hero')
|
|
68
77
|
})
|
|
69
78
|
})
|
|
70
79
|
|
|
@@ -220,4 +229,14 @@ describe(Alternates.name, () => {
|
|
|
220
229
|
])
|
|
221
230
|
})
|
|
222
231
|
})
|
|
232
|
+
|
|
233
|
+
describe('toString', () => {
|
|
234
|
+
test('should create a string separated by the separator', () => {
|
|
235
|
+
expect(new Alternates([
|
|
236
|
+
{ sex: 'A', value: 'A' },
|
|
237
|
+
{ sex: 'B', value: 'B' },
|
|
238
|
+
{ sex: 'C', value: 'C' },
|
|
239
|
+
]).toString(', ')).toBe('A, B, C')
|
|
240
|
+
})
|
|
241
|
+
})
|
|
223
242
|
})
|
|
@@ -27,7 +27,7 @@ describe(Badge.name, () => {
|
|
|
27
27
|
partials: [badgePartialDataFixture.create({ key: 'foo' })],
|
|
28
28
|
})
|
|
29
29
|
|
|
30
|
-
expect(new Badge(data).getPartial('foo')).not.
|
|
30
|
+
expect(new Badge(data).getPartial('foo')).not.toBeUndefined()
|
|
31
31
|
})
|
|
32
32
|
|
|
33
33
|
test(`should throw error for unknown partial`, () => {
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
import { BundleMetadata } from '../../main'
|
|
2
|
+
import { contentBundleFixture } from '../api/content-bundle.fixture'
|
|
3
|
+
|
|
4
|
+
describe(BundleMetadata.name, () => {
|
|
5
|
+
describe('Constructor', () => {
|
|
6
|
+
test(`should accept the test fixture`, () => {
|
|
7
|
+
new BundleMetadata(contentBundleFixture.create())
|
|
8
|
+
})
|
|
9
|
+
})
|
|
10
|
+
|
|
11
|
+
describe('name', () => {
|
|
12
|
+
test(`should be read from the bundle`, () => {
|
|
13
|
+
const bundle = new BundleMetadata(contentBundleFixture.create({ name: 'foo' }))
|
|
14
|
+
expect(bundle.name).toBe('foo')
|
|
15
|
+
})
|
|
16
|
+
})
|
|
17
|
+
|
|
18
|
+
describe('description', () => {
|
|
19
|
+
test(`should be read from the bundle`, () => {
|
|
20
|
+
const bundle = new BundleMetadata(contentBundleFixture.create({ description: 'foo' }))
|
|
21
|
+
expect(bundle.description).toBe('foo')
|
|
22
|
+
})
|
|
23
|
+
|
|
24
|
+
test(`should be optional`, () => {
|
|
25
|
+
const bundle = new BundleMetadata(contentBundleFixture.omit('description').create())
|
|
26
|
+
expect(bundle.description).toBeUndefined()
|
|
27
|
+
})
|
|
28
|
+
})
|
|
29
|
+
|
|
30
|
+
describe('repository', () => {
|
|
31
|
+
test(`should be read from the bundle`, () => {
|
|
32
|
+
const bundle = new BundleMetadata(contentBundleFixture.create({ repository: 'foo' }))
|
|
33
|
+
expect(bundle.repository).toBe('foo')
|
|
34
|
+
})
|
|
35
|
+
|
|
36
|
+
test(`should be optional`, () => {
|
|
37
|
+
const bundle = new BundleMetadata(contentBundleFixture.omit('repository').create())
|
|
38
|
+
expect(bundle.repository).toBeUndefined()
|
|
39
|
+
})
|
|
40
|
+
})
|
|
41
|
+
|
|
42
|
+
describe('links', () => {
|
|
43
|
+
test(`should be read from the bundle`, () => {
|
|
44
|
+
const bundle = new BundleMetadata(contentBundleFixture.create({ links: [{ title: 'foo', href: 'bar' }] }))
|
|
45
|
+
expect(bundle.links).toStrictEqual([{ title: 'foo', href: 'bar' }])
|
|
46
|
+
})
|
|
47
|
+
|
|
48
|
+
test(`should be optional`, () => {
|
|
49
|
+
const bundle = new BundleMetadata(contentBundleFixture.omit('links').create())
|
|
50
|
+
expect(bundle.links).toHaveLength(0)
|
|
51
|
+
})
|
|
52
|
+
})
|
|
53
|
+
|
|
54
|
+
describe('changelog', () => {
|
|
55
|
+
test(`should be read from the bundle`, () => {
|
|
56
|
+
const bundle = new BundleMetadata(contentBundleFixture.create({
|
|
57
|
+
changelog: [{ version: 'foo', date: new Date('2025-03-12'), description: 'bar' }],
|
|
58
|
+
}))
|
|
59
|
+
expect(bundle.changelog).toStrictEqual([{ version: 'foo', date: new Date('2025-03-12'), description: 'bar' }])
|
|
60
|
+
})
|
|
61
|
+
|
|
62
|
+
test(`should be optional`, () => {
|
|
63
|
+
const bundle = new BundleMetadata(contentBundleFixture.omit('changelog').create())
|
|
64
|
+
expect(bundle.changelog).toHaveLength(0)
|
|
65
|
+
})
|
|
66
|
+
})
|
|
67
|
+
})
|