import { add, evolve, fromPairs, keys, map, mapObjIndexed, mergeWith, reduce, } from 'ramda' import Engine, { EvaluationFunction } from '.' import { ASTNode, ConstantNode, Evaluation, EvaluatedNode, NodeKind, } from './AST/types' import { typeWarning } from './error' import { convertNodeToUnit, simplifyNodeUnit } from './nodeUnits' import parse from './parse' import { concatTemporals, liftTemporalNode, mapTemporal, pureTemporal, Temporal, temporalAverage, zipTemporals, } from './temporal' export const collectNodeMissing = ( node: EvaluatedNode | ASTNode ): Record => 'missingVariables' in node ? node.missingVariables : {} export const bonus = (missings, hasCondition = true) => hasCondition ? map((x) => x + 0.0001, missings || {}) : missings export const mergeMissing = ( left: Record | undefined, right: Record | undefined ): Record => mergeWith(add, left || {}, right || {}) export const mergeAllMissing = (missings: Array) => missings.map(collectNodeMissing).reduce(mergeMissing, {}) function convertNodesToSameUnit(nodes, contextRule, mecanismName) { const firstNodeWithUnit = nodes.find((node) => !!node.unit) if (!firstNodeWithUnit) { return nodes } return nodes.map((node) => { try { return convertNodeToUnit(firstNodeWithUnit.unit, node) } catch (e) { typeWarning( contextRule, `Dans le mécanisme ${mecanismName}, les unités des éléments suivants sont incompatibles entre elles : \n\t\t${ node?.name || node?.rawNode }\n\t\t${firstNodeWithUnit?.name || firstNodeWithUnit?.rawNode}'`, e ) return node } }) } export const evaluateArray: ( reducer: Parameters[0], start: Parameters[1] ) => EvaluationFunction = (reducer, start) => function (node: any) { const evaluate = this.evaluateNode.bind(this) const evaluatedNodes = convertNodesToSameUnit( node.explanation.map(evaluate), this.cache._meta.contextRule, node.name ) const temporalValues = concatTemporals( evaluatedNodes.map( ({ temporalValue, nodeValue }) => temporalValue ?? pureTemporal(nodeValue) ) ) const temporalValue = mapTemporal((values) => { if (values.some((value) => value === null)) { return null } return reduce(reducer, start, values) }, temporalValues) const baseEvaluation = { ...node, missingVariables: mergeAllMissing(evaluatedNodes), explanation: evaluatedNodes, ...(evaluatedNodes[0] && { unit: evaluatedNodes[0].unit }), } if (temporalValue.length === 1) { return { ...baseEvaluation, nodeValue: temporalValue[0].value, } } return { ...baseEvaluation, temporalValue, nodeValue: temporalAverage(temporalValue as any), } } export const defaultNode = (nodeValue: Evaluation) => ({ nodeValue, type: typeof nodeValue, isDefault: true, nodeKind: 'constant', } as ConstantNode) export const parseObject = (objectShape, value, context) => { const recurseOne = (key) => (defaultValue) => { if (value[key] == null && !defaultValue) throw new Error( `Il manque une clé '${key}' dans ${JSON.stringify(value)} ` ) return value[key] != null ? parse(value[key], context) : defaultValue } const transforms = fromPairs( map((k) => [k, recurseOne(k)], keys(objectShape)) as any ) return evolve(transforms as any, objectShape) } export function evaluateObject( effet: (this: Engine, explanations: any) => any ) { return function (node) { const evaluate = this.evaluateNode.bind(this) const evaluations: Record = mapObjIndexed( evaluate as any, (node as any).explanation ) const temporalExplanations = mapTemporal( Object.fromEntries, concatTemporals( Object.entries(evaluations).map(([key, node]) => zipTemporals(pureTemporal(key), liftTemporalNode(node as ASTNode)) ) ) ) const temporalExplanation = mapTemporal((explanations) => { const evaluation = effet.call(this, explanations) return { ...evaluation, explanation: { ...explanations, ...evaluation.explanation, }, } }, temporalExplanations) const sameUnitTemporalExplanation: Temporal< ASTNode & EvaluatedNode & { nodeValue: number } > = convertNodesToSameUnit( temporalExplanation.map((x) => x.value), this.cache._meta.contextRule, node.nodeKind ).map((node, i) => ({ ...temporalExplanation[i], value: simplifyNodeUnit(node), })) const temporalValue = mapTemporal( ({ nodeValue }) => nodeValue, sameUnitTemporalExplanation ) const nodeValue = temporalAverage(temporalValue) const baseEvaluation = { ...node, nodeValue, unit: sameUnitTemporalExplanation[0].value.unit, explanation: evaluations, missingVariables: mergeAllMissing(Object.values(evaluations)), } if (sameUnitTemporalExplanation.length === 1) { return { ...baseEvaluation, explanation: (sameUnitTemporalExplanation[0] as any).value.explanation, } } return { ...baseEvaluation, temporalValue, temporalExplanation, } } as EvaluationFunction }