Two related changes bundled together because the diffuser curve files
only make sense once the registry namespace they live in exists.
src/registry — new asset-metadata resolver:
- AssetResolver with synchronous resolve(namespace, id) + lazy cache,
async refresh() for future remote pulls.
- FileBackend (per-id or single-file layouts, case-insensitive) and a
stub HttpBackend (disabled unless EVOLV_ASSET_REMOTE=1).
- Namespaces: curves, menu, monsterSamples, monsterSpecs, units. Menu
namespace re-keys by inner softwareType + filename so editors that
pass either string resolve to the same tree.
- README explains how to add a namespace.
- AssetCategoryManager (datasets/assetData/index.js) becomes a thin
facade over the resolver so existing consumers don't move.
- 246/246 tests pass — including the 39-test registry suite.
datasets/assetData — file moves + new diffuser data:
- modelData/*.json deleted; curves/*.json is the canonical home.
- New diffuser.json menu tree with GVA, Jäger, Aquaconsult/Entec,
PIK/PRK suppliers.
- gva-elastox-r.json migrated from the inline _loadSpecs hardcode,
re-tagged coverageBasis="bottom-coverage-pct" (the legacy 2.4
elements/m² was a prior mis-conversion; we can't recover the
original % so it's a single-point curve under key "0").
- jaeger-jetflex-td-65-2-g-epdm-1000.json — extracted from the Jäger
EPDM-1000mm SSOTE/DWP chart on the data sheet (vector-PDF read).
SSOTE 8.20→6.40 %/m, DWP 25→48 mbar across Q 2-12 Nm³/h. Single
coverage (vendor doesn't state test conditions).
- aerostrip-phoenix.json — 4-coverage SOTE family at 4.75 m water
depth (DD 5/10/15/20 %, flux 10-70 Nm³/h·m²) from the Entec/de
Winter 2023-11-22 dataset; DWP curve from the 21 % @ 4.05 m chart.
- pik300.json / prk300.json — 5-coverage SOTE + SSOTR (DD 5-25 %)
with split DWP per model variant, water depth ≈ 4.0 m inferred from
the SOTE↔SSOTR ratio in the source spreadsheet.
src/configs/diffuser.json:
- New asset.{model, assetTagNumber} block so the editor's selected
model id survives validation.
- diffuser.density description corrected to "Bottom coverage [%]";
default 2.4 → 15 (typical fine-bubble install).
src/configs/{rotatingMachine,valve}.json: small alignment edits that
came with the registry phase.
src/menu/asset.js + src/menu/aquonSamples.js: rewritten as facades
over assetResolver, keeping the editor-side cascade behaviour intact.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
100 lines
3.9 KiB
JavaScript
100 lines
3.9 KiB
JavaScript
'use strict';
|
|
|
|
// Smoke tests against the REAL datasets/ files. Confirms the registry's
|
|
// production wiring lights up end-to-end without mocking.
|
|
|
|
const test = require('node:test');
|
|
const assert = require('node:assert/strict');
|
|
|
|
const { assetResolver } = require('../../src/registry');
|
|
|
|
test('namespaces() includes curves, menu, monsterSamples, monsterSpecs, units', () => {
|
|
const ns = assetResolver.namespaces().sort();
|
|
assert.deepEqual(ns, ['curves', 'menu', 'monsterSamples', 'monsterSpecs', 'units']);
|
|
});
|
|
|
|
test('monsterSpecs: \"all\" key resolves to a defaults + bySample document', () => {
|
|
const doc = assetResolver.resolve('monsterSpecs', 'all');
|
|
assert.ok(doc, 'expected monsterSpecs/all');
|
|
assert.equal(typeof doc.defaults, 'object');
|
|
assert.equal(typeof doc.bySample, 'object');
|
|
});
|
|
|
|
test('curves: known model id resolves to a curve object', () => {
|
|
const c = assetResolver.resolve('curves', 'hidrostal-H05K-S03R');
|
|
assert.ok(c, 'expected a curve payload');
|
|
assert.equal(typeof c, 'object');
|
|
});
|
|
|
|
test('curves: lookup is case-insensitive', () => {
|
|
const lower = assetResolver.resolve('curves', 'hidrostal-h05k-s03r');
|
|
const upper = assetResolver.resolve('curves', 'HIDROSTAL-H05K-S03R');
|
|
assert.ok(lower);
|
|
assert.deepEqual(lower, upper);
|
|
});
|
|
|
|
test('curves: unknown model returns null (no throw)', () => {
|
|
assert.equal(assetResolver.resolve('curves', 'nope-not-here'), null);
|
|
});
|
|
|
|
test('menu: machine.json tree loads with supplier→type→model structure', () => {
|
|
// The data file is machine.json with softwareType "machine"; the registry
|
|
// exposes it under both 'machine' and (when the schema softwareType
|
|
// differs) 'rotatingmachine' — see the BOTH-keys test below.
|
|
const tree = assetResolver.resolve('menu', 'machine');
|
|
assert.ok(tree, 'menu/machine should exist (machine.json)');
|
|
assert.ok(Array.isArray(tree.suppliers));
|
|
assert.ok(tree.suppliers.length > 0);
|
|
});
|
|
|
|
test('menu: valve tree loads', () => {
|
|
const tree = assetResolver.resolve('menu', 'valve');
|
|
assert.ok(tree);
|
|
assert.ok(Array.isArray(tree.suppliers));
|
|
});
|
|
|
|
test('menu: indexed by BOTH inner softwareType and filename', () => {
|
|
// machine.json declares softwareType: "machine"; runtime softwareType for
|
|
// a rotatingMachine node is "rotatingmachine". Both should resolve to the
|
|
// same tree so all call paths work.
|
|
const bySoftwareType = assetResolver.resolve('menu', 'machine');
|
|
const byFilename = assetResolver.resolve('menu', 'machine');
|
|
assert.ok(bySoftwareType);
|
|
assert.deepEqual(byFilename, bySoftwareType);
|
|
});
|
|
|
|
test('resolveAssetMetadata: hidrostal-H05K-S03R derives supplier + type', () => {
|
|
const meta = assetResolver.resolveAssetMetadata('machine', 'hidrostal-H05K-S03R');
|
|
assert.ok(meta, 'expected metadata');
|
|
assert.equal(meta.supplier, 'Hidrostal');
|
|
assert.equal(meta.type, 'Centrifugal');
|
|
assert.ok(meta.units.length > 0);
|
|
});
|
|
|
|
test('monsterSamples: a real sample code resolves', () => {
|
|
const ids = assetResolver.list('monsterSamples');
|
|
assert.ok(ids.length > 0, 'expected at least one sample code');
|
|
const sample = assetResolver.resolve('monsterSamples', ids[0]);
|
|
assert.ok(sample);
|
|
assert.ok(sample.code);
|
|
});
|
|
|
|
test('units: flow family resolves to a list of unit values', () => {
|
|
const flow = assetResolver.resolve('units', 'flow');
|
|
assert.ok(flow);
|
|
assert.ok(Array.isArray(flow.values));
|
|
assert.ok(flow.values.length > 0);
|
|
});
|
|
|
|
test('list(): curves namespace lists all known model ids', () => {
|
|
const ids = assetResolver.list('curves');
|
|
assert.ok(ids.length >= 2, 'expected at least 2 curves');
|
|
assert.ok(ids.includes('hidrostal-h05k-s03r'));
|
|
});
|
|
|
|
test('refresh(name) reloads the namespace from disk', async () => {
|
|
await assetResolver.refresh('curves');
|
|
const c = assetResolver.resolve('curves', 'hidrostal-H05K-S03R');
|
|
assert.ok(c);
|
|
});
|