mirror of
https://github.com/koala73/worldmonitor.git
synced 2026-04-25 17:14:57 +02:00
* feat(resilience): recovery capacity pillar — 6 new dimensions + 5 seeders (Phase 2 T2.2b) Add the recovery-capacity pillar with 6 new dimensions: - fiscalSpace: IMF GGR_G01_GDP_PT + GGXCNL_G01_GDP_PT + GGXWDG_NGDP_PT - reserveAdequacy: World Bank FI.RES.TOTL.MO - externalDebtCoverage: WB DT.DOD.DSTC.CD / FI.RES.TOTL.CD ratio - importConcentration: UN Comtrade HHI (stub seeder) - stateContinuity: derived from WGI + UCDP + displacement (no new fetch) - fuelStockDays: IEA/EIA (stub seeder, Enrichment tier) Each dimension has a scorer in _dimension-scorers.ts, registry entries in _indicator-registry.ts, methodology doc subsections, and fixture data. Seeders: fiscal-space (real, IMF WEO), reserve-adequacy (real, WB API), external-debt (real, WB API), import-hhi (stub), fuel-stocks (stub). Recovery domain weight is 0 until PR 4 (T2.3) ships the penalized weighted mean across pillars. The domain appears in responses structurally but does not affect the overall score. Bootstrap: STANDALONE_KEYS + SEED_META + EMPTY_DATA_OK_KEYS + ON_DEMAND_KEYS all updated in api/health.js. Source-failure mapping updated for stateContinuity (WGI adapter). Widget labels and LOCKED_PREVIEW updated. All 282 resilience tests pass, typecheck clean, methodology lint clean. * fix(resilience): ISO3→ISO2 normalization in WB recovery seeders (#2987 P1) Both seed-recovery-reserve-adequacy.mjs and seed-recovery-external-debt.mjs used countryiso3code from the World Bank API response then immediately rejected codes where length !== 2. WB returns ISO3 codes (USA, DEU, etc.), so all real rows were silently dropped and the feed was always empty. Fix: import scripts/shared/iso3-to-iso2.json and normalize before the length check. Also removed from EMPTY_DATA_OK_KEYS in health.js since empty results now indicate a real failure, not a structural absence. * fix(resilience): remove unused import + no-op overrides (#2987 review) * fix(test): update release-gate to expect 6 domains after recovery pillar
46 lines
1.8 KiB
JavaScript
46 lines
1.8 KiB
JavaScript
import { describe, it } from 'node:test';
|
|
import assert from 'node:assert/strict';
|
|
import { readFileSync } from 'node:fs';
|
|
import { join, dirname } from 'node:path';
|
|
import { fileURLToPath } from 'node:url';
|
|
|
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
const iso3ToIso2 = JSON.parse(readFileSync(join(__dirname, '..', 'scripts', 'shared', 'iso3-to-iso2.json'), 'utf8'));
|
|
|
|
describe('seed-recovery-external-debt ISO3→ISO2', () => {
|
|
it('iso3-to-iso2.json maps common WB API ISO3 codes to ISO2', () => {
|
|
assert.equal(iso3ToIso2['USA'], 'US');
|
|
assert.equal(iso3ToIso2['DEU'], 'DE');
|
|
assert.equal(iso3ToIso2['BRA'], 'BR');
|
|
assert.equal(iso3ToIso2['IND'], 'IN');
|
|
assert.equal(iso3ToIso2['ZAF'], 'ZA');
|
|
});
|
|
|
|
it('normalizes ISO3 countryiso3code from WB response to ISO2', () => {
|
|
const rawCode = 'DEU';
|
|
const iso2 = rawCode.length === 3 ? (iso3ToIso2[rawCode] ?? null) : (rawCode.length === 2 ? rawCode : null);
|
|
assert.equal(iso2, 'DE');
|
|
});
|
|
|
|
it('passes through already-ISO2 codes', () => {
|
|
const rawCode = 'DE';
|
|
const iso2 = rawCode.length === 3 ? (iso3ToIso2[rawCode] ?? null) : (rawCode.length === 2 ? rawCode : null);
|
|
assert.equal(iso2, 'DE');
|
|
});
|
|
|
|
it('rejects codes that are neither ISO2 nor ISO3', () => {
|
|
for (const bad of ['', 'X', 'ABCD']) {
|
|
const iso2 = bad.length === 3 ? (iso3ToIso2[bad] ?? null) : (bad.length === 2 ? bad : null);
|
|
assert.equal(iso2, null, `"${bad}" should be rejected`);
|
|
}
|
|
});
|
|
|
|
it('rejects WB aggregate codes (e.g. WLD, EAS) that have no ISO2 mapping', () => {
|
|
const aggregates = ['WLD', 'EAS', 'ECS', 'LCN', 'MEA', 'SAS', 'SSF'];
|
|
for (const agg of aggregates) {
|
|
const iso2 = agg.length === 3 ? (iso3ToIso2[agg] ?? null) : null;
|
|
assert.equal(iso2, null, `WB aggregate "${agg}" should not map to ISO2`);
|
|
}
|
|
});
|
|
});
|