mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Added lint task and enforced code style in static_generation and test.
This commit is contained in:
parent
19681d69a3
commit
3c8f7ba01e
@ -22,6 +22,10 @@
|
|||||||
"@typescript-eslint/explicit-member-accessibility": "off",
|
"@typescript-eslint/explicit-member-accessibility": "off",
|
||||||
"@typescript-eslint/no-explicit-any": "off",
|
"@typescript-eslint/no-explicit-any": "off",
|
||||||
"@typescript-eslint/no-inferrable-types": "warn",
|
"@typescript-eslint/no-inferrable-types": "warn",
|
||||||
|
"@typescript-eslint/no-object-literal-type-assertion": [
|
||||||
|
"warn",
|
||||||
|
{ "allowAsParameter": true }
|
||||||
|
],
|
||||||
"@typescript-eslint/no-parameter-properties": "off",
|
"@typescript-eslint/no-parameter-properties": "off",
|
||||||
"@typescript-eslint/no-use-before-define": "off",
|
"@typescript-eslint/no-use-before-define": "off",
|
||||||
"@typescript-eslint/prefer-interface": "off",
|
"@typescript-eslint/prefer-interface": "off",
|
||||||
|
@ -34,7 +34,8 @@
|
|||||||
"build": "craco build",
|
"build": "craco build",
|
||||||
"test": "craco test",
|
"test": "craco test",
|
||||||
"update_generic_data": "ts-node --project=tsconfig-scripts.json static_generation/update_generic_data.ts",
|
"update_generic_data": "ts-node --project=tsconfig-scripts.json static_generation/update_generic_data.ts",
|
||||||
"update_ephinea_data": "ts-node --project=tsconfig-scripts.json static_generation/update_ephinea_data.ts"
|
"update_ephinea_data": "ts-node --project=tsconfig-scripts.json static_generation/update_ephinea_data.ts",
|
||||||
|
"lint": "prettier --check \"{src,static_generation,test}/**/*.{ts,tsx}\" && eslint \"{src,static_generation,test}/**/*.{ts,tsx}\" && echo All code passes the prettier and eslint checks."
|
||||||
},
|
},
|
||||||
"eslintConfig": {
|
"eslintConfig": {
|
||||||
"extends": "react-app"
|
"extends": "react-app"
|
||||||
|
@ -47,7 +47,7 @@ test("reallocation of internal buffer when necessary", () => {
|
|||||||
expect(cursor.buffer.byteLength).toBeGreaterThanOrEqual(4);
|
expect(cursor.buffer.byteLength).toBeGreaterThanOrEqual(4);
|
||||||
});
|
});
|
||||||
|
|
||||||
function test_integer_read(method_name: string) {
|
function test_integer_read(method_name: string): void {
|
||||||
test(method_name, () => {
|
test(method_name, () => {
|
||||||
const bytes = parseInt(method_name.replace(/^[iu](\d+)$/, "$1"), 10) / 8;
|
const bytes = parseInt(method_name.replace(/^[iu](\d+)$/, "$1"), 10) / 8;
|
||||||
let test_number_1 = 0;
|
let test_number_1 = 0;
|
||||||
@ -98,7 +98,7 @@ test("u8_array", () => {
|
|||||||
expect(cursor.seek_start(5).u8_array(3)).toEqual([6, 7, 8]);
|
expect(cursor.seek_start(5).u8_array(3)).toEqual([6, 7, 8]);
|
||||||
});
|
});
|
||||||
|
|
||||||
function test_string_read(method_name: string, char_size: number) {
|
function test_string_read(method_name: string, char_size: number): void {
|
||||||
test(method_name, () => {
|
test(method_name, () => {
|
||||||
const char_array = [7, 65, 66, 0, 255, 13];
|
const char_array = [7, 65, 66, 0, 255, 13];
|
||||||
|
|
||||||
@ -145,7 +145,7 @@ function test_string_read(method_name: string, char_size: number) {
|
|||||||
test_string_read("string_ascii", 1);
|
test_string_read("string_ascii", 1);
|
||||||
test_string_read("string_utf16", 2);
|
test_string_read("string_utf16", 2);
|
||||||
|
|
||||||
function test_integer_write(method_name: string) {
|
function test_integer_write(method_name: string): void {
|
||||||
test(method_name, () => {
|
test(method_name, () => {
|
||||||
const bytes = parseInt(method_name.replace(/^write_[iu](\d+)$/, "$1"), 10) / 8;
|
const bytes = parseInt(method_name.replace(/^write_[iu](\d+)$/, "$1"), 10) / 8;
|
||||||
let test_number_1 = 0;
|
let test_number_1 = 0;
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { BufferCursor } from "../../BufferCursor";
|
import { BufferCursor } from "../../BufferCursor";
|
||||||
import { compress, decompress } from "../prs";
|
import { compress, decompress } from "../prs";
|
||||||
|
|
||||||
function test_with_bytes(bytes: number[], expected_compressed_size: number) {
|
function test_with_bytes(bytes: number[], expected_compressed_size: number): void {
|
||||||
const cursor = new BufferCursor(new Uint8Array(bytes).buffer, true);
|
const cursor = new BufferCursor(new Uint8Array(bytes).buffer, true);
|
||||||
|
|
||||||
for (const byte of bytes) {
|
for (const byte of bytes) {
|
||||||
@ -38,7 +38,7 @@ test("PRS compression and decompression, worst case", () => {
|
|||||||
const prng = new Prng();
|
const prng = new Prng();
|
||||||
|
|
||||||
// Compression factor: 1.124
|
// Compression factor: 1.124
|
||||||
test_with_bytes(new Array(1000).fill(0).map(_ => prng.next_integer(0, 255)), 1124);
|
test_with_bytes(new Array(1000).fill(0).map(() => prng.next_integer(0, 255)), 1124);
|
||||||
});
|
});
|
||||||
|
|
||||||
test("PRS compression and decompression, typical case", () => {
|
test("PRS compression and decompression, typical case", () => {
|
||||||
|
@ -48,7 +48,7 @@ class PrcDecryptor {
|
|||||||
return out_cursor;
|
return out_cursor;
|
||||||
}
|
}
|
||||||
|
|
||||||
private construct_keys(key: number) {
|
private construct_keys(key: number): void {
|
||||||
this.keys[55] = key;
|
this.keys[55] = key;
|
||||||
|
|
||||||
let idx;
|
let idx;
|
||||||
@ -68,7 +68,7 @@ class PrcDecryptor {
|
|||||||
this.mix_keys();
|
this.mix_keys();
|
||||||
}
|
}
|
||||||
|
|
||||||
private mix_keys() {
|
private mix_keys(): void {
|
||||||
let ptr = 1;
|
let ptr = 1;
|
||||||
|
|
||||||
for (let i = 24; i; --i, ++ptr) {
|
for (let i = 24; i; --i, ++ptr) {
|
||||||
@ -82,7 +82,7 @@ class PrcDecryptor {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private decrypt_u32(data: number) {
|
private decrypt_u32(data: number): number {
|
||||||
if (this.key_pos === 56) {
|
if (this.key_pos === 56) {
|
||||||
this.mix_keys();
|
this.mix_keys();
|
||||||
this.key_pos = 1;
|
this.key_pos = 1;
|
||||||
|
@ -51,7 +51,7 @@ test("parse_quest and write_quest_qst", () => {
|
|||||||
expect(testable_area_variants(test_quest)).toEqual(testable_area_variants(orig_quest));
|
expect(testable_area_variants(test_quest)).toEqual(testable_area_variants(orig_quest));
|
||||||
});
|
});
|
||||||
|
|
||||||
function testable_objects(quest: Quest) {
|
function testable_objects(quest: Quest): any[][] {
|
||||||
return quest.objects.map(object => [
|
return quest.objects.map(object => [
|
||||||
object.area_id,
|
object.area_id,
|
||||||
object.section_id,
|
object.section_id,
|
||||||
@ -60,10 +60,10 @@ function testable_objects(quest: Quest) {
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
function testable_npcs(quest: Quest) {
|
function testable_npcs(quest: Quest): any[][] {
|
||||||
return quest.npcs.map(npc => [npc.area_id, npc.section_id, npc.position, npc.type]);
|
return quest.npcs.map(npc => [npc.area_id, npc.section_id, npc.position, npc.type]);
|
||||||
}
|
}
|
||||||
|
|
||||||
function testable_area_variants(quest: Quest) {
|
function testable_area_variants(quest: Quest): any[][] {
|
||||||
return quest.area_variants.map(av => [av.area.id, av.id]);
|
return quest.area_variants.map(av => [av.area.id, av.id]);
|
||||||
}
|
}
|
||||||
|
@ -7,7 +7,7 @@ import Logger from "js-logger";
|
|||||||
|
|
||||||
const logger = Logger.get("static/update_drops_ephinea");
|
const logger = Logger.get("static/update_drops_ephinea");
|
||||||
|
|
||||||
export async function update_drops_from_website(item_types: ItemTypeDto[]) {
|
export async function update_drops_from_website(item_types: ItemTypeDto[]): Promise<void> {
|
||||||
logger.info("Updating item drops.");
|
logger.info("Updating item drops.");
|
||||||
|
|
||||||
const normal = await download(item_types, Difficulty.Normal);
|
const normal = await download(item_types, Difficulty.Normal);
|
||||||
@ -38,15 +38,15 @@ async function download(
|
|||||||
item_types: ItemTypeDto[],
|
item_types: ItemTypeDto[],
|
||||||
difficulty: Difficulty,
|
difficulty: Difficulty,
|
||||||
difficulty_url: string = Difficulty[difficulty].toLowerCase()
|
difficulty_url: string = Difficulty[difficulty].toLowerCase()
|
||||||
) {
|
): Promise<{ enemy_drops: EnemyDropDto[]; box_drops: BoxDropDto[]; items: Set<string> }> {
|
||||||
const response = await fetch(`https://ephinea.pioneer2.net/drop-charts/${difficulty_url}/`);
|
const response = await fetch(`https://ephinea.pioneer2.net/drop-charts/${difficulty_url}/`);
|
||||||
const body = await response.text();
|
const body = await response.text();
|
||||||
const $ = cheerio.load(body);
|
const $ = cheerio.load(body);
|
||||||
|
|
||||||
let episode = 1;
|
let episode = 1;
|
||||||
const data: {
|
const data: {
|
||||||
enemy_drops: Array<EnemyDropDto>;
|
enemy_drops: EnemyDropDto[];
|
||||||
box_drops: Array<BoxDropDto>;
|
box_drops: BoxDropDto[];
|
||||||
items: Set<string>;
|
items: Set<string>;
|
||||||
} = {
|
} = {
|
||||||
enemy_drops: [],
|
enemy_drops: [],
|
||||||
|
@ -33,6 +33,10 @@ const RESOURCE_DIR = "./static/resources/ephinea";
|
|||||||
* Used by production code.
|
* Used by production code.
|
||||||
*/
|
*/
|
||||||
const PUBLIC_DIR = "./public";
|
const PUBLIC_DIR = "./public";
|
||||||
|
/**
|
||||||
|
* Enable this if we ever get the Ephinea ItemPT.gsl file.
|
||||||
|
*/
|
||||||
|
const USE_ITEMPT = false;
|
||||||
|
|
||||||
update().catch(e => logger.error(e));
|
update().catch(e => logger.error(e));
|
||||||
|
|
||||||
@ -43,18 +47,21 @@ update().catch(e => logger.error(e));
|
|||||||
* - Clio is equipable by HUnewearls
|
* - Clio is equipable by HUnewearls
|
||||||
* - Red Ring has a requirement of 180, not 108
|
* - Red Ring has a requirement of 180, not 108
|
||||||
*/
|
*/
|
||||||
async function update() {
|
async function update(): Promise<void> {
|
||||||
logger.info("Updating static Ephinea data.");
|
logger.info("Updating static Ephinea data.");
|
||||||
|
|
||||||
const unitxt = load_unitxt();
|
const unitxt = load_unitxt();
|
||||||
const item_names = unitxt[1];
|
const item_names = unitxt[1];
|
||||||
const items = update_items(item_names);
|
const items = update_items(item_names);
|
||||||
await update_drops_from_website(items);
|
|
||||||
update_quests();
|
|
||||||
|
|
||||||
// Use this if we ever get the Ephinea drop files.
|
if (USE_ITEMPT) {
|
||||||
// const item_pt = await load_item_pt();
|
const item_pt = await load_item_pt();
|
||||||
// await update_drops(item_pt);
|
await update_drops(item_pt);
|
||||||
|
} else {
|
||||||
|
await update_drops_from_website(items);
|
||||||
|
}
|
||||||
|
|
||||||
|
update_quests();
|
||||||
|
|
||||||
logger.info("Done updating static Ephinea data.");
|
logger.info("Done updating static Ephinea data.");
|
||||||
}
|
}
|
||||||
@ -72,7 +79,7 @@ async function update() {
|
|||||||
* - The Value of Money (quest3_e.dat, can't be parsed, luckily doesn't have enemies)
|
* - The Value of Money (quest3_e.dat, can't be parsed, luckily doesn't have enemies)
|
||||||
* Note: The MA4R quests use a random area variation per area from the ABC MA quests. E.g. MA4-1R will use a random caves 2 variation from MA4-1A, MA4-1B or MA4-1C. Same for mines 2 and ruins 2.
|
* Note: The MA4R quests use a random area variation per area from the ABC MA quests. E.g. MA4-1R will use a random caves 2 variation from MA4-1A, MA4-1B or MA4-1C. Same for mines 2 and ruins 2.
|
||||||
*/
|
*/
|
||||||
function update_quests() {
|
function update_quests(): Promise<void> {
|
||||||
logger.info("Updating quest data.");
|
logger.info("Updating quest data.");
|
||||||
|
|
||||||
const quests = new Array<QuestDto>();
|
const quests = new Array<QuestDto>();
|
||||||
@ -97,7 +104,7 @@ function update_quests() {
|
|||||||
logger.info("Done updating quest data.");
|
logger.info("Done updating quest data.");
|
||||||
}
|
}
|
||||||
|
|
||||||
function process_quest_dir(path: string, quests: QuestDto[]) {
|
function process_quest_dir(path: string, quests: QuestDto[]): void {
|
||||||
const stat = fs.statSync(path);
|
const stat = fs.statSync(path);
|
||||||
|
|
||||||
if (stat.isFile()) {
|
if (stat.isFile()) {
|
||||||
@ -109,7 +116,7 @@ function process_quest_dir(path: string, quests: QuestDto[]) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function process_quest(path: string, quests: QuestDto[]) {
|
function process_quest(path: string, quests: QuestDto[]): void {
|
||||||
try {
|
try {
|
||||||
const buf = fs.readFileSync(path);
|
const buf = fs.readFileSync(path);
|
||||||
const q = parse_quest(new BufferCursor(buf.buffer, true), true);
|
const q = parse_quest(new BufferCursor(buf.buffer, true), true);
|
||||||
@ -157,7 +164,7 @@ function load_unitxt(): Unitxt {
|
|||||||
return unitxt;
|
return unitxt;
|
||||||
}
|
}
|
||||||
|
|
||||||
function update_items(item_names: Array<string>): ItemTypeDto[] {
|
function update_items(item_names: string[]): ItemTypeDto[] {
|
||||||
logger.info("Updating item type data.");
|
logger.info("Updating item type data.");
|
||||||
|
|
||||||
const buf = fs.readFileSync(`${RESOURCE_DIR}/ship-config/param/ItemPMT.bin`);
|
const buf = fs.readFileSync(`${RESOURCE_DIR}/ship-config/param/ItemPMT.bin`);
|
||||||
@ -262,7 +269,7 @@ function update_items(item_names: Array<string>): ItemTypeDto[] {
|
|||||||
return item_types;
|
return item_types;
|
||||||
}
|
}
|
||||||
|
|
||||||
function update_drops(item_pt: ItemPt) {
|
function update_drops(item_pt: ItemPt): void {
|
||||||
logger.info("Updating drop data.");
|
logger.info("Updating drop data.");
|
||||||
|
|
||||||
const enemy_drops = new Array<EnemyDropDto>();
|
const enemy_drops = new Array<EnemyDropDto>();
|
||||||
@ -295,7 +302,7 @@ function update_drops(item_pt: ItemPt) {
|
|||||||
type ItemP = {
|
type ItemP = {
|
||||||
dar_table: Map<NpcType, number>;
|
dar_table: Map<NpcType, number>;
|
||||||
};
|
};
|
||||||
type ItemPt = Array<Array<Array<ItemP>>>;
|
type ItemPt = ItemP[][][];
|
||||||
|
|
||||||
async function load_item_pt(): Promise<ItemPt> {
|
async function load_item_pt(): Promise<ItemPt> {
|
||||||
logger.info("Loading ItemPT.gsl.");
|
logger.info("Loading ItemPT.gsl.");
|
||||||
@ -560,8 +567,8 @@ function load_box_drops(
|
|||||||
difficulty: Difficulty,
|
difficulty: Difficulty,
|
||||||
episode: Episode,
|
episode: Episode,
|
||||||
section_id: SectionId
|
section_id: SectionId
|
||||||
): Array<BoxDropDto> {
|
): BoxDropDto[] {
|
||||||
const drops: Array<BoxDropDto> = [];
|
const drops: BoxDropDto[] = [];
|
||||||
const drops_buf = fs.readFileSync(
|
const drops_buf = fs.readFileSync(
|
||||||
`${RESOURCE_DIR}/login-config/drop/ep${episode}_box_${difficulty}_${section_id}.txt`
|
`${RESOURCE_DIR}/login-config/drop/ep${episode}_box_${difficulty}_${section_id}.txt`
|
||||||
);
|
);
|
||||||
@ -599,7 +606,18 @@ function load_box_drops(
|
|||||||
return drops;
|
return drops;
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_stat_boosts(item_pmt: ItemPmt, stat_boost_index: number) {
|
function get_stat_boosts(
|
||||||
|
item_pmt: ItemPmt,
|
||||||
|
stat_boost_index: number
|
||||||
|
): {
|
||||||
|
atp: number;
|
||||||
|
ata: number;
|
||||||
|
minEvp: number;
|
||||||
|
minDfp: number;
|
||||||
|
mst: number;
|
||||||
|
hp: number;
|
||||||
|
lck: number;
|
||||||
|
} {
|
||||||
const stat_boost = item_pmt.stat_boosts[stat_boost_index];
|
const stat_boost = item_pmt.stat_boosts[stat_boost_index];
|
||||||
let atp = 0;
|
let atp = 0;
|
||||||
let ata = 0;
|
let ata = 0;
|
||||||
@ -675,7 +693,7 @@ function get_stat_boosts(item_pmt: ItemPmt, stat_boost_index: number) {
|
|||||||
return { atp, ata, minEvp: min_evp, minDfp: min_dfp, mst, hp, lck };
|
return { atp, ata, minEvp: min_evp, minDfp: min_dfp, mst, hp, lck };
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_enemy_type(episode: Episode, index: number) {
|
function get_enemy_type(episode: Episode, index: number): NpcType | undefined {
|
||||||
if (episode === Episode.I) {
|
if (episode === Episode.I) {
|
||||||
return [
|
return [
|
||||||
undefined,
|
undefined,
|
||||||
|
@ -18,7 +18,7 @@ const PUBLIC_DIR = "./public";
|
|||||||
|
|
||||||
update();
|
update();
|
||||||
|
|
||||||
function update() {
|
function update(): void {
|
||||||
logger.info("Updating generic static data.");
|
logger.info("Updating generic static data.");
|
||||||
|
|
||||||
logger.info("Extracting player animations.");
|
logger.info("Extracting player animations.");
|
||||||
|
@ -8,7 +8,7 @@ import * as fs from "fs";
|
|||||||
export function walk_qst_files(
|
export function walk_qst_files(
|
||||||
f: (path: string, file_name: string, contents: Buffer) => void,
|
f: (path: string, file_name: string, contents: Buffer) => void,
|
||||||
dir: string = "test/resources/tethealla_v0.143_quests"
|
dir: string = "test/resources/tethealla_v0.143_quests"
|
||||||
) {
|
): void {
|
||||||
for (const [path, file] of get_qst_files(dir)) {
|
for (const [path, file] of get_qst_files(dir)) {
|
||||||
f(path, file, fs.readFileSync(path));
|
f(path, file, fs.readFileSync(path));
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user