import { add, map, pluck, any, equals, reduce, mergeWith, fromPairs, keys, values, evolve, filter, is } from 'ramda' export let makeJsx = node => typeof node.jsx == 'function' ? node.jsx(node.nodeValue, node.explanation) : node.jsx export let collectNodeMissing = node => node.missingVariables || {} export let bonus = (missings, hasCondition = true) => hasCondition ? map(x => x + 0.0001, missings || {}) : missings export let mergeAllMissing = missings => reduce(mergeWith(add), {}, map(collectNodeMissing, missings)) export let mergeMissing = (left, right) => mergeWith(add, left || {}, right || {}) export let evaluateNode = (cache, situationGate, parsedRules, node) => node.evaluate ? node.evaluate(cache, situationGate, parsedRules, node) : node export let rewriteNode = (node, nodeValue, explanation, missingVariables) => ({ ...node, nodeValue, explanation, missingVariables }) export let evaluateArray = (reducer, start) => ( cache, situationGate, parsedRules, node ) => { let evaluateOne = child => evaluateNode(cache, situationGate, parsedRules, child), explanation = map(evaluateOne, node.explanation), values = pluck('nodeValue', explanation), nodeValue = any(equals(null), values) ? null : reduce(reducer, start, values), missingVariables = node.nodeValue == null ? mergeAllMissing(explanation) : {} // console.log("".padStart(cache.parseLevel), missingVariables) return rewriteNode(node, nodeValue, explanation, missingVariables) } export let evaluateArrayWithFilter = (evaluationFilter, reducer, start) => ( cache, situationGate, parsedRules, node ) => { let evaluateOne = child => evaluateNode(cache, situationGate, parsedRules, child), explanation = map( evaluateOne, filter(evaluationFilter(situationGate), node.explanation) ), values = pluck('nodeValue', explanation), nodeValue = any(equals(null), values) ? null : reduce(reducer, start, values), missingVariables = node.nodeValue == null ? mergeAllMissing(explanation) : {} return rewriteNode(node, nodeValue, explanation, missingVariables) } export let defaultNode = nodeValue => ({ nodeValue, // eslint-disable-next-line jsx: nodeValue => {nodeValue}, isDefault: true }) export let parseObject = (recurse, objectShape, value) => { let recurseOne = key => defaultValue => { if (!value[key] && !defaultValue) throw new Error( `Il manque une valeur '${key}' dans ${JSON.stringify(value)} ` ) return value[key] ? recurse(value[key]) : defaultValue } let transforms = fromPairs(map(k => [k, recurseOne(k)], keys(objectShape))) return evolve(transforms, objectShape) } export let evaluateObject = (objectShape, effect) => ( cache, situationGate, parsedRules, node ) => { let evaluateOne = child => evaluateNode(cache, situationGate, parsedRules, child) let transforms = map(k => [k, evaluateOne], keys(objectShape)), automaticExplanation = evolve(fromPairs(transforms))(node.explanation) // the result of effect can either be just a nodeValue, or an object {additionalExplanation, nodeValue}. The latter is useful for a richer JSX visualisation of the mecanism : the view should not duplicate code to recompute intermediate values (e.g. for a marginal 'barème', the marginal 'tranche') let evaluated = effect(automaticExplanation), explanation = is(Object, evaluated) ? { ...automaticExplanation, ...evaluated.additionalExplanation } : automaticExplanation, nodeValue = is(Object, evaluated) ? evaluated.nodeValue : evaluated, missingVariables = mergeAllMissing(values(explanation)) // console.log("".padStart(cache.parseLevel),map(node => length(flatten(collectNodeMissing(node))) ,explanation)) return rewriteNode(node, nodeValue, explanation, missingVariables) } export let E = (cache, situationGate, parsedRules) => { let missingVariables = {} let val = element => { let evaluated = evaluateNode(cache, situationGate, parsedRules, element) // automatically add missing variables when a variable is evaluated and thus needed in this mecanism's evaluation missingVariables = mergeMissing( missingVariables, evaluated.missingVariables ) return evaluated.nodeValue } return { val, missingVariables: () => missingVariables } }