phantasmal-world/static/updateEphineaData.ts

891 lines
28 KiB
TypeScript

import fs from 'fs';
import { ArrayBufferCursor } from '../src/bin-data/ArrayBufferCursor';
import { parseItemPmt, ItemPmt } from '../src/bin-data/parsing/itempmt';
import { parseUnitxt, Unitxt } from '../src/bin-data/parsing/unitxt';
import { Difficulties, Difficulty, Episode, Episodes, NpcType, SectionId, SectionIds } from '../src/domain';
import { NpcTypes } from '../src/domain/NpcType';
import { BoxDropDto, EnemyDropDto, ItemTypeDto } from '../src/dto';
import { updateDropsFromWebsite } from './updateDropsEphinea';
/**
* Used by scripts.
*/
const RESOURCE_DIR = './static/resources/ephinea';
/**
* Used by production code.
*/
const PUBLIC_DIR = './public';
update().catch(e => console.error(e));
/**
* ItemPMT.bin and ItemPT.gsl comes from stock Tethealla. ItemPT.gsl is not used at the moment.
* unitxt_j.prs comes from the Ephinea client.
* TODO: manual fixes:
* - Clio is equipable by HUnewearls
* - Red Ring has a requirement of 180, not 108
*/
async function update() {
const unitxt = await loadUnitxt();
const itemNames = unitxt[1];
const items = await updateItems(itemNames);
await updateDropsFromWebsite(items);
// Use this if we ever get the Ephinea drop files.
// const itemPt = await loadItemPt();
// await updateDrops(itemPt);
}
async function loadUnitxt(): Promise<Unitxt> {
const buf = await fs.promises.readFile(
`${RESOURCE_DIR}/client/data/unitxt_j.prs`
);
const unitxt = parseUnitxt(new ArrayBufferCursor(buf.buffer, true));
// Strip custom Ephinea items until we have the Ephinea ItemPMT.bin.
unitxt[1].splice(177, 50);
unitxt[1].splice(639, 59);
return unitxt;
}
async function updateItems(itemNames: Array<string>): Promise<ItemTypeDto[]> {
const buf = await fs.promises.readFile(
`${RESOURCE_DIR}/ship-config/param/ItemPMT.bin`
);
const itemPmt = parseItemPmt(new ArrayBufferCursor(buf.buffer, true));
const itemTypes = new Array<ItemTypeDto>();
const ids = new Set<number>();
itemPmt.weapons.forEach((category, categoryI) => {
category.forEach((weapon, i) => {
const id = (categoryI << 8) + i;
if (!ids.has(id)) {
ids.add(id);
itemTypes.push({
class: 'weapon',
id,
name: itemNames[weapon.id],
minAtp: weapon.minAtp,
maxAtp: weapon.maxAtp,
ata: weapon.ata,
maxGrind: weapon.maxGrind,
requiredAtp: weapon.reqAtp,
});
}
});
});
itemPmt.armors.forEach((armor, i) => {
const id = 0x10100 + i;
if (!ids.has(id)) {
ids.add(id);
const stats = getStatBoosts(itemPmt, armor.statBoost);
stats.minEvp += armor.evp;
stats.minDfp += armor.dfp;
itemTypes.push({
class: 'armor',
id,
name: itemNames[armor.id],
...stats,
maxEvp: stats.minEvp + armor.evpRange,
maxDfp: stats.minDfp + armor.dfpRange,
});
}
});
itemPmt.shields.forEach((shield, i) => {
const id = 0x10200 + i;
if (!ids.has(id)) {
ids.add(id);
const stats = getStatBoosts(itemPmt, shield.statBoost);
stats.minEvp += shield.evp;
stats.minDfp += shield.dfp;
itemTypes.push({
class: 'shield',
id,
name: itemNames[shield.id],
...stats,
maxEvp: stats.minEvp + shield.evpRange,
maxDfp: stats.minDfp + shield.dfpRange,
});
}
});
itemPmt.units.forEach((unit, i) => {
const id = 0x10300 + i;
if (!ids.has(id)) {
ids.add(id);
itemTypes.push({
class: 'unit',
id,
name: itemNames[unit.id],
});
}
});
itemPmt.tools.forEach((category, categoryI) => {
category.forEach((tool, i) => {
const id = (0x30000 | (categoryI << 8)) + i;
if (!ids.has(id)) {
ids.add(id);
itemTypes.push({
class: 'tool',
id,
name: itemNames[tool.id],
});
}
});
});
await fs.promises.writeFile(
`${PUBLIC_DIR}/itemTypes.ephinea.json`,
JSON.stringify(itemTypes, null, 4)
);
return itemTypes;
}
async function updateDrops(itemPt: ItemPt) {
const enemyDrops = new Array<EnemyDropDto>();
for (const diff of Difficulties) {
for (const ep of Episodes) {
for (const sid of SectionIds) {
enemyDrops.push(...await loadEnemyDrops(itemPt, diff, ep, sid));
}
}
}
await fs.promises.writeFile(
`${PUBLIC_DIR}/enemyDrops.ephinea.json`,
JSON.stringify(enemyDrops, null, 4)
);
const boxDrops = new Array<BoxDropDto>();
for (const diff of Difficulties) {
for (const ep of Episodes) {
for (const sid of SectionIds) {
boxDrops.push(...await loadBoxDrops(diff, ep, sid));
}
}
}
await fs.promises.writeFile(
`${PUBLIC_DIR}/boxDrops.ephinea.json`,
JSON.stringify(boxDrops, null, 4)
);
}
type ItemP = {
darTable: Map<NpcType, number>
}
type ItemPt = Array<Array<Array<ItemP>>>
async function loadItemPt(): Promise<ItemPt> {
const table: ItemPt = [];
const buf = await fs.promises.readFile(
`${RESOURCE_DIR}/ship-config/param/ItemPT.gsl`
);
const cursor = new ArrayBufferCursor(buf.buffer, false);
cursor.seek(0x3000);
// ItemPT.gsl was extracted from PSO for XBox, so it only contains data for ep. I and II.
// Episode IV data is based on the ep. I and II data.
for (const episode of [Episode.I, Episode.II]) {
table[episode] = [];
for (const diff of Difficulties) {
table[episode][diff] = [];
for (const sid of SectionIds) {
const darTable = new Map<NpcType, number>();
table[episode][diff][sid] = {
darTable
};
const startPos = cursor.position;
cursor.seek(1608);
const enemyDar = cursor.u8Array(100);
for (const npc of NpcTypes) {
if (npc.episode !== episode) continue;
switch (npc) {
case NpcType.Dragon:
case NpcType.DeRolLe:
case NpcType.VolOpt:
case NpcType.DarkFalz:
case NpcType.BarbaRay:
case NpcType.GolDragon:
case NpcType.GalGryphon:
case NpcType.OlgaFlow:
case NpcType.SaintMilion:
case NpcType.Shambertin:
case NpcType.Kondrieu:
darTable.set(npc, 1);
continue;
}
const ptIndex = npcTypeToPtIndex(npc);
if (ptIndex != null) {
darTable.set(npc, enemyDar[ptIndex] / 100);
}
}
cursor.seekStart(startPos + 0x1000);
}
}
}
table[Episode.IV] = [];
for (const diff of Difficulties) {
table[Episode.IV][diff] = [];
for (const sid of SectionIds) {
const darTable = new Map<NpcType, number>();
table[Episode.IV][diff][sid] = {
darTable
};
for (const npc of NpcTypes) {
if (npc.episode !== Episode.IV) continue;
switch (npc) {
case NpcType.SandRappy:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.RagRappy)!
);
break;
case NpcType.DelRappy:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.AlRappy)!
);
break;
case NpcType.Astark:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.Hildebear)!
);
break;
case NpcType.SatelliteLizard:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.SavageWolf)!
);
break;
case NpcType.Yowie:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.BarbarousWolf)!
);
break;
case NpcType.MerissaA:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.PofuillySlime)!
);
break;
case NpcType.MerissaAA:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.PouillySlime)!
);
break;
case NpcType.Girtablulu:
darTable.set(
npc,
table[Episode.II][diff][sid].darTable.get(NpcType.Mericarol)!
);
break;
case NpcType.Zu:
darTable.set(
npc,
table[Episode.II][diff][sid].darTable.get(NpcType.GiGue)!
);
break;
case NpcType.Pazuzu:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.Hildeblue)!
);
break;
case NpcType.Boota:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.Booma)!
);
break;
case NpcType.ZeBoota:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.Gobooma)!
);
break;
case NpcType.BaBoota:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.Gigobooma)!
);
break;
case NpcType.Dorphon:
darTable.set(
npc,
table[Episode.II][diff][sid].darTable.get(NpcType.Delbiter)!
);
break;
case NpcType.DorphonEclair:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.Hildeblue)!
);
break;
case NpcType.Goran:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.Dimenian)!
);
break;
case NpcType.PyroGoran:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.LaDimenian)!
);
break;
case NpcType.GoranDetonator:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.SoDimenian)!
);
break;
case NpcType.SaintMilion:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.DarkFalz)!
);
break;
case NpcType.Shambertin:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.DarkFalz)!
);
break;
case NpcType.Kondrieu:
darTable.set(
npc,
table[Episode.I][diff][sid].darTable.get(NpcType.DarkFalz)!
);
break;
}
}
}
}
return table;
}
async function loadEnemyDrops(
itemPt: ItemPt,
difficulty: Difficulty,
episode: Episode,
sectionId: SectionId
): Promise<Array<EnemyDropDto>> {
const drops: Array<EnemyDropDto> = [];
const dropsBuf = await fs.promises.readFile(
`${RESOURCE_DIR}/login-config/drop/ep${episode}_mob_${difficulty}_${sectionId}.txt`
);
let lineNo = 0;
let prevLine = '';
for (const line of dropsBuf.toString('utf8').split('\n')) {
const trimmed = line.trim();
if (trimmed.startsWith('#')) continue;
if (lineNo % 2 == 1) {
let enemy = getEnemyType(episode, Math.floor(lineNo / 2));
if (enemy) {
const rareRate = expandDropRate(parseInt(prevLine, 10));
const itemTypeId = parseInt(trimmed, 16);
const dar = itemPt[episode][difficulty][sectionId].darTable.get(enemy);
if (dar == null) {
console.error(`No DAR found for ${enemy.name}.`);
} else if (rareRate > 0 && itemTypeId) {
drops.push({
difficulty: Difficulty[difficulty],
episode: episode,
sectionId: SectionId[sectionId],
enemy: enemy.code,
itemTypeId,
dropRate: dar,
rareRate,
});
}
}
}
prevLine = trimmed;
lineNo++;
}
return drops;
}
async function loadBoxDrops(
difficulty: Difficulty,
episode: Episode,
sectionId: SectionId
): Promise<Array<BoxDropDto>> {
const drops: Array<BoxDropDto> = [];
const dropsBuf = await fs.promises.readFile(
`${RESOURCE_DIR}/login-config/drop/ep${episode}_box_${difficulty}_${sectionId}.txt`
);
let lineNo = 0;
let prevLine = '';
let prevPrevLine = '';
for (const line of dropsBuf.toString('utf8').split('\n')) {
const trimmed = line.trim();
if (trimmed.startsWith('#')) continue;
if (lineNo % 3 == 2) {
const areaId = parseInt(prevPrevLine, 10);
const dropRate = expandDropRate(parseInt(prevLine, 10));
const itemTypeId = parseInt(trimmed, 16);
if (dropRate > 0 && itemTypeId) {
drops.push({
difficulty: Difficulty[difficulty],
episode: episode,
sectionId: SectionId[sectionId],
areaId,
itemTypeId,
dropRate,
});
}
}
prevPrevLine = prevLine;
prevLine = trimmed;
lineNo++;
}
return drops;
}
function getStatBoosts(itemPmt: ItemPmt, statBoostIndex: number) {
const statBoost = itemPmt.statBoosts[statBoostIndex];
let atp = 0;
let ata = 0;
let minEvp = 0;
let minDfp = 0;
let mst = 0;
let hp = 0;
let lck = 0;
switch (statBoost.stat1) {
case 1: atp += statBoost.amount1; break;
case 2: ata += statBoost.amount1; break;
case 3: minEvp += statBoost.amount1; break;
case 4: minDfp += statBoost.amount1; break;
case 5: mst += statBoost.amount1; break;
case 6: hp += statBoost.amount1; break;
case 7: lck += statBoost.amount1; break;
case 8:
atp += statBoost.amount1;
ata += statBoost.amount1;
minEvp += statBoost.amount1;
minDfp += statBoost.amount1;
mst += statBoost.amount1;
hp += statBoost.amount1;
lck += statBoost.amount1;
break;
case 9: atp -= statBoost.amount1; break;
case 10: ata -= statBoost.amount1; break;
case 11: minEvp -= statBoost.amount1; break;
case 12: minDfp -= statBoost.amount1; break;
case 13: mst -= statBoost.amount1; break;
case 14: hp -= statBoost.amount1; break;
case 15: lck -= statBoost.amount1; break;
case 16:
atp -= statBoost.amount1;
ata -= statBoost.amount1;
minEvp -= statBoost.amount1;
minDfp -= statBoost.amount1;
mst -= statBoost.amount1;
hp -= statBoost.amount1;
lck -= statBoost.amount1;
break;
}
return { atp, ata, minEvp, minDfp, mst, hp, lck };
}
function getEnemyType(episode: Episode, index: number) {
if (episode === Episode.I) {
return [
undefined,
NpcType.Hildebear,
NpcType.Hildeblue,
NpcType.Mothmant,
NpcType.Monest,
NpcType.RagRappy,
NpcType.AlRappy,
NpcType.SavageWolf,
NpcType.BarbarousWolf,
NpcType.Booma,
NpcType.Gobooma,
NpcType.Gigobooma,
NpcType.GrassAssassin,
NpcType.PoisonLily,
NpcType.NarLily,
NpcType.NanoDragon,
NpcType.EvilShark,
NpcType.PalShark,
NpcType.GuilShark,
NpcType.PofuillySlime,
NpcType.PouillySlime,
NpcType.PanArms,
NpcType.Migium,
NpcType.Hidoom,
NpcType.Dubchic,
NpcType.Garanz,
NpcType.SinowBeat,
NpcType.SinowGold,
NpcType.Canadine,
NpcType.Canane,
NpcType.Delsaber,
NpcType.ChaosSorcerer,
undefined,
undefined,
NpcType.DarkGunner,
NpcType.DeathGunner,
NpcType.ChaosBringer,
NpcType.DarkBelra,
NpcType.Claw,
NpcType.Bulk,
NpcType.Bulclaw,
NpcType.Dimenian,
NpcType.LaDimenian,
NpcType.SoDimenian,
NpcType.Dragon,
NpcType.DeRolLe,
NpcType.VolOpt,
NpcType.DarkFalz,
undefined,
undefined,
NpcType.Gilchic
][index];
} else if (episode === Episode.II) {
return [
undefined,
NpcType.Hildebear2,
NpcType.Hildeblue2,
NpcType.Mothmant2,
NpcType.Monest2,
NpcType.RagRappy2,
undefined,
NpcType.SavageWolf2,
NpcType.BarbarousWolf2,
undefined,
undefined,
undefined,
NpcType.GrassAssassin2,
NpcType.PoisonLily2,
NpcType.NarLily2,
undefined,
undefined,
undefined,
undefined,
undefined,
undefined,
NpcType.PanArms2,
NpcType.Migium2,
NpcType.Hidoom2,
NpcType.Dubchic2,
NpcType.Garanz2,
undefined,
undefined,
undefined,
undefined,
NpcType.Delsaber2,
NpcType.ChaosSorcerer2,
undefined,
undefined,
undefined,
undefined,
undefined,
NpcType.DarkBelra2,
undefined,
undefined,
undefined,
NpcType.Dimenian2,
NpcType.LaDimenian2,
NpcType.SoDimenian2,
undefined,
undefined,
undefined,
undefined,
undefined,
undefined,
NpcType.Gilchic2,
NpcType.LoveRappy,
NpcType.Merillia,
NpcType.Meriltas,
NpcType.Gee,
NpcType.GiGue,
NpcType.Mericarol,
NpcType.Merikle,
NpcType.Mericus,
NpcType.UlGibbon,
NpcType.ZolGibbon,
NpcType.Gibbles,
NpcType.SinowBerill,
NpcType.SinowSpigell,
NpcType.Dolmolm,
NpcType.Dolmdarl,
NpcType.Morfos,
undefined,
NpcType.Recon,
NpcType.SinowZoa,
NpcType.SinowZele,
NpcType.Deldepth,
NpcType.Delbiter,
NpcType.BarbaRay,
undefined,
undefined,
NpcType.GolDragon,
NpcType.GalGryphon,
NpcType.OlgaFlow,
NpcType.StRappy,
NpcType.HalloRappy,
NpcType.EggRappy,
NpcType.IllGill,
NpcType.DelLily,
NpcType.Epsilon,
][index];
} else {
return [
undefined,
NpcType.Astark,
NpcType.Yowie,
NpcType.SatelliteLizard,
NpcType.MerissaA,
NpcType.MerissaAA,
NpcType.Girtablulu,
NpcType.Zu,
NpcType.Pazuzu,
NpcType.Boota,
NpcType.ZeBoota,
NpcType.BaBoota,
NpcType.Dorphon,
NpcType.DorphonEclair,
NpcType.Goran,
NpcType.GoranDetonator,
NpcType.PyroGoran,
NpcType.SandRappy,
NpcType.DelRappy,
NpcType.SaintMilion,
NpcType.Shambertin,
NpcType.Kondrieu,
][index];
}
}
function expandDropRate(pc: number): number {
let shift = ((pc >> 3) & 0x1F) - 4;
if (shift < 0) shift = 0;
return ((2 << shift) * ((pc & 7) + 7)) / 4294967296;
}
function npcTypeToPtIndex(type: NpcType): number | undefined {
switch (type) {
// Episode I Forest
case NpcType.Hildebear: return 1;
case NpcType.Hildeblue: return 2;
case NpcType.RagRappy: return 5;
case NpcType.AlRappy: return 6;
case NpcType.Monest: return 4;
case NpcType.Mothmant: return 3;
case NpcType.SavageWolf: return 7;
case NpcType.BarbarousWolf: return 8;
case NpcType.Booma: return 9;
case NpcType.Gobooma: return 10;
case NpcType.Gigobooma: return 11;
case NpcType.Dragon: return 44;
// Episode I Caves
case NpcType.GrassAssassin: return 12;
case NpcType.PoisonLily: return 13;
case NpcType.NarLily: return 14;
case NpcType.NanoDragon: return 15;
case NpcType.EvilShark: return 16;
case NpcType.PalShark: return 17;
case NpcType.GuilShark: return 18;
case NpcType.PofuillySlime: return 19;
case NpcType.PouillySlime: return 20;
case NpcType.PanArms: return 21;
case NpcType.Migium: return 22;
case NpcType.Hidoom: return 23;
case NpcType.DeRolLe: return 45;
// Episode I Mines
case NpcType.Dubchic: return 24;
case NpcType.Gilchic: return 50;
case NpcType.Garanz: return 25;
case NpcType.SinowBeat: return 26;
case NpcType.SinowGold: return 27;
case NpcType.Canadine: return 28;
case NpcType.Canane: return 29;
case NpcType.Dubswitch: return undefined;
case NpcType.VolOpt: return 46;
// Episode I Ruins
case NpcType.Delsaber: return 30;
case NpcType.ChaosSorcerer: return 31;
case NpcType.DarkGunner: return 34;
case NpcType.DeathGunner: return 35;
case NpcType.ChaosBringer: return 36;
case NpcType.DarkBelra: return 37;
case NpcType.Dimenian: return 41;
case NpcType.LaDimenian: return 42;
case NpcType.SoDimenian: return 43;
case NpcType.Bulclaw: return 40;
case NpcType.Bulk: return 39;
case NpcType.Claw: return 38;
case NpcType.DarkFalz: return 47;
// Episode II VR Temple
case NpcType.Hildebear2: return 1;
case NpcType.Hildeblue2: return 2;
case NpcType.RagRappy2: return 5;
case NpcType.LoveRappy: return 51;
case NpcType.StRappy: return 79;
case NpcType.HalloRappy: return 80;
case NpcType.EggRappy: return 81;
case NpcType.Monest2: return 4;
case NpcType.Mothmant2: return 3;
case NpcType.PoisonLily2: return 13;
case NpcType.NarLily2: return 14;
case NpcType.GrassAssassin2: return 12;
case NpcType.Dimenian2: return 41;
case NpcType.LaDimenian2: return 42;
case NpcType.SoDimenian2: return 43;
case NpcType.DarkBelra2: return 37;
case NpcType.BarbaRay: return 73;
// Episode II VR Spaceship
case NpcType.SavageWolf2: return 7;
case NpcType.BarbarousWolf2: return 8;
case NpcType.PanArms2: return 21;
case NpcType.Migium2: return 22;
case NpcType.Hidoom2: return 23;
case NpcType.Dubchic2: return 24;
case NpcType.Gilchic2: return 50;
case NpcType.Garanz2: return 25;
case NpcType.Dubswitch2: return undefined;
case NpcType.Delsaber2: return 30;
case NpcType.ChaosSorcerer2: return 31;
case NpcType.GolDragon: return 76;
// Episode II Central Control Area
case NpcType.SinowBerill: return 62;
case NpcType.SinowSpigell: return 63;
case NpcType.Merillia: return 52;
case NpcType.Meriltas: return 53;
case NpcType.Mericarol: return 56;
case NpcType.Mericus: return 58;
case NpcType.Merikle: return 57;
case NpcType.UlGibbon: return 59;
case NpcType.ZolGibbon: return 60;
case NpcType.Gibbles: return 61;
case NpcType.Gee: return 54;
case NpcType.GiGue: return 55;
case NpcType.IllGill: return 82;
case NpcType.DelLily: return 83;
case NpcType.Epsilon: return 84;
case NpcType.GalGryphon: return 77;
// Episode II Seabed
case NpcType.Deldepth: return 71;
case NpcType.Delbiter: return 72;
case NpcType.Dolmolm: return 64;
case NpcType.Dolmdarl: return 65;
case NpcType.Morfos: return 66;
case NpcType.Recobox: return 67;
case NpcType.Recon: return 68;
case NpcType.SinowZoa: return 69;
case NpcType.SinowZele: return 70;
case NpcType.OlgaFlow: return 78;
// Episode IV
case NpcType.SandRappy: return 17;
case NpcType.DelRappy: return 18;
case NpcType.Astark: return 1;
case NpcType.SatelliteLizard: return 3;
case NpcType.Yowie: return 2;
case NpcType.MerissaA: return 4;
case NpcType.MerissaAA: return 5;
case NpcType.Girtablulu: return 6;
case NpcType.Zu: return 7;
case NpcType.Pazuzu: return 8;
case NpcType.Boota: return 9;
case NpcType.ZeBoota: return 10;
case NpcType.BaBoota: return 11;
case NpcType.Dorphon: return 12;
case NpcType.DorphonEclair: return 13;
case NpcType.Goran: return 14;
case NpcType.PyroGoran: return 16;
case NpcType.GoranDetonator: return 15;
case NpcType.SaintMilion: return 19;
case NpcType.Shambertin: return 20;
case NpcType.Kondrieu: return 21;
default: return undefined;
}
}