import yaml from 'yaml' import { ParsedRules } from '.' import { transformAST, traverseParsedRules } from './AST' import parse from './parse' import { getReplacements, inlineReplacements } from './replacement' import { Rule, RuleNode } from './rule' import { disambiguateRuleReference } from './ruleUtils' export type Context = { dottedName: string parsedRules: Record ruleTitle?: string } type RawRule = Omit | string | undefined | number export type RawPublicodes = Record | string export default function parsePublicodes( rawRules: RawPublicodes, partialContext: Partial = {} ): ParsedRules { // STEP 1: parse Yaml let rules = typeof rawRules === 'string' ? (yaml.parse(('' + rawRules).replace(/\t/g, ' ')) as Record< string, RawRule >) : { ...rawRules } // STEP 2: transpile [ref] writing rules = transpileRef(rules) // STEP 3: Rules parsing const context: Context = { dottedName: partialContext.dottedName ?? '', parsedRules: partialContext.parsedRules ?? {}, } Object.entries(rules).forEach(([dottedName, rule]) => { if (rule == null) { rule = {} } if (typeof rule !== 'object') { rule = { formule: '' + rule, } } parse({ nom: dottedName, ...rule }, context) }) let parsedRules = context.parsedRules // STEP 4: Disambiguate reference parsedRules = traverseParsedRules( disambiguateReference(parsedRules), parsedRules ) // STEP 5: Inline replacements const replacements = getReplacements(parsedRules) parsedRules = traverseParsedRules( inlineReplacements(replacements), parsedRules ) // TODO STEP 6: check for cycle // TODO STEP 7: type check return parsedRules } // We recursively traverse the YAML tree in order to transform named parameters // into rules. function transpileRef(object: Record | string | Array) { if (Array.isArray(object)) { return object.map(transpileRef) } if (!object || typeof object !== 'object') { return object } object return Object.entries(object).reduce((obj, [key, value]) => { const match = /\[ref( (.+))?\]$/.exec(key) if (!match) { return { ...obj, [key]: transpileRef(value) } } const argumentType = key.replace(match[0], '').trim() const argumentName = match[2]?.trim() || argumentType return { ...obj, [argumentType]: { nom: argumentName, valeur: transpileRef(value), }, } }, {}) } export const disambiguateReference = (parsedRules: Record) => transformAST((node) => { if (node.nodeKind === 'reference') { const dottedName = disambiguateRuleReference( parsedRules, node.contextDottedName, node.name ) return { ...node, dottedName, title: parsedRules[dottedName].title, acronym: parsedRules[dottedName].rawNode.acronyme, } } })