mon-entreprise/source/engine/parseReference.js

257 lines
7.2 KiB
JavaScript
Raw Normal View History

// Reference to a variable
2019-02-06 07:44:48 +00:00
import React from 'react'
import { Trans } from 'react-i18next'
import { evaluateNode, makeJsx, rewriteNode } from './evaluation'
import { Leaf, Node } from './mecanismViews/common'
import {
disambiguateRuleReference,
findParentDependency,
findRuleByDottedName
} from './rules'
2019-07-09 09:12:34 +00:00
import { getSituationValue } from './getSituationValue'
import parseRule from 'Engine/parseRule'
export let parseReference = (rules, rule, parsedRules, filter) => ({
fragments
}) => {
let partialReference = fragments.join(' . '),
dottedName = disambiguateRuleReference(rules, rule, partialReference)
2019-06-14 08:43:09 +00:00
let parsedRule =
parsedRules[dottedName] ||
parseRule(rules, findRuleByDottedName(rules, dottedName), parsedRules)
let evaluate = (cache, situation, parsedRules, node) => {
let dottedName = node.dottedName,
// On va vérifier dans le cache courant, dict, si la variable n'a pas été déjà évaluée
// En effet, l'évaluation dans le cas d'une variable qui a une formule, est coûteuse !
cacheName = dottedName + (filter ? '.' + filter : ''),
cached = cache[cacheName]
if (cached) return cached
2019-06-14 08:43:09 +00:00
let variable =
typeof parsedRule === 'object' ? parsedRule : parsedRules[dottedName],
variableHasFormula = variable.formule != null,
variableHasCond =
variable['applicable si'] != null ||
variable['non applicable si'] != null ||
findParentDependency(parsedRules, variable),
situationValue = getSituationValue(situation, dottedName, variable),
needsEvaluation =
situationValue == null && (variableHasCond || variableHasFormula)
let explanation = needsEvaluation
? evaluateNode(cache, situation, parsedRules, variable)
: variable
let cacheAndNode = (nodeValue, missingVariables, customExplanation) => {
cache[cacheName] = rewriteNode(
node,
nodeValue,
customExplanation || explanation,
missingVariables
)
return cache[cacheName]
}
2019-02-06 07:44:48 +00:00
const variableScore = variable.defaultValue ? 1 : 2
// SITUATION 1 : La variable est directement renseignée
if (situationValue != null) {
return cacheAndNode(
situationValue,
{},
{ ...explanation, nodeValue: situationValue }
)
}
// SITUATION 2 : La variable est calculée
if (situationValue == null && variableHasFormula)
return cacheAndNode(explanation.nodeValue, explanation.missingVariables)
// SITUATION 3 : La variable est une question sans condition dont la valeur n'a pas été renseignée
if (situationValue == null && !variableHasFormula && !variableHasCond)
return cacheAndNode(null, { [dottedName]: variableScore })
// SITUATION 4 : La variable est une question avec conditions
if (situationValue == null && !variableHasFormula && variableHasCond) {
// SITUATION 4.1 : La condition est connue et vrai
if (explanation.isApplicable)
return variable.question
? cacheAndNode(null, { [dottedName]: variableScore })
: cacheAndNode(true, {})
// SITUATION 4.2 : La condition est connue et fausse
if (explanation.isApplicable === false) return cacheAndNode(false, {})
// SITUATION 4.3 : La condition n'est pas connue
return cacheAndNode(null, explanation.missingVariables)
}
}
return {
evaluate,
//eslint-disable-next-line react/display-name
jsx: nodeValue => (
<Leaf
classes="variable filtered"
filter={filter}
name={fragments.join(' . ')}
dottedName={dottedName}
value={nodeValue}
unit={parsedRule.unit}
/>
),
name: partialReference,
2019-07-01 15:59:57 +00:00
category: 'reference',
fragments,
dottedName,
unit: parsedRule.unit
}
}
// This function is a wrapper that can apply :
// - temporal transformations to the value of the variable.
// See the période.yaml test suite for details
// - filters on the variable to select one part of the variable's 'composantes'
export let parseReferenceTransforms = (
rules,
rule,
parsedRules
) => parseResult => {
let evaluateTransforms = originalEval => (
cache,
situation,
parsedRules,
node
) => {
// Filter transformation
let filteringSituation = name =>
name == 'sys.filter' ? parseResult.filter : situation(name)
let filteredNode = originalEval(
cache,
parseResult.filter ? filteringSituation : situation,
parsedRules,
node
)
let nodeValue = filteredNode.nodeValue
// Temporal transformation
let supportedPeriods = ['mois', 'année', 'flexible']
2018-11-07 16:55:42 +00:00
if (nodeValue == null) return filteredNode
let ruleToTransform = findRuleByDottedName(
2019-05-14 11:52:23 +00:00
parsedRules,
filteredNode.explanation.dottedName
)
2018-11-13 15:14:57 +00:00
let inlinePeriodTransform = { mensuel: 'mois', annuel: 'année' }[
parseResult.temporalTransform
]
// Exceptions
2018-11-13 15:14:57 +00:00
if (!rule.période && !inlinePeriodTransform) {
if (supportedPeriods.includes(ruleToTransform.période))
2018-11-07 16:55:42 +00:00
throw new Error(
`Attention, une variable sans période, ${
rule.dottedName
}, qui appelle une variable à période, ${
2018-11-07 16:55:42 +00:00
ruleToTransform.dottedName
}, c'est suspect !
Si la période de la variable appelée est neutralisée dans la formule de calcul, par exemple un montant mensuel divisé par 30 (comprendre 30 jours), utilisez "période: aucune" pour taire cette erreur et rassurer tout le monde.
2018-11-07 16:55:42 +00:00
`
)
return filteredNode
}
if (!ruleToTransform.période) return filteredNode
let environmentPeriod = situation('période') || 'mois'
2018-11-07 16:55:42 +00:00
let callingPeriod =
2019-02-06 07:44:48 +00:00
inlinePeriodTransform ||
(rule.période === 'flexible' ? environmentPeriod : rule.période)
2018-11-07 16:55:42 +00:00
let calledPeriod =
2019-02-06 07:44:48 +00:00
ruleToTransform.période === 'flexible'
? environmentPeriod
: ruleToTransform.période
let transformedNodeValue =
callingPeriod === 'mois' && calledPeriod === 'année'
? nodeValue / 12
: callingPeriod === 'année' && calledPeriod === 'mois'
2018-11-13 15:14:57 +00:00
? nodeValue * 12
: nodeValue,
periodTransform = nodeValue !== transformedNodeValue
let result = rewriteNode(
{
...filteredNode,
periodTransform: periodTransform,
...(periodTransform ? { originPeriodValue: nodeValue } : {})
},
transformedNodeValue,
filteredNode.explanation,
filteredNode.missingVariables
)
2019-02-14 17:00:52 +00:00
return result
}
let node = parseReference(rules, rule, parsedRules, parseResult.filter)(
parseResult.variable
)
return {
...node,
// Decorate node with the composante filter (either who is paying, either tax free)
...(parseResult.filter
? {
cotisation: {
...node.cotisation,
'dû par': parseResult.filter,
'impôt sur le revenu': parseResult.filter
}
}
: {}),
evaluate: evaluateTransforms(node.evaluate)
}
}
export let parseNegatedReference = variable => {
let evaluate = (cache, situation, parsedRules, node) => {
let explanation = evaluateNode(
cache,
situation,
parsedRules,
node.explanation
),
nodeValue = explanation.nodeValue == null ? null : !explanation.nodeValue,
missingVariables = explanation.missingVariables
return rewriteNode(node, nodeValue, explanation, missingVariables)
}
let jsx = (nodeValue, explanation) => (
<Node
classes="inlineExpression negation"
value={nodeValue}
child={
<span className="nodeContent">
<Trans i18nKey="inlineExpressionNegation">Non</Trans>{' '}
{makeJsx(explanation)}
</span>
}
/>
)
return {
evaluate,
jsx,
category: 'mecanism',
name: 'négation',
type: 'boolean',
explanation: variable
}
}