import { filter, mapObjIndexed, pick } from 'ramda' import { ASTNode, EvaluatedNode } from './AST/types' import { bonus, mergeMissing } from './evaluation' import { registerEvaluationFunction } from './evaluationFunctions' import parse, { mecanismKeys } from './parse' import { Context } from './parsePublicodes' import { ReferenceNode } from './reference' import { parseRendNonApplicable, parseReplacements, ReplacementRule, } from './replacement' import { nameLeaf, ruleParents } from './ruleUtils' import { capitalise0 } from './utils' export type Rule = { formule?: Record | string question?: string description?: string unité?: string acronyme?: string exemples?: any nom: string résumé?: string icônes?: string titre?: string cotisation?: { branche: string } type?: string note?: string remplace?: RendNonApplicable | Array 'rend non applicable'?: Remplace | Array suggestions?: Record> références?: { [source: string]: string } API?: string } type Remplace = | { règle: string par?: Record | string | number dans?: Array | string 'sauf dans'?: Array | string } | string type RendNonApplicable = Exclude export type RuleNode = { dottedName: string title: string nodeKind: 'rule' virtualRule: boolean rawNode: Rule replacements: Array explanation: { parent: ASTNode | false valeur: ASTNode } suggestions: Record } export default function parseRule( rawRule: Rule, context: Context ): ReferenceNode { const dottedName = [context.dottedName, rawRule.nom] .filter(Boolean) .join(' . ') if (context.parsedRules[dottedName]) { throw new Error(`La référence '${dottedName}' a déjà été définie`) } const ruleValue = { ...pick(mecanismKeys, rawRule), ...('formule' in rawRule && { valeur: rawRule.formule }), 'nom dans la situation': dottedName, } const ruleContext = { ...context, dottedName } let name = nameLeaf(dottedName) if (context.dottedName) { name = `${nameLeaf(context.dottedName)} (${name})` } const [parent] = ruleParents(dottedName) const explanation = { valeur: parse(ruleValue, ruleContext), parent: !!parent && parse(parent, context), } context.parsedRules[dottedName] = filter(Boolean, { dottedName, replacements: [ ...parseRendNonApplicable(rawRule['rend non applicable'], ruleContext), ...parseReplacements(rawRule.remplace, ruleContext), ], title: capitalise0(rawRule['titre'] || name), suggestions: mapObjIndexed( (node) => parse(node, ruleContext), rawRule.suggestions ?? {} ), nodeKind: 'rule', explanation, rawNode: rawRule, virtualRule: !!context.dottedName, }) as RuleNode // We return the parsedReference return parse(rawRule.nom, context) as ReferenceNode } registerEvaluationFunction('rule', function evaluate(node) { if (this.cache[node.dottedName]) { return this.cache[node.dottedName] } const explanation = { ...node.explanation } this.cache._meta.contextRule.push(node.dottedName) this.cache._meta.parentEvaluationStack ??= [] let parent: EvaluatedNode | null = null if ( explanation.parent && !this.cache._meta.parentEvaluationStack.includes(node.dottedName) ) { this.cache._meta.parentEvaluationStack.push(node.dottedName) parent = this.evaluateNode(explanation.parent) as EvaluatedNode explanation.parent = parent this.cache._meta.parentEvaluationStack.pop() } let valeur: EvaluatedNode | null = null if (!parent || parent.nodeValue !== false) { valeur = this.evaluateNode(explanation.valeur) as EvaluatedNode explanation.valeur = valeur } const evaluation = { ...node, explanation, nodeValue: valeur && 'nodeValue' in valeur ? valeur.nodeValue : false, missingVariables: mergeMissing( valeur?.missingVariables, bonus(parent?.missingVariables) ), ...(valeur && 'unit' in valeur && { unit: valeur.unit }), } this.cache._meta.contextRule.pop() this.cache[node.dottedName] = evaluation return evaluation })