// This should be the new way to implement mecanisms // In a specific file // TODO import them automatically // TODO convert the legacy functions to new files import barème from 'Engine/mecanisms/barème.js' import barèmeContinu from 'Engine/mecanisms/barème-continu.js' import barèmeLinéaire from 'Engine/mecanisms/barème-linéaire.js' import { Parser } from 'nearley' import { add, curry, divide, equals, gt, gte, keys, without, lt, lte, map, multiply, propOr, subtract, fromPairs, is, cond, T } from 'ramda' import React from 'react' import { evaluateNode, makeJsx, mergeMissing, rewriteNode } from './evaluation' import Grammar from './grammar.ne' import { mecanismAllOf, mecanismComplement, mecanismError, mecanismInversion, mecanismMax, mecanismMin, mecanismNumericalSwitch, mecanismOneOf, mecanismProduct, mecanismReduction, mecanismSum, mecanismSynchronisation, mecanismVariations, mecanismOnePossibility } from './mecanisms' import { Node } from './mecanismViews/common' import { parseReferenceTransforms } from './parseReference' import { inferUnit } from 'Engine/units' export let parse = (rules, rule, parsedRules) => rawNode => { let onNodeType = cond([ [is(String), parseString(rules, rule, parsedRules)], [is(Number), parseNumber], [is(Object), parseObject(rules, rule, parsedRules)], [T, parseOther] ]) let defaultEvaluate = (cache, situationGate, parsedRules, node) => node let parsedNode = onNodeType(rawNode) return parsedNode.evaluate ? parsedNode : { ...parsedNode, evaluate: defaultEvaluate } } export let nearley = () => new Parser(Grammar.ParserRules, Grammar.ParserStart) export let parseString = (rules, rule, parsedRules) => rawNode => { /* Strings correspond to infix expressions. * Indeed, a subset of expressions like simple arithmetic operations `3 + (quantity * 2)` or like `salary [month]` are more explicit that their prefixed counterparts. * This function makes them prefixed operations. */ let [parseResult] = nearley().feed(rawNode).results return parseObject(rules, rule, parsedRules)(parseResult) } export let parseNumber = rawNode => ({ text: '' + rawNode, category: 'number', nodeValue: rawNode, type: 'numeric', jsx: {rawNode} }) export let parseOther = rawNode => { throw new Error( 'Cette donnée : ' + rawNode + ' doit être un Number, String ou Object' ) } export let parseObject = (rules, rule, parsedRules) => rawNode => { /* TODO instead of describing mecanisms in knownMecanisms.yaml, externalize the mecanisms themselves in an individual file and describe it let mecanisms = intersection(keys(rawNode), keys(knownMecanisms)) if (mecanisms.length != 1) { } */ let attributes = keys(rawNode), descriptiveAttributes = ['description', 'note', 'référence'], relevantAttributes = without(descriptiveAttributes, attributes) if (relevantAttributes.length !== 1) throw new Error(`OUPS : On ne devrait reconnaître que un et un seul mécanisme dans cet objet (au-delà des attributs descriptifs tels que "description", "commentaire", etc.) Objet YAML : ${JSON.stringify(rawNode)} Cette liste doit avoir un et un seul élément. Si vous venez tout juste d'ajouter un nouveau mécanisme, vérifier qu'il est bien intégré dans le dispatch de parse.js `) let k = relevantAttributes[0], v = rawNode[k] let knownOperations = { '*': [multiply, '∗'], '/': [divide, '∕'], '+': [add], '-': [subtract, '−'], '<': [lt], '<=': [lte, '≤'], '>': [gt], '>=': [gte, '≥'], '=': [equals], '!=': [(a, b) => !equals(a, b), '≠'] }, operationDispatch = fromPairs( Object.entries(knownOperations).map(([k, [f, symbol]]) => [ k, mecanismOperation(k, f, symbol) ]) ) let dispatch = { 'une de ces conditions': mecanismOneOf, 'toutes ces conditions': mecanismAllOf, 'aiguillage numérique': mecanismNumericalSwitch, somme: mecanismSum, multiplication: mecanismProduct, barème, 'barème linéaire': barèmeLinéaire, 'barème continu': barèmeContinu, 'le maximum de': mecanismMax, 'le minimum de': mecanismMin, complément: mecanismComplement, 'une possibilité': mecanismOnePossibility(rule.dottedName), 'inversion numérique': mecanismInversion(rule.dottedName), allègement: mecanismReduction, variations: mecanismVariations, synchronisation: mecanismSynchronisation, ...operationDispatch, filter: () => parseReferenceTransforms(rules, rule, parsedRules)({ filter: v.filter, variable: v.explanation }), variable: () => parseReferenceTransforms(rules, rule, parsedRules)({ variable: v }), temporalTransform: () => parseReferenceTransforms(rules, rule, parsedRules)({ variable: v.explanation, temporalTransform: v.temporalTransform }), constant: () => ({ type: v.type, nodeValue: v.nodeValue, unit: v.unit, // eslint-disable-next-line jsx: () => {v.rawNode} }) }, action = propOr(mecanismError, k, dispatch) return action(parse(rules, rule, parsedRules), k, v) } let mecanismOperation = (k, operatorFunction, symbol) => (recurse, k, v) => { let evaluate = (cache, situation, parsedRules, node) => { let explanation = map( curry(evaluateNode)(cache, situation, parsedRules), node.explanation ), value1 = explanation[0].nodeValue, value2 = explanation[1].nodeValue, nodeValue = value1 == null || value2 == null ? null : operatorFunction(value1, value2), missingVariables = mergeMissing( explanation[0].missingVariables, explanation[1].missingVariables ) return rewriteNode(node, nodeValue, explanation, missingVariables) } let explanation = v.explanation.map(recurse) let unit = inferUnit(k, [explanation[0].unit, explanation[1].unit]) let jsx = (nodeValue, explanation) => ( {makeJsx(explanation[0])} {symbol || k} {makeJsx(explanation[1])} } /> ) return { ...v, evaluate, jsx, operator: symbol || k, // is this useful ? text: rawNode, explanation, unit } }