2026-03-31 13:02:29 +08:00
|
|
|
import { parse } from 'csv-parse/sync';
|
|
|
|
|
import { parseSchema, createValidator, parseValue } from '../index.js';
|
2026-04-11 22:56:01 +08:00
|
|
|
import type { Schema, ReferenceSchema } from '../types.js';
|
|
|
|
|
import * as fs from 'fs';
|
|
|
|
|
import * as path from 'path';
|
2026-03-31 13:02:29 +08:00
|
|
|
|
2026-04-15 13:58:14 +08:00
|
|
|
function hasNestedReferences(schema: Schema): boolean {
|
|
|
|
|
switch (schema.type) {
|
|
|
|
|
case 'reference':
|
|
|
|
|
return true;
|
|
|
|
|
case 'tuple':
|
|
|
|
|
return schema.elements.some(el => hasNestedReferences(el.schema));
|
|
|
|
|
case 'array':
|
|
|
|
|
return hasNestedReferences(schema.element);
|
|
|
|
|
case 'union':
|
|
|
|
|
return schema.members.some(m => hasNestedReferences(m));
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function loadReferenceTable(
|
|
|
|
|
schema: ReferenceSchema,
|
|
|
|
|
refBaseDir: string | undefined,
|
|
|
|
|
defaultPrimaryKey: string,
|
|
|
|
|
currentFilePath: string | undefined
|
|
|
|
|
): { lookup: Map<string, Record<string, unknown>>; refTable: Record<string, unknown>[] } {
|
|
|
|
|
const baseDir = refBaseDir || (currentFilePath ? path.dirname(currentFilePath) : process.cwd());
|
|
|
|
|
const fileName = `${schema.tableName}.csv`;
|
|
|
|
|
const refFilePath = path.isAbsolute(fileName)
|
|
|
|
|
? fileName
|
|
|
|
|
: path.join(baseDir, fileName);
|
|
|
|
|
|
|
|
|
|
let refTable: Record<string, unknown>[];
|
|
|
|
|
if (referenceTableCache.has(refFilePath)) {
|
|
|
|
|
refTable = referenceTableCache.get(refFilePath)!;
|
|
|
|
|
} else {
|
|
|
|
|
if (loadingFiles.has(refFilePath)) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`Circular reference detected: table "${schema.tableName}" (${refFilePath}) is already being loaded`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
loadingFiles.add(refFilePath);
|
|
|
|
|
try {
|
|
|
|
|
const refContent = fs.readFileSync(refFilePath, 'utf-8');
|
|
|
|
|
const refResult = parseCsv(refContent, {
|
|
|
|
|
currentFilePath: refFilePath,
|
|
|
|
|
emitTypes: false,
|
|
|
|
|
});
|
|
|
|
|
refTable = refResult.data;
|
|
|
|
|
referenceTableCache.set(refFilePath, refTable);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`Failed to load referenced table "${schema.tableName}" from ${refFilePath}: ${error instanceof Error ? error.message : String(error)}`
|
|
|
|
|
);
|
|
|
|
|
} finally {
|
|
|
|
|
loadingFiles.delete(refFilePath);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const lookup = new Map<string, Record<string, unknown>>();
|
|
|
|
|
refTable.forEach(row => {
|
|
|
|
|
const pkValue = row[defaultPrimaryKey];
|
|
|
|
|
if (pkValue !== undefined) {
|
|
|
|
|
lookup.set(String(pkValue), row);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return { lookup, refTable };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function resolveReferenceId(
|
|
|
|
|
id: string,
|
|
|
|
|
lookup: Map<string, Record<string, unknown>>,
|
|
|
|
|
tableName: string
|
|
|
|
|
): Record<string, unknown> {
|
|
|
|
|
const obj = lookup.get(id);
|
|
|
|
|
if (!obj) {
|
|
|
|
|
throw new Error(`Reference to "${tableName}" with id="${id}" not found`);
|
|
|
|
|
}
|
|
|
|
|
return obj;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-15 14:36:52 +08:00
|
|
|
function parseReferenceIds(schema: ReferenceSchema, valueString: string): unknown {
|
2026-04-17 11:41:06 +08:00
|
|
|
const trimmed = valueString.trim();
|
|
|
|
|
if (schema.isOptional && trimmed === '') {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
const valueParser = new ReferenceValueParser(trimmed);
|
2026-04-15 14:36:52 +08:00
|
|
|
const ids = valueParser.parseIds(schema.isArray);
|
|
|
|
|
if (schema.isArray) {
|
|
|
|
|
return ids;
|
|
|
|
|
}
|
|
|
|
|
return ids[0];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function parseValueWithReferenceIds(
|
|
|
|
|
valueString: string,
|
|
|
|
|
schema: Schema
|
|
|
|
|
): unknown {
|
|
|
|
|
if (!hasNestedReferences(schema)) {
|
|
|
|
|
return parseValue(schema, valueString);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (schema.type) {
|
|
|
|
|
case 'reference':
|
|
|
|
|
return parseReferenceIds(schema, valueString);
|
|
|
|
|
case 'tuple': {
|
|
|
|
|
const parsed = parseValue(schema, valueString) as unknown[];
|
|
|
|
|
return schema.elements.map((el, i) =>
|
|
|
|
|
hasNestedReferences(el.schema)
|
|
|
|
|
? extractNestedReferenceIds(parsed[i], el.schema)
|
|
|
|
|
: parsed[i]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
case 'array': {
|
|
|
|
|
const parsed = parseValue(schema, valueString) as unknown[];
|
|
|
|
|
return parsed.map(item =>
|
|
|
|
|
hasNestedReferences(schema.element)
|
|
|
|
|
? extractNestedReferenceIds(item, schema.element)
|
|
|
|
|
: item
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
case 'union': {
|
|
|
|
|
for (const member of schema.members) {
|
|
|
|
|
if (hasNestedReferences(member)) {
|
|
|
|
|
try {
|
|
|
|
|
const parsed = parseValue(member, valueString);
|
|
|
|
|
return extractNestedReferenceIds(parsed, member);
|
|
|
|
|
} catch {}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return parseValue(schema, valueString);
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
return parseValue(schema, valueString);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function extractNestedReferenceIds(value: unknown, schema: Schema): unknown {
|
|
|
|
|
switch (schema.type) {
|
|
|
|
|
case 'reference':
|
2026-04-17 11:41:06 +08:00
|
|
|
if (value === null || value === undefined) return value;
|
2026-04-15 14:36:52 +08:00
|
|
|
if (schema.isArray) {
|
|
|
|
|
const ids = Array.isArray(value) ? value : [value];
|
|
|
|
|
return ids.map(id => String(id));
|
|
|
|
|
}
|
|
|
|
|
return String(value);
|
|
|
|
|
case 'tuple': {
|
|
|
|
|
if (!Array.isArray(value)) return value;
|
|
|
|
|
return schema.elements.map((el, i) =>
|
|
|
|
|
hasNestedReferences(el.schema)
|
|
|
|
|
? extractNestedReferenceIds(value[i], el.schema)
|
|
|
|
|
: value[i]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
case 'array': {
|
|
|
|
|
if (!Array.isArray(value)) return value;
|
|
|
|
|
return value.map(item =>
|
|
|
|
|
hasNestedReferences(schema.element)
|
|
|
|
|
? extractNestedReferenceIds(item, schema.element)
|
|
|
|
|
: item
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
case 'union': {
|
|
|
|
|
for (const member of schema.members) {
|
|
|
|
|
if (hasNestedReferences(member)) {
|
|
|
|
|
try {
|
|
|
|
|
return extractNestedReferenceIds(value, member);
|
|
|
|
|
} catch {}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function collectReferenceFields(schema: Schema, name: string): ReferenceFieldInfo[] {
|
|
|
|
|
const fields: ReferenceFieldInfo[] = [];
|
|
|
|
|
switch (schema.type) {
|
|
|
|
|
case 'reference':
|
|
|
|
|
fields.push({ name, tableName: schema.tableName, isArray: schema.isArray, schema });
|
|
|
|
|
break;
|
|
|
|
|
case 'tuple':
|
|
|
|
|
for (const el of schema.elements) {
|
|
|
|
|
fields.push(...collectReferenceFields(el.schema, name));
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'array':
|
|
|
|
|
fields.push(...collectReferenceFields(schema.element, name));
|
|
|
|
|
break;
|
|
|
|
|
case 'union':
|
|
|
|
|
for (const member of schema.members) {
|
|
|
|
|
fields.push(...collectReferenceFields(member, name));
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return fields;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-15 13:58:14 +08:00
|
|
|
function parseValueWithReferences(
|
|
|
|
|
valueString: string,
|
|
|
|
|
schema: Schema,
|
|
|
|
|
refBaseDir: string | undefined,
|
|
|
|
|
defaultPrimaryKey: string,
|
|
|
|
|
currentFilePath: string | undefined
|
|
|
|
|
): unknown {
|
|
|
|
|
if (!hasNestedReferences(schema)) {
|
|
|
|
|
return parseValue(schema, valueString);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (schema.type) {
|
|
|
|
|
case 'reference':
|
|
|
|
|
return parseReferenceValue(schema, valueString, refBaseDir, defaultPrimaryKey, currentFilePath);
|
|
|
|
|
case 'tuple': {
|
|
|
|
|
const parsed = parseValue(schema, valueString) as unknown[];
|
|
|
|
|
return schema.elements.map((el, i) =>
|
|
|
|
|
resolveNestedReferences(parsed[i], el.schema, refBaseDir, defaultPrimaryKey, currentFilePath)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
case 'array': {
|
|
|
|
|
const parsed = parseValue(schema, valueString) as unknown[];
|
|
|
|
|
return parsed.map(item =>
|
|
|
|
|
resolveNestedReferences(item, schema.element, refBaseDir, defaultPrimaryKey, currentFilePath)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
case 'union': {
|
|
|
|
|
const errors: Error[] = [];
|
|
|
|
|
for (const member of schema.members) {
|
|
|
|
|
if (hasNestedReferences(member)) {
|
|
|
|
|
try {
|
|
|
|
|
const parsed = parseValue(member, valueString);
|
|
|
|
|
return resolveNestedReferences(parsed, member, refBaseDir, defaultPrimaryKey, currentFilePath);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
errors.push(e instanceof Error ? e : new Error(String(e)));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (errors.length > 0 && errors.every(e => /not found|Circular reference|Failed to load/.test(e.message))) {
|
|
|
|
|
for (const member of schema.members) {
|
|
|
|
|
if (!hasNestedReferences(member)) {
|
|
|
|
|
try {
|
|
|
|
|
return parseValue(member, valueString);
|
|
|
|
|
} catch {}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return parseValue(schema, valueString);
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
return parseValue(schema, valueString);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function resolveNestedReferences(
|
|
|
|
|
value: unknown,
|
|
|
|
|
schema: Schema,
|
|
|
|
|
refBaseDir: string | undefined,
|
|
|
|
|
defaultPrimaryKey: string,
|
|
|
|
|
currentFilePath: string | undefined
|
|
|
|
|
): unknown {
|
|
|
|
|
switch (schema.type) {
|
|
|
|
|
case 'reference': {
|
2026-04-17 11:41:06 +08:00
|
|
|
if (value === null || value === undefined) return value;
|
2026-04-15 13:58:14 +08:00
|
|
|
const { lookup } = loadReferenceTable(schema, refBaseDir, defaultPrimaryKey, currentFilePath);
|
|
|
|
|
if (schema.isArray) {
|
|
|
|
|
const ids = Array.isArray(value) ? value : [value];
|
|
|
|
|
return ids.map(id => resolveReferenceId(String(id), lookup, schema.tableName));
|
|
|
|
|
}
|
|
|
|
|
return resolveReferenceId(String(value), lookup, schema.tableName);
|
|
|
|
|
}
|
|
|
|
|
case 'tuple': {
|
|
|
|
|
if (!Array.isArray(value)) return value;
|
|
|
|
|
return schema.elements.map((el, i) =>
|
|
|
|
|
resolveNestedReferences(value[i], el.schema, refBaseDir, defaultPrimaryKey, currentFilePath)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
case 'array': {
|
|
|
|
|
if (!Array.isArray(value)) return value;
|
|
|
|
|
return value.map(item =>
|
|
|
|
|
resolveNestedReferences(item, schema.element, refBaseDir, defaultPrimaryKey, currentFilePath)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
case 'union': {
|
|
|
|
|
const errors: Error[] = [];
|
|
|
|
|
for (const member of schema.members) {
|
|
|
|
|
if (hasNestedReferences(member)) {
|
|
|
|
|
try {
|
|
|
|
|
return resolveNestedReferences(value, member, refBaseDir, defaultPrimaryKey, currentFilePath);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
errors.push(e instanceof Error ? e : new Error(String(e)));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (errors.length > 0) {
|
|
|
|
|
throw errors[0];
|
|
|
|
|
}
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-31 13:02:29 +08:00
|
|
|
export interface CsvLoaderOptions {
|
|
|
|
|
delimiter?: string;
|
|
|
|
|
quote?: string;
|
|
|
|
|
escape?: string;
|
2026-03-31 14:45:02 +08:00
|
|
|
bom?: boolean;
|
|
|
|
|
comment?: string | false;
|
|
|
|
|
trim?: boolean;
|
2026-03-31 15:19:03 +08:00
|
|
|
/** Generate TypeScript declaration file (.d.ts) */
|
|
|
|
|
emitTypes?: boolean;
|
|
|
|
|
/** Output directory for generated type files (relative to output path) */
|
|
|
|
|
typesOutputDir?: string;
|
2026-03-31 15:54:38 +08:00
|
|
|
/** Write .d.ts files to disk (useful for dev server) */
|
|
|
|
|
writeToDisk?: boolean;
|
2026-04-11 22:56:01 +08:00
|
|
|
/** Base directory for resolving referenced CSV files (default: directory of current file) */
|
|
|
|
|
refBaseDir?: string;
|
|
|
|
|
/** Primary key field name for referenced tables (default: 'id') */
|
|
|
|
|
defaultPrimaryKey?: string;
|
|
|
|
|
/** Current file path (used to resolve relative references) */
|
|
|
|
|
currentFilePath?: string;
|
2026-04-15 14:36:52 +08:00
|
|
|
/**
|
|
|
|
|
* When false, reference fields store parsed IDs instead of resolved objects.
|
|
|
|
|
* Used by csvToModule to emit accessor-based code with lazy resolution.
|
|
|
|
|
* Default: true (resolves references eagerly by loading referenced CSV files).
|
|
|
|
|
*/
|
|
|
|
|
resolveReferences?: boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface ReferenceFieldInfo {
|
|
|
|
|
/** Column name in the CSV */
|
|
|
|
|
name: string;
|
|
|
|
|
/** Referenced table name */
|
|
|
|
|
tableName: string;
|
|
|
|
|
/** Whether it's an array reference */
|
|
|
|
|
isArray: boolean;
|
|
|
|
|
/** The schema of this field (for nested references) */
|
|
|
|
|
schema: Schema;
|
2026-03-31 13:02:29 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-02 17:32:13 +08:00
|
|
|
export interface CsvParseResult {
|
|
|
|
|
/** Parsed CSV data as array of objects */
|
|
|
|
|
data: Record<string, unknown>[];
|
|
|
|
|
/** Generated TypeScript type definition string (if emitTypes is true) */
|
|
|
|
|
typeDefinition?: string;
|
|
|
|
|
/** Property configurations for the CSV columns */
|
|
|
|
|
propertyConfigs: PropertyConfig[];
|
2026-04-11 22:56:01 +08:00
|
|
|
/** Referenced table names */
|
|
|
|
|
references: Set<string>;
|
2026-04-15 14:36:52 +08:00
|
|
|
/** Reference field metadata (populated when resolveReferences is false) */
|
|
|
|
|
referenceFields: ReferenceFieldInfo[];
|
2026-04-02 17:32:13 +08:00
|
|
|
}
|
|
|
|
|
|
2026-03-31 13:02:29 +08:00
|
|
|
interface PropertyConfig {
|
|
|
|
|
name: string;
|
|
|
|
|
schema: any;
|
|
|
|
|
validator: (value: unknown) => boolean;
|
|
|
|
|
parser: (valueString: string) => unknown;
|
2026-04-11 22:56:01 +08:00
|
|
|
/** Whether this property is a reference to another table */
|
|
|
|
|
isReference?: boolean;
|
|
|
|
|
/** Referenced table name (if isReference is true) */
|
|
|
|
|
referenceTableName?: string;
|
|
|
|
|
/** Whether it's an array reference */
|
|
|
|
|
referenceIsArray?: boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Cache for loaded referenced tables */
|
2026-04-15 13:58:14 +08:00
|
|
|
const referenceTableCache = new Map<string, Record<string,unknown>[]>();
|
|
|
|
|
|
|
|
|
|
/** Set of file paths currently being loaded (to detect circular references) */
|
|
|
|
|
const loadingFiles = new Set<string>();
|
2026-04-11 22:56:01 +08:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Parse and resolve a reference value.
|
|
|
|
|
* Loads the referenced table and replaces IDs with actual objects.
|
|
|
|
|
*/
|
|
|
|
|
function parseReferenceValue(
|
|
|
|
|
schema: ReferenceSchema,
|
|
|
|
|
valueString: string,
|
|
|
|
|
refBaseDir: string | undefined,
|
|
|
|
|
defaultPrimaryKey: string,
|
|
|
|
|
currentFilePath: string | undefined
|
|
|
|
|
): unknown {
|
2026-04-17 11:41:06 +08:00
|
|
|
const trimmed = valueString.trim();
|
|
|
|
|
if (schema.isOptional && trimmed === '') {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-15 13:58:14 +08:00
|
|
|
const { lookup } = loadReferenceTable(schema, refBaseDir, defaultPrimaryKey, currentFilePath);
|
2026-04-11 22:56:01 +08:00
|
|
|
|
2026-04-17 11:41:06 +08:00
|
|
|
const valueParser = new ReferenceValueParser(trimmed);
|
2026-04-11 22:56:01 +08:00
|
|
|
const ids = valueParser.parseIds(schema.isArray);
|
|
|
|
|
|
|
|
|
|
if (schema.isArray) {
|
2026-04-15 13:58:14 +08:00
|
|
|
return ids.map(id => resolveReferenceId(id, lookup, schema.tableName));
|
2026-04-11 22:56:01 +08:00
|
|
|
}
|
2026-04-15 13:58:14 +08:00
|
|
|
|
|
|
|
|
return resolveReferenceId(ids[0], lookup, schema.tableName);
|
2026-04-11 22:56:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Parser for reference values (extracts IDs from value string)
|
|
|
|
|
*/
|
|
|
|
|
class ReferenceValueParser {
|
|
|
|
|
private input: string;
|
|
|
|
|
private pos: number = 0;
|
|
|
|
|
|
|
|
|
|
constructor(input: string) {
|
|
|
|
|
this.input = input;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private peek(): string {
|
|
|
|
|
return this.input[this.pos] || '';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private consume(): string {
|
|
|
|
|
return this.input[this.pos++] || '';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private skipWhitespace(): void {
|
|
|
|
|
while (this.pos < this.input.length && /\s/.test(this.input[this.pos])) {
|
|
|
|
|
this.pos++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private consumeStr(str: string): boolean {
|
|
|
|
|
if (this.input.slice(this.pos, this.pos + str.length) === str) {
|
|
|
|
|
this.pos += str.length;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
parseIds(isArray: boolean): string[] {
|
|
|
|
|
this.skipWhitespace();
|
|
|
|
|
|
|
|
|
|
if (isArray) {
|
|
|
|
|
// Parse array format: [id1; id2; id3]
|
|
|
|
|
if (this.peek() === '[') {
|
|
|
|
|
this.consume();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.skipWhitespace();
|
|
|
|
|
|
|
|
|
|
if (this.peek() === ']') {
|
|
|
|
|
this.consume();
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const ids: string[] = [];
|
|
|
|
|
while (true) {
|
|
|
|
|
this.skipWhitespace();
|
|
|
|
|
let id = '';
|
|
|
|
|
while (this.pos < this.input.length && this.peek() !== ';' && this.peek() !== ']') {
|
|
|
|
|
id += this.consume();
|
|
|
|
|
}
|
|
|
|
|
const trimmedId = id.trim();
|
|
|
|
|
if (trimmedId) {
|
|
|
|
|
ids.push(trimmedId);
|
|
|
|
|
}
|
|
|
|
|
this.skipWhitespace();
|
|
|
|
|
|
|
|
|
|
if (!this.consumeStr(';')) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.skipWhitespace();
|
|
|
|
|
if (this.peek() === ']') {
|
|
|
|
|
this.consume();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ids;
|
|
|
|
|
} else {
|
|
|
|
|
// Parse single ID
|
|
|
|
|
let id = '';
|
|
|
|
|
while (this.pos < this.input.length) {
|
|
|
|
|
const char = this.peek();
|
|
|
|
|
if (char === ';' || char === ']' || char === ',') {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
id += this.consume();
|
|
|
|
|
}
|
|
|
|
|
return [id.trim()];
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-03-31 13:02:29 +08:00
|
|
|
}
|
|
|
|
|
|
2026-03-31 15:19:03 +08:00
|
|
|
/**
|
|
|
|
|
* Convert a schema to TypeScript type string
|
|
|
|
|
*/
|
2026-04-17 15:11:46 +08:00
|
|
|
export function schemaToTypeString(schema: Schema, resourceNames?: Map<string, string>): string {
|
2026-03-31 15:19:03 +08:00
|
|
|
switch (schema.type) {
|
|
|
|
|
case 'string':
|
|
|
|
|
return 'string';
|
|
|
|
|
case 'number':
|
2026-04-04 17:08:29 +08:00
|
|
|
case 'int':
|
|
|
|
|
case 'float':
|
2026-03-31 15:19:03 +08:00
|
|
|
return 'number';
|
|
|
|
|
case 'boolean':
|
|
|
|
|
return 'boolean';
|
2026-04-13 10:16:56 +08:00
|
|
|
case 'stringLiteral':
|
|
|
|
|
return `"${schema.value}"`;
|
|
|
|
|
case 'union':
|
|
|
|
|
return schema.members.map(m => schemaToTypeString(m, resourceNames)).join(' | ');
|
2026-04-11 22:56:01 +08:00
|
|
|
case 'reference': {
|
2026-04-13 10:16:56 +08:00
|
|
|
const typeName = resourceNames?.get(schema.tableName) ||
|
2026-04-11 22:56:01 +08:00
|
|
|
schema.tableName.charAt(0).toUpperCase() + schema.tableName.slice(1);
|
2026-04-17 14:53:49 +08:00
|
|
|
const baseType = schema.isArray ? `${typeName}[]` : typeName;
|
2026-04-17 11:41:06 +08:00
|
|
|
return schema.isOptional ? `${baseType} | null` : baseType;
|
2026-04-11 22:56:01 +08:00
|
|
|
}
|
2026-03-31 15:19:03 +08:00
|
|
|
case 'array':
|
|
|
|
|
if (schema.element.type === 'tuple') {
|
2026-03-31 16:57:52 +08:00
|
|
|
const tupleElements = schema.element.elements.map((el) => {
|
2026-04-11 22:56:01 +08:00
|
|
|
const typeStr = schemaToTypeString(el.schema, resourceNames);
|
2026-04-17 14:53:49 +08:00
|
|
|
return el.name ? `${el.name}: ${typeStr}` : typeStr;
|
2026-03-31 16:57:52 +08:00
|
|
|
});
|
2026-04-17 15:11:46 +08:00
|
|
|
return `[${tupleElements.join(', ')}][]`;
|
2026-03-31 15:19:03 +08:00
|
|
|
}
|
2026-04-13 10:16:56 +08:00
|
|
|
// Wrap union types in parentheses to maintain correct precedence
|
|
|
|
|
const elementType = schemaToTypeString(schema.element, resourceNames);
|
|
|
|
|
if (schema.element.type === 'union') {
|
2026-04-17 14:53:49 +08:00
|
|
|
return `(${elementType})[]`;
|
2026-04-13 10:16:56 +08:00
|
|
|
}
|
2026-04-17 14:53:49 +08:00
|
|
|
return `${elementType}[]`;
|
2026-03-31 15:19:03 +08:00
|
|
|
case 'tuple':
|
2026-03-31 16:57:52 +08:00
|
|
|
const tupleElements = schema.elements.map((el) => {
|
2026-04-11 22:56:01 +08:00
|
|
|
const typeStr = schemaToTypeString(el.schema, resourceNames);
|
2026-04-17 14:53:49 +08:00
|
|
|
return el.name ? `${el.name}: ${typeStr}` : typeStr;
|
2026-03-31 16:57:52 +08:00
|
|
|
});
|
2026-04-17 14:53:49 +08:00
|
|
|
return `[${tupleElements.join(', ')}]`;
|
2026-03-31 15:19:03 +08:00
|
|
|
default:
|
|
|
|
|
return 'unknown';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Generate TypeScript interface for the CSV data
|
|
|
|
|
*/
|
|
|
|
|
function generateTypeDefinition(
|
|
|
|
|
resourceName: string,
|
2026-04-11 22:56:01 +08:00
|
|
|
propertyConfigs: PropertyConfig[],
|
|
|
|
|
references: Set<string>,
|
2026-04-15 14:52:41 +08:00
|
|
|
currentFilePath?: string
|
2026-03-31 15:19:03 +08:00
|
|
|
): string {
|
2026-04-05 12:38:33 +08:00
|
|
|
const typeName = resourceName ? `${resourceName}Table` : 'Table';
|
2026-04-15 14:46:03 +08:00
|
|
|
const currentTableName = currentFilePath
|
|
|
|
|
? path.basename(currentFilePath, path.extname(currentFilePath))
|
|
|
|
|
: undefined;
|
|
|
|
|
const singularType = resourceName
|
|
|
|
|
? resourceName.charAt(0).toUpperCase() + resourceName.slice(1)
|
|
|
|
|
: `${typeName}[number]`;
|
2026-04-11 22:56:01 +08:00
|
|
|
|
|
|
|
|
// Generate import statements for referenced tables
|
|
|
|
|
const imports: string[] = [];
|
|
|
|
|
const resourceNames = new Map<string, string>();
|
|
|
|
|
|
|
|
|
|
references.forEach(tableName => {
|
2026-04-15 14:46:03 +08:00
|
|
|
if (tableName === currentTableName) {
|
|
|
|
|
resourceNames.set(tableName, singularType);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2026-04-11 23:05:22 +08:00
|
|
|
// Convert table name to type name by capitalizing
|
|
|
|
|
const typeBase = tableName.charAt(0).toUpperCase() + tableName.slice(1);
|
2026-04-11 22:56:01 +08:00
|
|
|
resourceNames.set(tableName, typeBase);
|
|
|
|
|
|
2026-04-11 23:05:22 +08:00
|
|
|
// Generate import path based on current file path
|
|
|
|
|
let importPath: string;
|
|
|
|
|
if (currentFilePath) {
|
|
|
|
|
importPath = `./${tableName}.csv`;
|
|
|
|
|
} else {
|
|
|
|
|
importPath = `../${tableName}.csv`;
|
|
|
|
|
}
|
2026-04-11 22:56:01 +08:00
|
|
|
imports.push(`import type { ${typeBase} } from '${importPath}';`);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const importSection = imports.length > 0 ? imports.join('\n') + '\n\n' : '';
|
|
|
|
|
|
2026-03-31 15:19:03 +08:00
|
|
|
const properties = propertyConfigs
|
2026-04-11 22:56:01 +08:00
|
|
|
.map((config) => ` readonly ${config.name}: ${schemaToTypeString(config.schema, resourceNames)};`)
|
2026-03-31 15:19:03 +08:00
|
|
|
.join('\n');
|
2026-04-05 12:38:33 +08:00
|
|
|
|
2026-04-11 23:05:22 +08:00
|
|
|
let exportAlias = '';
|
|
|
|
|
if (resourceName) {
|
|
|
|
|
const singularType = resourceName.charAt(0).toUpperCase() + resourceName.slice(1);
|
|
|
|
|
exportAlias = `\nexport type ${singularType} = ${typeName}[number];`;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-15 14:52:41 +08:00
|
|
|
return `${importSection}type ${typeName} = readonly {
|
2026-04-15 14:36:52 +08:00
|
|
|
${properties}
|
|
|
|
|
}[];
|
|
|
|
|
${exportAlias}
|
|
|
|
|
|
|
|
|
|
declare function getData(): ${typeName};
|
|
|
|
|
export default getData;
|
2026-03-31 15:19:03 +08:00
|
|
|
`;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-02 17:32:13 +08:00
|
|
|
/**
|
|
|
|
|
* Parse CSV content string into structured data with schema validation.
|
|
|
|
|
* This is a standalone function that doesn't depend on webpack/rspack LoaderContext.
|
2026-04-11 22:56:01 +08:00
|
|
|
*
|
2026-04-02 17:32:13 +08:00
|
|
|
* @param content - CSV content string (must have at least headers + schema row + 1 data row)
|
|
|
|
|
* @param options - Parsing options
|
|
|
|
|
* @returns CsvParseResult containing parsed data and optional type definitions
|
|
|
|
|
*/
|
|
|
|
|
export function parseCsv(
|
|
|
|
|
content: string,
|
2026-04-05 12:38:33 +08:00
|
|
|
options: CsvLoaderOptions & { resourceName?: string } = {}
|
2026-04-02 17:32:13 +08:00
|
|
|
): CsvParseResult {
|
|
|
|
|
const delimiter = options.delimiter ?? ',';
|
|
|
|
|
const quote = options.quote ?? '"';
|
|
|
|
|
const escape = options.escape ?? '\\';
|
|
|
|
|
const bom = options.bom ?? true;
|
|
|
|
|
const comment = options.comment === false ? undefined : (options.comment ?? '#');
|
|
|
|
|
const trim = options.trim ?? true;
|
|
|
|
|
const emitTypes = options.emitTypes ?? true;
|
2026-04-11 22:56:01 +08:00
|
|
|
const refBaseDir = options.refBaseDir;
|
|
|
|
|
const defaultPrimaryKey = options.defaultPrimaryKey ?? 'id';
|
2026-03-31 13:02:29 +08:00
|
|
|
|
|
|
|
|
const records = parse(content, {
|
|
|
|
|
delimiter,
|
|
|
|
|
quote,
|
|
|
|
|
escape,
|
2026-03-31 14:45:02 +08:00
|
|
|
bom,
|
|
|
|
|
comment,
|
|
|
|
|
trim,
|
2026-04-07 12:11:01 +08:00
|
|
|
skip_empty_lines: true,
|
2026-03-31 13:02:29 +08:00
|
|
|
relax_column_count: true,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (records.length < 2) {
|
|
|
|
|
throw new Error('CSV must have at least 2 rows: headers and schemas');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const headers = records[0];
|
|
|
|
|
const schemas = records[1];
|
|
|
|
|
|
|
|
|
|
if (headers.length !== schemas.length) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`Header count (${headers.length}) does not match schema count (${schemas.length})`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-15 14:36:52 +08:00
|
|
|
const resolveReferences = options.resolveReferences ?? true;
|
|
|
|
|
|
2026-03-31 13:02:29 +08:00
|
|
|
const propertyConfigs: PropertyConfig[] = headers.map((header: string, index: number) => {
|
|
|
|
|
const schemaString = schemas[index];
|
|
|
|
|
const schema = parseSchema(schemaString);
|
2026-04-11 22:56:01 +08:00
|
|
|
|
|
|
|
|
const config: PropertyConfig = {
|
2026-03-31 13:02:29 +08:00
|
|
|
name: header,
|
|
|
|
|
schema,
|
|
|
|
|
validator: createValidator(schema),
|
|
|
|
|
parser: (valueString: string) => parseValue(schema, valueString),
|
|
|
|
|
};
|
2026-04-11 22:56:01 +08:00
|
|
|
|
|
|
|
|
if (schema.type === 'reference') {
|
|
|
|
|
config.isReference = true;
|
|
|
|
|
config.referenceTableName = schema.tableName;
|
|
|
|
|
config.referenceIsArray = schema.isArray;
|
2026-04-15 14:36:52 +08:00
|
|
|
if (resolveReferences) {
|
|
|
|
|
config.parser = (valueString: string) => {
|
|
|
|
|
return parseReferenceValue(schema, valueString, refBaseDir, defaultPrimaryKey, options.currentFilePath);
|
|
|
|
|
};
|
|
|
|
|
} else {
|
|
|
|
|
config.parser = (valueString: string) => {
|
|
|
|
|
return parseReferenceIds(schema, valueString);
|
|
|
|
|
};
|
|
|
|
|
}
|
2026-04-15 13:58:14 +08:00
|
|
|
} else if (hasNestedReferences(schema)) {
|
|
|
|
|
config.isReference = true;
|
2026-04-15 14:36:52 +08:00
|
|
|
if (resolveReferences) {
|
|
|
|
|
config.parser = (valueString: string) => {
|
|
|
|
|
return parseValueWithReferences(valueString, schema, refBaseDir, defaultPrimaryKey, options.currentFilePath);
|
|
|
|
|
};
|
|
|
|
|
} else {
|
|
|
|
|
config.parser = (valueString: string) => {
|
|
|
|
|
return parseValueWithReferenceIds(valueString, schema);
|
|
|
|
|
};
|
|
|
|
|
}
|
2026-04-11 22:56:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return config;
|
|
|
|
|
});
|
|
|
|
|
|
2026-04-15 13:24:51 +08:00
|
|
|
// Collect all referenced tables (including nested references in tuples/arrays)
|
2026-04-11 22:56:01 +08:00
|
|
|
const references = new Set<string>();
|
2026-04-15 13:24:51 +08:00
|
|
|
function collectReferences(schema: Schema): void {
|
|
|
|
|
if (schema.type === 'reference') {
|
|
|
|
|
references.add(schema.tableName);
|
|
|
|
|
} else if (schema.type === 'tuple') {
|
|
|
|
|
schema.elements.forEach(el => collectReferences(el.schema));
|
|
|
|
|
} else if (schema.type === 'array') {
|
|
|
|
|
collectReferences(schema.element);
|
|
|
|
|
} else if (schema.type === 'union') {
|
|
|
|
|
schema.members.forEach(m => collectReferences(m));
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-04-11 22:56:01 +08:00
|
|
|
propertyConfigs.forEach(config => {
|
|
|
|
|
if (config.isReference && config.referenceTableName) {
|
|
|
|
|
references.add(config.referenceTableName);
|
|
|
|
|
}
|
2026-04-15 13:24:51 +08:00
|
|
|
collectReferences(config.schema);
|
2026-03-31 13:02:29 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const dataRows = records.slice(2);
|
|
|
|
|
const objects = dataRows.map((row: string[], rowIndex: number) => {
|
|
|
|
|
const obj: Record<string, unknown> = {};
|
|
|
|
|
propertyConfigs.forEach((config, colIndex) => {
|
|
|
|
|
const rawValue = row[colIndex] ?? '';
|
|
|
|
|
try {
|
|
|
|
|
const parsed = config.parser(rawValue);
|
2026-04-11 22:56:01 +08:00
|
|
|
// Skip validation for reference fields (validation happens during reference resolution)
|
|
|
|
|
if (!config.isReference && !config.validator(parsed)) {
|
2026-03-31 13:02:29 +08:00
|
|
|
throw new Error(
|
|
|
|
|
`Validation failed for property "${config.name}" at row ${rowIndex + 3}: ${rawValue}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
obj[config.name] = parsed;
|
|
|
|
|
} catch (error) {
|
|
|
|
|
if (error instanceof Error) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`Failed to parse property "${config.name}" at row ${rowIndex + 3}, column ${colIndex + 1}: ${error.message}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
throw error;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return obj;
|
|
|
|
|
});
|
|
|
|
|
|
2026-04-15 14:36:52 +08:00
|
|
|
const referenceFields: ReferenceFieldInfo[] = [];
|
|
|
|
|
if (!resolveReferences) {
|
|
|
|
|
for (const config of propertyConfigs) {
|
|
|
|
|
if (hasNestedReferences(config.schema)) {
|
|
|
|
|
referenceFields.push(...collectReferenceFields(config.schema, config.name));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-02 17:32:13 +08:00
|
|
|
const result: CsvParseResult = {
|
|
|
|
|
data: objects,
|
|
|
|
|
propertyConfigs,
|
2026-04-11 22:56:01 +08:00
|
|
|
references,
|
2026-04-15 14:36:52 +08:00
|
|
|
referenceFields,
|
2026-04-02 17:32:13 +08:00
|
|
|
};
|
2026-03-31 15:49:05 +08:00
|
|
|
|
2026-03-31 15:19:03 +08:00
|
|
|
if (emitTypes) {
|
2026-04-11 22:56:01 +08:00
|
|
|
result.typeDefinition = generateTypeDefinition(
|
|
|
|
|
options.resourceName || '',
|
|
|
|
|
propertyConfigs,
|
|
|
|
|
references,
|
2026-04-15 14:52:41 +08:00
|
|
|
options.currentFilePath
|
2026-04-11 22:56:01 +08:00
|
|
|
);
|
2026-04-02 17:32:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-15 14:36:52 +08:00
|
|
|
/**
|
|
|
|
|
* Generate runtime reference resolution code for a schema.
|
|
|
|
|
* Returns a JS expression string that resolves references using lookup maps.
|
|
|
|
|
*/
|
|
|
|
|
function generateSchemaResolutionCode(
|
|
|
|
|
schema: Schema,
|
|
|
|
|
valueExpr: string,
|
|
|
|
|
lookupVar: (tableName: string) => string,
|
|
|
|
|
pkField: string
|
|
|
|
|
): string {
|
|
|
|
|
switch (schema.type) {
|
|
|
|
|
case 'reference': {
|
|
|
|
|
const lookup = lookupVar(schema.tableName);
|
2026-04-17 11:41:06 +08:00
|
|
|
if (schema.isOptional) {
|
|
|
|
|
if (schema.isArray) {
|
|
|
|
|
return `(${valueExpr} === null || ${valueExpr} === undefined ? ${valueExpr} : (Array.isArray(${valueExpr}) ? ${valueExpr}.map(id => ${lookup}.get(String(id))) : ${lookup}.get(String(${valueExpr}))))`;
|
|
|
|
|
}
|
|
|
|
|
return `(${valueExpr} === null || ${valueExpr} === undefined ? ${valueExpr} : ${lookup}.get(String(${valueExpr})))`;
|
|
|
|
|
}
|
2026-04-15 14:36:52 +08:00
|
|
|
if (schema.isArray) {
|
|
|
|
|
return `(Array.isArray(${valueExpr}) ? ${valueExpr}.map(id => ${lookup}.get(String(id))) : ${valueExpr})`;
|
|
|
|
|
}
|
|
|
|
|
return `${lookup}.get(String(${valueExpr}))`;
|
|
|
|
|
}
|
|
|
|
|
case 'tuple': {
|
|
|
|
|
const elementResolvers = schema.elements.map((el, i) => {
|
|
|
|
|
if (hasNestedReferences(el.schema)) {
|
|
|
|
|
return generateSchemaResolutionCode(el.schema, `${valueExpr}[${i}]`, lookupVar, pkField);
|
|
|
|
|
}
|
|
|
|
|
return `${valueExpr}[${i}]`;
|
|
|
|
|
});
|
|
|
|
|
return `[${elementResolvers.join(', ')}]`;
|
|
|
|
|
}
|
|
|
|
|
case 'array': {
|
|
|
|
|
if (hasNestedReferences(schema.element)) {
|
|
|
|
|
const itemResolve = generateSchemaResolutionCode(schema.element, 'item', lookupVar, pkField);
|
|
|
|
|
return `(${valueExpr}).map(item => ${itemResolve})`;
|
|
|
|
|
}
|
|
|
|
|
return valueExpr;
|
|
|
|
|
}
|
|
|
|
|
case 'union': {
|
|
|
|
|
const refMembers = schema.members.filter(m => hasNestedReferences(m));
|
|
|
|
|
const nonRefMembers = schema.members.filter(m => !hasNestedReferences(m));
|
2026-04-15 14:46:03 +08:00
|
|
|
const resolveParts: string[] = [];
|
2026-04-15 14:36:52 +08:00
|
|
|
for (const member of refMembers) {
|
|
|
|
|
const resolveCode = generateSchemaResolutionCode(member, valueExpr, lookupVar, pkField);
|
2026-04-15 14:46:03 +08:00
|
|
|
resolveParts.push(resolveCode);
|
2026-04-15 14:36:52 +08:00
|
|
|
}
|
|
|
|
|
if (nonRefMembers.length > 0) {
|
2026-04-15 14:46:03 +08:00
|
|
|
resolveParts.push(valueExpr);
|
2026-04-15 14:36:52 +08:00
|
|
|
}
|
2026-04-15 14:46:03 +08:00
|
|
|
if (resolveParts.length === 0) return valueExpr;
|
|
|
|
|
if (resolveParts.length === 1) return resolveParts[0];
|
|
|
|
|
return `(${resolveParts.join(' ?? ')})`;
|
2026-04-15 14:36:52 +08:00
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
return valueExpr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-02 17:32:13 +08:00
|
|
|
/**
|
|
|
|
|
* Generate JavaScript module code from CSV content.
|
2026-04-15 14:36:52 +08:00
|
|
|
* Emits an accessor function for tables with references (lazy resolution),
|
|
|
|
|
* or static JSON for tables without references.
|
2026-04-02 17:32:13 +08:00
|
|
|
*/
|
|
|
|
|
export function csvToModule(
|
|
|
|
|
content: string,
|
2026-04-05 12:38:33 +08:00
|
|
|
options: CsvLoaderOptions & { resourceName?: string } = {}
|
2026-04-02 17:32:13 +08:00
|
|
|
): { js: string; dts?: string } {
|
2026-04-15 14:36:52 +08:00
|
|
|
const result = parseCsv(content, { ...options, resolveReferences: false });
|
|
|
|
|
|
|
|
|
|
const hasRefs = result.referenceFields.length > 0;
|
|
|
|
|
const defaultPrimaryKey = options.defaultPrimaryKey ?? 'id';
|
2026-04-07 11:25:02 +08:00
|
|
|
|
2026-04-15 14:36:52 +08:00
|
|
|
const imports: string[] = [];
|
|
|
|
|
const lookupInits: string[] = [];
|
|
|
|
|
const lookupVarMap = new Map<string, string>();
|
|
|
|
|
|
2026-04-15 14:46:03 +08:00
|
|
|
const currentTableName = options.currentFilePath
|
|
|
|
|
? path.basename(options.currentFilePath, path.extname(options.currentFilePath))
|
|
|
|
|
: undefined;
|
|
|
|
|
|
2026-04-15 14:36:52 +08:00
|
|
|
const uniqueTables = new Set(result.referenceFields.map(f => f.tableName));
|
|
|
|
|
uniqueTables.forEach(tableName => {
|
2026-04-15 14:46:03 +08:00
|
|
|
const lookupVar = `_${tableName}Lookup`;
|
|
|
|
|
lookupVarMap.set(tableName, lookupVar);
|
|
|
|
|
|
|
|
|
|
if (tableName === currentTableName) {
|
|
|
|
|
lookupInits.push(
|
|
|
|
|
`const ${lookupVar} = new Map(_raw.map(p => [String(p.${defaultPrimaryKey}), p]));`
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
const varName = `_${tableName}`;
|
|
|
|
|
imports.push(`import ${varName} from './${tableName}.csv';`);
|
|
|
|
|
lookupInits.push(
|
|
|
|
|
`const ${lookupVar} = new Map(${varName}().map(p => [String(p.${defaultPrimaryKey}), p]));`
|
|
|
|
|
);
|
|
|
|
|
}
|
2026-04-15 14:36:52 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const lookupVar = (tableName: string) => lookupVarMap.get(tableName)!;
|
|
|
|
|
|
|
|
|
|
const rowResolvers: string[] = [];
|
|
|
|
|
for (const config of result.propertyConfigs) {
|
|
|
|
|
if (hasNestedReferences(config.schema)) {
|
|
|
|
|
const resolveCode = generateSchemaResolutionCode(
|
|
|
|
|
config.schema,
|
|
|
|
|
`row.${config.name}`,
|
|
|
|
|
lookupVar,
|
|
|
|
|
defaultPrimaryKey
|
|
|
|
|
);
|
|
|
|
|
rowResolvers.push(` ${config.name}: ${resolveCode},`);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const rawJson = JSON.stringify(result.data, null, 2);
|
|
|
|
|
|
2026-04-15 14:52:41 +08:00
|
|
|
const js = [
|
|
|
|
|
...imports,
|
|
|
|
|
'',
|
|
|
|
|
`const _raw = ${rawJson};`,
|
|
|
|
|
'',
|
|
|
|
|
'let _resolved = null;',
|
|
|
|
|
'',
|
|
|
|
|
'export default function getData() {',
|
|
|
|
|
' if (_resolved) return _resolved;',
|
|
|
|
|
' _resolved = _raw;',
|
|
|
|
|
...lookupInits.map(l => ` ${l}`),
|
|
|
|
|
...rowResolvers.length > 0 ? [
|
2026-04-15 14:36:52 +08:00
|
|
|
' _resolved = _raw.map(row => ({',
|
|
|
|
|
' ...row,',
|
|
|
|
|
...rowResolvers,
|
|
|
|
|
' }));',
|
2026-04-15 14:52:41 +08:00
|
|
|
] : [],
|
|
|
|
|
' return _resolved;',
|
|
|
|
|
'}',
|
|
|
|
|
].join('\n');
|
2026-04-07 11:25:02 +08:00
|
|
|
|
2026-04-02 17:32:13 +08:00
|
|
|
return {
|
|
|
|
|
js,
|
|
|
|
|
dts: result.typeDefinition,
|
|
|
|
|
};
|
|
|
|
|
}
|