2020-08-31 05:39:31 +00:00
|
|
|
import { map, pick, pipe } from 'ramda'
|
2019-11-28 11:03:23 +00:00
|
|
|
import { typeWarning } from './error'
|
2020-06-08 15:43:55 +00:00
|
|
|
import {
|
|
|
|
bonus,
|
|
|
|
evaluateNode,
|
|
|
|
mergeMissing,
|
|
|
|
mergeAllMissing
|
|
|
|
} from './evaluation'
|
2019-11-28 11:03:23 +00:00
|
|
|
import { convertNodeToUnit } from './nodeUnits'
|
2020-09-09 11:31:34 +00:00
|
|
|
import { EvaluatedNode, ParsedRule } from './types'
|
2019-06-13 16:01:49 +00:00
|
|
|
|
2019-11-04 13:07:19 +00:00
|
|
|
export const evaluateApplicability = (
|
|
|
|
cache,
|
2020-05-06 07:23:21 +00:00
|
|
|
situation,
|
2019-11-04 13:07:19 +00:00
|
|
|
parsedRules,
|
2020-03-30 17:14:03 +00:00
|
|
|
node: ParsedRule
|
2020-09-09 11:31:34 +00:00
|
|
|
): EvaluatedNode => {
|
2020-04-30 15:13:45 +00:00
|
|
|
const evaluatedAttributes = pipe(
|
2020-03-10 10:52:53 +00:00
|
|
|
pick(['non applicable si', 'applicable si', 'rendu non applicable']) as (
|
|
|
|
x: any
|
|
|
|
) => any,
|
2020-05-06 07:23:21 +00:00
|
|
|
map(value => evaluateNode(cache, situation, parsedRules, value))
|
2020-03-10 10:52:53 +00:00
|
|
|
)(node) as any,
|
2019-06-13 16:01:49 +00:00
|
|
|
{
|
|
|
|
'non applicable si': notApplicable,
|
2019-07-29 07:13:05 +00:00
|
|
|
'applicable si': applicable,
|
2019-08-19 07:36:27 +00:00
|
|
|
'rendu non applicable': disabled
|
2019-06-13 16:01:49 +00:00
|
|
|
} = evaluatedAttributes,
|
2019-11-07 11:34:03 +00:00
|
|
|
parentDependencies = node.parentDependencies.map(parent =>
|
2020-05-06 07:23:21 +00:00
|
|
|
evaluateNode(cache, situation, parsedRules, parent)
|
2020-05-06 20:08:49 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const anyDisabledParent = parentDependencies.find(
|
|
|
|
parent => parent?.nodeValue === false
|
|
|
|
)
|
|
|
|
|
2020-05-15 04:05:46 +00:00
|
|
|
const { nodeValue, missingVariables = {} } = anyDisabledParent
|
2020-05-06 20:08:49 +00:00
|
|
|
? anyDisabledParent
|
|
|
|
: notApplicable?.nodeValue === true
|
|
|
|
? {
|
|
|
|
nodeValue: false,
|
|
|
|
missingVariables: notApplicable.missingVariables
|
|
|
|
}
|
2020-08-31 05:39:31 +00:00
|
|
|
: disabled?.nodeValue === true
|
|
|
|
? { nodeValue: false }
|
2020-05-06 20:08:49 +00:00
|
|
|
: applicable?.nodeValue === false
|
|
|
|
? { nodeValue: false, missingVariables: applicable.missingVariables }
|
|
|
|
: {
|
|
|
|
nodeValue: [notApplicable, applicable, ...parentDependencies].some(
|
|
|
|
n => n?.nodeValue === null
|
|
|
|
)
|
|
|
|
? null
|
|
|
|
: !notApplicable?.nodeValue &&
|
|
|
|
(applicable?.nodeValue == undefined || !!applicable?.nodeValue),
|
2020-06-08 15:43:55 +00:00
|
|
|
missingVariables: mergeAllMissing(
|
|
|
|
[...parentDependencies, notApplicable, disabled, applicable].filter(
|
|
|
|
Boolean
|
|
|
|
)
|
|
|
|
)
|
2020-05-06 20:08:49 +00:00
|
|
|
}
|
2019-11-11 18:06:49 +00:00
|
|
|
|
2019-11-04 13:07:19 +00:00
|
|
|
return {
|
2020-04-23 07:30:03 +00:00
|
|
|
...node,
|
2020-05-06 20:08:49 +00:00
|
|
|
nodeValue,
|
|
|
|
isApplicable: nodeValue,
|
2019-11-04 13:07:19 +00:00
|
|
|
missingVariables,
|
2020-04-23 07:30:03 +00:00
|
|
|
parentDependencies,
|
2019-11-07 11:34:03 +00:00
|
|
|
...evaluatedAttributes
|
2019-11-04 13:07:19 +00:00
|
|
|
}
|
2019-11-04 10:03:01 +00:00
|
|
|
}
|
2019-11-04 11:20:50 +00:00
|
|
|
|
2020-10-19 15:25:12 +00:00
|
|
|
export const evaluateFormula = (cache, situation, parsedRules, node) => {
|
|
|
|
const explanation = evaluateNode(
|
|
|
|
cache,
|
|
|
|
situation,
|
|
|
|
parsedRules,
|
|
|
|
node.explanation
|
|
|
|
),
|
|
|
|
{ nodeValue, unit, missingVariables, temporalValue } = explanation
|
|
|
|
|
|
|
|
return {
|
|
|
|
...node,
|
|
|
|
nodeValue,
|
|
|
|
unit,
|
|
|
|
missingVariables,
|
|
|
|
explanation,
|
|
|
|
temporalValue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export const evaluateRule = (cache, situation, parsedRules, node) => {
|
2019-11-28 11:03:23 +00:00
|
|
|
cache._meta.contextRule.push(node.dottedName)
|
2020-03-10 10:52:53 +00:00
|
|
|
const applicabilityEvaluation = evaluateApplicability(
|
|
|
|
cache,
|
2020-05-06 07:23:21 +00:00
|
|
|
situation,
|
2020-03-10 10:52:53 +00:00
|
|
|
parsedRules,
|
|
|
|
node
|
|
|
|
)
|
|
|
|
const {
|
|
|
|
missingVariables: condMissing,
|
|
|
|
nodeValue: isApplicable
|
|
|
|
} = applicabilityEvaluation
|
|
|
|
|
|
|
|
// evaluate the formula lazily, only if the applicability is known and true
|
2020-06-08 15:43:55 +00:00
|
|
|
let evaluatedFormula =
|
|
|
|
isApplicable && node.formule
|
|
|
|
? evaluateNode(cache, situation, parsedRules, node.formule)
|
|
|
|
: node.formule
|
2019-11-28 11:03:23 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
if (node.unit) {
|
2019-11-28 11:03:23 +00:00
|
|
|
try {
|
2020-04-23 07:30:03 +00:00
|
|
|
evaluatedFormula = convertNodeToUnit(node.unit, evaluatedFormula)
|
2019-11-28 11:03:23 +00:00
|
|
|
} catch (e) {
|
|
|
|
typeWarning(
|
|
|
|
node.dottedName,
|
2020-04-30 15:13:45 +00:00
|
|
|
"L'unité de la règle est incompatible avec celle de sa formule",
|
2019-11-28 11:03:23 +00:00
|
|
|
e
|
|
|
|
)
|
|
|
|
}
|
2019-11-11 18:06:49 +00:00
|
|
|
}
|
2020-04-23 07:30:03 +00:00
|
|
|
const missingVariables = mergeMissing(
|
|
|
|
bonus(condMissing, !!Object.keys(condMissing).length),
|
|
|
|
evaluatedFormula.missingVariables
|
|
|
|
)
|
2019-11-28 11:03:23 +00:00
|
|
|
|
2020-04-30 15:13:45 +00:00
|
|
|
const temporalValue = evaluatedFormula.temporalValue
|
2019-11-28 11:03:23 +00:00
|
|
|
cache._meta.contextRule.pop()
|
2019-06-13 16:01:49 +00:00
|
|
|
return {
|
|
|
|
...node,
|
2019-11-04 10:03:01 +00:00
|
|
|
...applicabilityEvaluation,
|
2019-11-28 11:03:23 +00:00
|
|
|
...(node.formule && { formule: evaluatedFormula }),
|
2020-04-23 07:30:03 +00:00
|
|
|
nodeValue: evaluatedFormula.nodeValue,
|
|
|
|
unit: node.unit ?? evaluatedFormula.unit,
|
2020-02-16 18:56:07 +00:00
|
|
|
temporalValue,
|
2019-06-13 16:01:49 +00:00
|
|
|
isApplicable,
|
2019-11-04 13:07:19 +00:00
|
|
|
missingVariables
|
2019-06-13 16:01:49 +00:00
|
|
|
}
|
|
|
|
}
|
2020-10-19 15:25:12 +00:00
|
|
|
|
|
|
|
export 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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export const evaluateCondition = (cache, situation, parsedRules, node) => {
|
|
|
|
const explanation = evaluateNode(
|
|
|
|
cache,
|
|
|
|
situation,
|
|
|
|
parsedRules,
|
|
|
|
node.explanation
|
|
|
|
)
|
|
|
|
const nodeValue = explanation.nodeValue
|
|
|
|
const missingVariables = explanation.missingVariables
|
|
|
|
|
|
|
|
return { ...node, nodeValue, explanation, missingVariables }
|
|
|
|
}
|