import React from 'react' import { Parser } from 'nearley' import Grammar from './grammar.ne' import { contains, propEq, curry, cond, equals, divide, multiply, map, intersection, keys, propOr, always, head, gte, lte, lt, gt, add, subtract } from 'ramda' import { evaluateNode, rewriteNode, makeJsx, mergeMissing } from './evaluation' import { Node } from './mecanismViews/common' import { treatVariable, treatNegatedVariable, treatFilteredVariable } from './treatVariable' import { treat } from './traverse' import knownMecanisms from './known-mecanisms.yaml' import { mecanismOneOf, mecanismAllOf, mecanismNumericalSwitch, mecanismSum, mecanismProduct, mecanismScale, mecanismMax, mecanismMin, mecanismError, mecanismComplement, mecanismSelection, mecanismInversion, mecanismReduction } from './mecanisms' let nearley = () => new Parser(Grammar.ParserRules, Grammar.ParserStart) export let treatString = (rules, rule) => rawNode => { /* On a affaire à un string, donc à une expression infixe. Elle sera traité avec le parser obtenu grâce à NearleyJs et notre grammaire `grammar.ne`. On obtient un objet de type Variable (avec potentiellement un 'modifier', par exemple temporel (TODO)), CalcExpression ou Comparison. Cet objet est alors rebalancé à 'treat'. */ let [parseResult, ...additionnalResults] = nearley().feed(rawNode).results if (additionnalResults && additionnalResults.length > 0) throw "Attention ! L'expression <" + rawNode + '> ne peut être traitée de façon univoque' if ( !contains(parseResult.category)([ 'variable', 'calcExpression', 'filteredVariable', 'comparison', 'negatedVariable', 'percentage' ]) ) throw "Attention ! Erreur de traitement de l'expression : " + rawNode if (parseResult.category == 'variable') return treatVariable(rules, rule)(parseResult) if (parseResult.category == 'filteredVariable') { return treatFilteredVariable(rules, rule)( parseResult.filter, parseResult.variable ) } if (parseResult.category == 'negatedVariable') return treatNegatedVariable( treatVariable(rules, rule)(parseResult.variable) ) // We don't need to handle category == 'value' because YAML then returns it as // numerical value, not a String: it goes to treatNumber if (parseResult.category == 'percentage') { return { nodeValue: parseResult.nodeValue, // eslint-disable-next-line jsx: () => {rawNode} } } if ( parseResult.category == 'calcExpression' || parseResult.category == 'comparison' ) { let evaluate = (cache, situation, parsedRules, node) => { let operatorFunction = { '*': multiply, '/': divide, '+': add, '-': subtract, '<': lt, '<=': lte, '>': gt, '>=': gte, '=': equals, '!=': (a, b) => !equals(a, b) }[node.operator], 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 treatFilteredVariableClosure = parseResult => treatFilteredVariable(rules, rule)( parseResult.filter, parseResult.variable ) let explanation = parseResult.explanation.map( cond([ [propEq('category', 'variable'), treatVariable(rules, rule)], [ propEq('category', 'filteredVariable'), treatFilteredVariableClosure ], [ propEq('category', 'value'), node => ({ nodeValue: node.nodeValue, // eslint-disable-next-line jsx: nodeValue => {nodeValue} }) ], [ propEq('category', 'percentage'), node => ({ nodeValue: node.nodeValue, // eslint-disable-next-line jsx: nodeValue => ( {nodeValue * 100}% ) }) ] ]) ), operator = parseResult.operator let operatorToUnicode = operator => ({ '>=': '≥', '<=': '≤', '!=': '≠', '*': '∗', '/': '∕', '-': '−' }[operator] || operator) let jsx = (nodeValue, explanation) => ( {makeJsx(explanation[0])} {operatorToUnicode(parseResult.operator)} {makeJsx(explanation[1])} } /> ) return { evaluate, jsx, operator, text: rawNode, category: parseResult.category, type: parseResult.category == 'calcExpression' ? 'numeric' : 'boolean', explanation } } } export let treatNumber = rawNode => { return { text: '' + rawNode, category: 'number', nodeValue: rawNode, type: 'numeric', jsx: {rawNode} } } export let treatOther = rawNode => { throw new Error( 'Cette donnée : ' + rawNode + ' doit être un Number, String ou Object' ) } export let treatObject = (rules, rule) => rawNode => { let mecanisms = intersection(keys(rawNode), keys(knownMecanisms)) if (mecanisms.length != 1) { // eslint-disable-next-line no-console console.log( 'Erreur : On ne devrait reconnaître que un et un seul mécanisme dans cet objet', mecanisms, rawNode ) throw new Error('OUPS !') } let k = head(mecanisms), v = rawNode[k] let dispatch = { 'une de ces conditions': mecanismOneOf, 'toutes ces conditions': mecanismAllOf, 'aiguillage numérique': mecanismNumericalSwitch, somme: mecanismSum, multiplication: mecanismProduct, barème: mecanismScale, 'le maximum de': mecanismMax, 'le minimum de': mecanismMin, complément: mecanismComplement, sélection: mecanismSelection, 'une possibilité': always({ ...v, 'une possibilité': 'oui', missingVariables: { [rule.dottedName]: 1 } }), inversion: mecanismInversion(rule.dottedName), allègement: mecanismReduction }, action = propOr(mecanismError, k, dispatch) return action(treat(rules, rule), k, v) }