2019-06-13 16:17:22 +00:00
|
|
|
import { evolve, map } from 'ramda'
|
|
|
|
import React from 'react'
|
2020-01-21 18:15:23 +00:00
|
|
|
import { Trans } from 'react-i18next'
|
2020-05-19 10:01:42 +00:00
|
|
|
import { Mecanism } from './components/mecanisms/common'
|
2020-05-08 10:04:00 +00:00
|
|
|
import { RuleLinkWithContext } from './components/RuleLink'
|
2020-09-27 15:42:12 +00:00
|
|
|
import { compilationError, warning } from './error'
|
2020-03-10 10:52:53 +00:00
|
|
|
import evaluate from './evaluateRule'
|
2019-11-18 18:17:44 +00:00
|
|
|
import { evaluateNode, makeJsx, mergeAllMissing } from './evaluation'
|
2020-03-10 10:52:53 +00:00
|
|
|
import { parse } from './parse'
|
2020-03-26 15:03:19 +00:00
|
|
|
import {
|
|
|
|
disambiguateRuleReference,
|
|
|
|
findParentDependencies,
|
|
|
|
nameLeaf
|
|
|
|
} from './ruleUtils'
|
2020-03-30 17:14:03 +00:00
|
|
|
import { ParsedRule, Rule, Rules } from './types'
|
2020-09-27 15:42:12 +00:00
|
|
|
import {
|
|
|
|
areUnitConvertible,
|
|
|
|
parseUnit,
|
|
|
|
serializeUnit,
|
|
|
|
simplifyUnit
|
|
|
|
} from './units'
|
2020-05-08 10:04:00 +00:00
|
|
|
import { capitalise0, coerceArray } from './utils'
|
2019-06-13 16:17:22 +00:00
|
|
|
|
2020-03-30 17:14:03 +00:00
|
|
|
export default function<Names extends string>(
|
|
|
|
rules: Rules<Names>,
|
|
|
|
dottedName,
|
|
|
|
parsedRules
|
|
|
|
): ParsedRule<Names> {
|
2020-03-26 15:03:19 +00:00
|
|
|
if (parsedRules[dottedName]) return parsedRules[dottedName]
|
2020-03-30 17:14:03 +00:00
|
|
|
|
2020-03-26 15:03:19 +00:00
|
|
|
parsedRules[dottedName] = 'being parsed'
|
2019-06-13 16:01:49 +00:00
|
|
|
/*
|
2019-12-13 16:22:18 +00:00
|
|
|
The parseRule function will traverse the tree of the `rule` and produce an
|
|
|
|
AST, an object containing other objects containing other objects... Some of
|
|
|
|
the attributes of the rule are dynamic, they need to be parsed. It is the
|
|
|
|
case of `non applicable si`, `applicable si`, `formule`. These attributes'
|
|
|
|
values themselves may have mechanism properties (e. g. `barème`) or inline
|
|
|
|
expressions (e. g. `maVariable + 3`). These mechanisms or variables are in
|
|
|
|
turn traversed by `parse()`. During this processing, 'evaluate' and'jsx'
|
|
|
|
functions are attached to the objects of the AST. They will be evaluated
|
|
|
|
during the evaluation phase, called "analyse".
|
|
|
|
*/
|
2020-03-30 17:14:03 +00:00
|
|
|
|
2020-04-30 15:13:45 +00:00
|
|
|
const parentDependencies = findParentDependencies(rules, dottedName)
|
2020-03-30 17:14:03 +00:00
|
|
|
let rawRule = rules[dottedName]
|
|
|
|
if (rawRule == null) {
|
|
|
|
rawRule = {}
|
|
|
|
}
|
|
|
|
if (typeof rawRule === 'string') {
|
|
|
|
rawRule = {
|
|
|
|
formule: rawRule
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rawRule as Rule
|
2020-03-26 15:03:19 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
if (
|
|
|
|
rawRule['par défaut'] &&
|
|
|
|
rawRule['formule'] &&
|
|
|
|
!rawRule.formule['une possibilité']
|
|
|
|
) {
|
|
|
|
throw new warning(
|
|
|
|
dottedName,
|
|
|
|
'Une règle ne peut pas avoir à la fois une formule ET une valeur par défaut.'
|
2020-03-26 15:03:19 +00:00
|
|
|
)
|
|
|
|
}
|
2019-06-13 16:01:49 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
const name = nameLeaf(dottedName)
|
2020-04-30 15:13:45 +00:00
|
|
|
const unit = rawRule.unité != null ? parseUnit(rawRule.unité) : undefined
|
2020-04-23 07:30:03 +00:00
|
|
|
|
2020-03-30 17:14:03 +00:00
|
|
|
const rule = {
|
|
|
|
...rawRule,
|
2020-05-27 09:40:25 +00:00
|
|
|
rawRule,
|
2020-03-26 15:03:19 +00:00
|
|
|
name,
|
2020-03-30 17:14:03 +00:00
|
|
|
dottedName,
|
|
|
|
type: rawRule.type,
|
|
|
|
title: capitalise0(rawRule['titre'] || name),
|
|
|
|
examples: rawRule['exemples'],
|
|
|
|
icons: rawRule['icônes'],
|
|
|
|
summary: rawRule['résumé'],
|
2020-03-26 15:03:19 +00:00
|
|
|
unit,
|
2020-04-23 07:30:03 +00:00
|
|
|
parentDependencies,
|
|
|
|
defaultValue: rawRule['par défaut']
|
2020-03-26 15:03:19 +00:00
|
|
|
}
|
2019-11-11 15:50:05 +00:00
|
|
|
|
2020-04-30 15:13:45 +00:00
|
|
|
const parsedRule = evolve({
|
2019-06-13 16:01:49 +00:00
|
|
|
// Voilà les attributs d'une règle qui sont aujourd'hui dynamiques, donc à traiter
|
|
|
|
// Les métadonnées d'une règle n'en font pas aujourd'hui partie
|
|
|
|
|
|
|
|
// condition d'applicabilité de la règle
|
2019-11-07 11:34:03 +00:00
|
|
|
parentDependencies: parents =>
|
|
|
|
parents.map(parent => {
|
2020-04-30 15:13:45 +00:00
|
|
|
const node = parse(rules, rule, parsedRules)(parent)
|
2020-03-30 17:14:03 +00:00
|
|
|
|
2020-04-30 15:13:45 +00:00
|
|
|
const jsx = ({ nodeValue, explanation }) =>
|
2020-04-23 07:30:03 +00:00
|
|
|
nodeValue === null ? (
|
|
|
|
<div>Active seulement si {makeJsx(explanation)}</div>
|
|
|
|
) : nodeValue === true ? (
|
|
|
|
<div>Active car {makeJsx(explanation)}</div>
|
|
|
|
) : nodeValue === false ? (
|
|
|
|
<div>Non active car {makeJsx(explanation)}</div>
|
|
|
|
) : null
|
2019-06-13 16:01:49 +00:00
|
|
|
|
2019-11-07 11:34:03 +00:00
|
|
|
return {
|
|
|
|
evaluate: (cache, situation, parsedRules) =>
|
|
|
|
node.evaluate(cache, situation, parsedRules, node),
|
|
|
|
jsx,
|
|
|
|
category: 'ruleProp',
|
|
|
|
rulePropType: 'cond',
|
|
|
|
name: 'parentDependencies',
|
|
|
|
type: 'numeric',
|
|
|
|
explanation: node
|
|
|
|
}
|
|
|
|
}),
|
2019-06-13 16:17:22 +00:00
|
|
|
'non applicable si': evolveCond(
|
|
|
|
'non applicable si',
|
|
|
|
rule,
|
|
|
|
rules,
|
|
|
|
parsedRules
|
|
|
|
),
|
|
|
|
'applicable si': evolveCond('applicable si', rule, rules, parsedRules),
|
2019-08-19 07:36:27 +00:00
|
|
|
'rend non applicable': nonApplicableRules =>
|
2019-11-04 17:06:43 +00:00
|
|
|
coerceArray(nonApplicableRules).map(referenceName => {
|
2020-03-30 17:14:03 +00:00
|
|
|
return disambiguateRuleReference(rules, dottedName, referenceName)
|
2019-08-19 07:36:27 +00:00
|
|
|
}),
|
2019-11-04 15:09:09 +00:00
|
|
|
remplace: evolveReplacement(rules, rule, parsedRules),
|
2020-04-23 07:30:03 +00:00
|
|
|
defaultValue: value =>
|
2020-09-18 11:08:38 +00:00
|
|
|
typeof value === 'string' || typeof value === 'number'
|
2020-04-23 07:30:03 +00:00
|
|
|
? parse(rules, rule, parsedRules)(value)
|
2020-09-09 10:10:18 +00:00
|
|
|
: // TODO : An "object" default value is only used in the
|
|
|
|
// "synchronisation" mecanism. This should be refactored to not use the
|
|
|
|
// attribute "defaultValue"
|
|
|
|
typeof value === 'object'
|
|
|
|
? { ...value, evaluate: () => value }
|
2020-04-23 07:30:03 +00:00
|
|
|
: value,
|
2019-06-13 16:01:49 +00:00
|
|
|
formule: value => {
|
2020-04-30 15:13:45 +00:00
|
|
|
const child = parse(rules, rule, parsedRules)(value)
|
2019-06-13 16:01:49 +00:00
|
|
|
|
2020-04-30 15:13:45 +00:00
|
|
|
const jsx = ({ explanation }) => makeJsx(explanation)
|
2019-06-13 16:01:49 +00:00
|
|
|
|
|
|
|
return {
|
2020-09-28 16:10:11 +00:00
|
|
|
evaluate: evaluateFormula,
|
2019-06-13 16:01:49 +00:00
|
|
|
jsx,
|
|
|
|
category: 'ruleProp',
|
|
|
|
rulePropType: 'formula',
|
|
|
|
name: 'formule',
|
2019-11-28 11:03:23 +00:00
|
|
|
unit: child.unit,
|
2019-06-13 16:01:49 +00:00
|
|
|
explanation: child
|
|
|
|
}
|
2020-09-07 10:14:24 +00:00
|
|
|
}
|
2020-03-30 17:14:03 +00:00
|
|
|
})(rule)
|
2019-06-13 16:01:49 +00:00
|
|
|
|
2020-03-30 17:14:03 +00:00
|
|
|
parsedRules[dottedName] = {
|
|
|
|
// Pas de propriété explanation et jsx ici car on est parti du (mauvais)
|
|
|
|
// principe que 'non applicable si' et 'formule' sont particuliers, alors
|
|
|
|
// qu'ils pourraient être rangé avec les autres mécanismes
|
|
|
|
...parsedRule,
|
2019-06-13 16:01:49 +00:00
|
|
|
evaluate,
|
2019-06-19 09:54:47 +00:00
|
|
|
parsed: true,
|
2020-04-23 07:30:03 +00:00
|
|
|
unit:
|
|
|
|
parsedRule.unit ??
|
|
|
|
(parsedRule.formule?.unit && simplifyUnit(parsedRule.formule.unit)) ??
|
|
|
|
parsedRule.defaultValue?.unit,
|
2020-03-16 17:09:41 +00:00
|
|
|
isDisabledBy: [],
|
2019-11-28 11:03:23 +00:00
|
|
|
replacedBy: []
|
2019-06-13 16:01:49 +00:00
|
|
|
}
|
2020-03-30 17:14:03 +00:00
|
|
|
parsedRules[dottedName]['rendu non applicable'] = {
|
2020-09-28 16:10:11 +00:00
|
|
|
evaluate: evaluateDisabledBy,
|
2020-04-20 09:46:13 +00:00
|
|
|
jsx: ({ explanation: { isDisabledBy } }) => {
|
2019-08-19 07:36:27 +00:00
|
|
|
return (
|
|
|
|
isDisabledBy.length > 0 && (
|
|
|
|
<>
|
|
|
|
<h3>Exception{isDisabledBy.length > 1 && 's'}</h3>
|
|
|
|
<p>
|
2020-01-21 18:15:23 +00:00
|
|
|
<Trans>Cette règle ne s'applique pas pour</Trans> :{' '}
|
2019-08-19 07:36:27 +00:00
|
|
|
{isDisabledBy.map((rule, i) => (
|
2019-08-26 11:51:35 +00:00
|
|
|
<React.Fragment key={i}>
|
2019-08-19 07:36:27 +00:00
|
|
|
{i > 0 && ', '}
|
2020-05-08 10:04:00 +00:00
|
|
|
<RuleLinkWithContext dottedName={dottedName} />
|
2019-08-26 11:51:35 +00:00
|
|
|
</React.Fragment>
|
2019-08-19 07:36:27 +00:00
|
|
|
))}
|
|
|
|
</p>
|
|
|
|
</>
|
|
|
|
)
|
|
|
|
)
|
|
|
|
},
|
2019-07-29 07:13:05 +00:00
|
|
|
category: 'ruleProp',
|
|
|
|
rulePropType: 'cond',
|
2019-08-19 07:36:27 +00:00
|
|
|
name: 'rendu non applicable',
|
2019-07-29 07:13:05 +00:00
|
|
|
type: 'boolean',
|
2020-03-30 17:14:03 +00:00
|
|
|
explanation: parsedRules[dottedName]
|
2019-07-29 07:13:05 +00:00
|
|
|
}
|
2020-09-27 15:42:12 +00:00
|
|
|
|
|
|
|
if (process.env.NODE_ENV === 'development') {
|
|
|
|
Object.values(parsedRules[dottedName]['suggestions'] ?? {}).forEach(
|
|
|
|
suggestion => {
|
|
|
|
const parsedSuggestion = parse(rules, rule, parsedRules)(suggestion)
|
|
|
|
if (
|
|
|
|
!areUnitConvertible(
|
|
|
|
parsedRules[dottedName].unit,
|
|
|
|
parsedSuggestion.unit
|
|
|
|
) &&
|
|
|
|
parsedSuggestion.category !== 'reference'
|
|
|
|
) {
|
|
|
|
compilationError(
|
|
|
|
dottedName,
|
|
|
|
`La suggestion "${suggestion}" n'a pas une unité compatible avec la règle :
|
|
|
|
"${serializeUnit(parsedRules[dottedName].unit)}" et "${serializeUnit(
|
|
|
|
parsedSuggestion.unit
|
|
|
|
)}"`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
2020-03-30 17:14:03 +00:00
|
|
|
return parsedRules[dottedName]
|
2019-06-13 16:01:49 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 16:10:11 +00:00
|
|
|
const evaluateFormula = (cache, situation, parsedRules, node) => {
|
|
|
|
const explanation = evaluateNode(
|
|
|
|
cache,
|
|
|
|
situation,
|
|
|
|
parsedRules,
|
|
|
|
node.explanation
|
|
|
|
),
|
|
|
|
{ nodeValue, unit, missingVariables, temporalValue } = explanation
|
2019-07-29 07:13:05 +00:00
|
|
|
|
2020-09-28 16:10:11 +00:00
|
|
|
return {
|
|
|
|
...node,
|
|
|
|
nodeValue,
|
|
|
|
unit,
|
|
|
|
missingVariables,
|
|
|
|
explanation,
|
|
|
|
temporalValue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const evaluateDisabledBy = (cache, situation, parsedRules, node) => {
|
|
|
|
const isDisabledBy = node.explanation.isDisabledBy.map(disablerNode =>
|
|
|
|
evaluateNode(cache, situation, parsedRules, disablerNode)
|
|
|
|
)
|
|
|
|
const nodeValue = isDisabledBy.some(
|
|
|
|
x => x.nodeValue !== false && x.nodeValue !== null
|
|
|
|
)
|
|
|
|
const explanation = { ...node.explanation, isDisabledBy }
|
|
|
|
return {
|
|
|
|
...node,
|
|
|
|
explanation,
|
|
|
|
nodeValue,
|
|
|
|
missingVariables: mergeAllMissing(isDisabledBy)
|
2019-06-13 16:01:49 +00:00
|
|
|
}
|
2020-09-28 16:10:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const evaluateEvolveCond = (cache, situation, parsedRules, node) => {
|
|
|
|
const explanation = evaluateNode(
|
|
|
|
cache,
|
|
|
|
situation,
|
|
|
|
parsedRules,
|
|
|
|
node.explanation
|
|
|
|
),
|
|
|
|
nodeValue = explanation.nodeValue,
|
|
|
|
missingVariables = explanation.missingVariables
|
2019-06-13 16:01:49 +00:00
|
|
|
|
2020-09-28 16:10:11 +00:00
|
|
|
return { ...node, nodeValue, explanation, missingVariables }
|
|
|
|
}
|
|
|
|
|
|
|
|
const evolveCond = (dottedName, rule, rules, parsedRules) => value => {
|
2020-04-30 15:13:45 +00:00
|
|
|
const child = parse(rules, rule, parsedRules)(value)
|
2019-06-13 16:01:49 +00:00
|
|
|
|
2020-04-30 15:13:45 +00:00
|
|
|
const jsx = ({ nodeValue, explanation, unit }) => (
|
2020-05-19 10:01:42 +00:00
|
|
|
<Mecanism name={dottedName} value={nodeValue} unit={unit}>
|
2020-02-27 09:35:38 +00:00
|
|
|
{explanation.category === 'variable' ? (
|
|
|
|
<div className="node">{makeJsx(explanation)}</div>
|
|
|
|
) : (
|
|
|
|
makeJsx(explanation)
|
|
|
|
)}
|
2020-05-19 10:01:42 +00:00
|
|
|
</Mecanism>
|
2019-06-13 16:01:49 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
return {
|
2020-09-28 16:10:11 +00:00
|
|
|
evaluate: evaluateEvolveCond,
|
2019-06-13 16:01:49 +00:00
|
|
|
jsx,
|
|
|
|
category: 'ruleProp',
|
|
|
|
rulePropType: 'cond',
|
2020-03-30 17:14:03 +00:00
|
|
|
dottedName,
|
2019-06-13 16:01:49 +00:00
|
|
|
type: 'boolean',
|
|
|
|
explanation: child
|
|
|
|
}
|
|
|
|
}
|
2019-11-04 15:09:09 +00:00
|
|
|
|
2020-04-30 15:13:45 +00:00
|
|
|
const evolveReplacement = (rules, rule, parsedRules) => replacements =>
|
2019-11-04 15:09:09 +00:00
|
|
|
coerceArray(replacements).map(reference => {
|
|
|
|
const referenceName =
|
|
|
|
typeof reference === 'string' ? reference : reference.règle
|
|
|
|
let replacementNode = reference.par
|
|
|
|
if (replacementNode != null) {
|
|
|
|
replacementNode = parse(rules, rule, parsedRules)(replacementNode)
|
|
|
|
}
|
2020-04-30 15:13:45 +00:00
|
|
|
const [whiteListedNames, blackListedNames] = [
|
2019-11-04 15:09:09 +00:00
|
|
|
reference.dans,
|
|
|
|
reference['sauf dans']
|
|
|
|
]
|
2020-03-30 17:14:03 +00:00
|
|
|
.map(dottedName => dottedName && coerceArray(dottedName))
|
2019-11-04 15:09:09 +00:00
|
|
|
.map(
|
|
|
|
names =>
|
|
|
|
names &&
|
2020-03-30 17:14:03 +00:00
|
|
|
names.map(dottedName =>
|
|
|
|
disambiguateRuleReference(rules, rule.dottedName, dottedName)
|
2020-03-26 15:03:19 +00:00
|
|
|
)
|
2019-11-04 15:09:09 +00:00
|
|
|
)
|
2019-11-06 15:13:53 +00:00
|
|
|
|
2019-11-04 15:09:09 +00:00
|
|
|
return {
|
2020-03-26 15:03:19 +00:00
|
|
|
referenceName: disambiguateRuleReference(
|
|
|
|
rules,
|
|
|
|
rule.dottedName,
|
|
|
|
referenceName
|
|
|
|
),
|
2019-11-04 15:09:09 +00:00
|
|
|
replacementNode,
|
|
|
|
whiteListedNames,
|
|
|
|
blackListedNames
|
|
|
|
}
|
|
|
|
})
|