2020-11-04 15:47:12 +00:00
|
|
|
import Engine, { EvaluatedNode, evaluationFunction } from '.'
|
2020-10-19 15:25:12 +00:00
|
|
|
import { typeWarning } from './error'
|
|
|
|
import { evaluateApplicability } from './evaluateRule'
|
2020-11-04 15:47:12 +00:00
|
|
|
import { mergeMissing } from './evaluation'
|
2020-10-19 15:25:12 +00:00
|
|
|
import { convertNodeToUnit } from './nodeUnits'
|
2020-11-04 15:47:12 +00:00
|
|
|
import { ParsedRule } from './types'
|
|
|
|
import { areUnitConvertible, serializeUnit } from './units'
|
2020-10-19 15:25:12 +00:00
|
|
|
|
2020-11-04 15:47:12 +00:00
|
|
|
export const evaluateReference: evaluationFunction = function(node) {
|
|
|
|
const rule = this.parsedRules[node.dottedName]
|
2020-10-19 15:25:12 +00:00
|
|
|
// When a rule exists in different version (created using the `replace` mecanism), we add
|
|
|
|
// a redirection in the evaluation of references to use a potential active replacement
|
|
|
|
const [
|
|
|
|
applicableReplacements,
|
|
|
|
replacementMissingVariableList
|
2020-11-04 15:47:12 +00:00
|
|
|
] = getApplicableReplacements.call(
|
|
|
|
this,
|
2020-10-19 15:25:12 +00:00
|
|
|
node.explanation?.contextRuleName ?? '',
|
|
|
|
rule
|
|
|
|
)
|
|
|
|
if (applicableReplacements.length) {
|
|
|
|
if (applicableReplacements.length > 1) {
|
|
|
|
// eslint-disable-next-line no-console
|
|
|
|
console.warn(`
|
|
|
|
Règle ${rule.dottedName}: plusieurs remplacements valides ont été trouvés :
|
|
|
|
\n\t${applicableReplacements.map(node => node.rawNode).join('\n\t')}
|
|
|
|
|
|
|
|
Par défaut, seul le premier s'applique. Si vous voulez un autre comportement, vous pouvez :
|
|
|
|
- Restreindre son applicabilité via "applicable si" sur la règle de définition
|
|
|
|
- Restreindre sa portée en ajoutant une liste blanche (via le mot clé "dans") ou une liste noire (via le mot clé "sauf dans")
|
|
|
|
`)
|
|
|
|
}
|
2020-11-04 15:47:12 +00:00
|
|
|
return this.evaluateNode(applicableReplacements[0])
|
2020-10-19 15:25:12 +00:00
|
|
|
}
|
|
|
|
const addReplacementMissingVariable = node => ({
|
|
|
|
...node,
|
|
|
|
missingVariables: replacementMissingVariableList.reduce(
|
|
|
|
mergeMissing,
|
|
|
|
node.missingVariables
|
|
|
|
)
|
|
|
|
})
|
|
|
|
const 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 !
|
|
|
|
const cacheName =
|
|
|
|
dottedName + (node.explanation.filter ? ' .' + node.explanation.filter : '')
|
2020-11-04 15:47:12 +00:00
|
|
|
const cached = this.cache[cacheName]
|
2020-10-19 15:25:12 +00:00
|
|
|
|
|
|
|
if (cached) return addReplacementMissingVariable(cached)
|
|
|
|
|
|
|
|
const cacheNode = (
|
|
|
|
nodeValue: EvaluatedNode['nodeValue'],
|
|
|
|
missingVariables: EvaluatedNode['missingVariables'],
|
|
|
|
explanation?: Record<string, unknown>
|
|
|
|
) => {
|
2020-11-04 15:47:12 +00:00
|
|
|
this.cache[cacheName] = {
|
2020-10-19 15:25:12 +00:00
|
|
|
...node,
|
|
|
|
nodeValue,
|
|
|
|
...(explanation && {
|
|
|
|
explanation
|
|
|
|
}),
|
|
|
|
...(explanation?.temporalValue && {
|
|
|
|
temporalValue: explanation.temporalValue
|
|
|
|
}),
|
|
|
|
...(explanation?.unit && { unit: explanation.unit }),
|
|
|
|
missingVariables
|
|
|
|
}
|
2020-11-04 15:47:12 +00:00
|
|
|
return addReplacementMissingVariable(this.cache[cacheName])
|
2020-10-19 15:25:12 +00:00
|
|
|
}
|
2020-11-04 15:47:12 +00:00
|
|
|
const applicabilityEvaluation = evaluateApplicability.call(this, rule as any)
|
|
|
|
|
2020-10-19 15:25:12 +00:00
|
|
|
if (!applicabilityEvaluation.nodeValue) {
|
|
|
|
return cacheNode(
|
|
|
|
applicabilityEvaluation.nodeValue,
|
|
|
|
applicabilityEvaluation.missingVariables,
|
|
|
|
applicabilityEvaluation
|
|
|
|
)
|
|
|
|
}
|
2020-11-04 15:47:12 +00:00
|
|
|
if (this.parsedSituation[dottedName]) {
|
2020-10-19 15:25:12 +00:00
|
|
|
// Conditional evaluation is required because some mecanisms like
|
|
|
|
// "synchronisation" store raw JS objects in the situation.
|
2020-11-04 15:47:12 +00:00
|
|
|
const situationValue = this.parsedSituation[dottedName]?.nodeKind
|
|
|
|
? this.evaluateNode(this.parsedSituation[dottedName])
|
|
|
|
: this.parsedSituation[dottedName]
|
2020-10-19 15:25:12 +00:00
|
|
|
const unit =
|
|
|
|
!situationValue.unit || serializeUnit(situationValue.unit) === ''
|
|
|
|
? rule.unit
|
|
|
|
: situationValue.unit
|
|
|
|
return cacheNode(
|
|
|
|
situationValue?.nodeValue !== undefined
|
|
|
|
? situationValue.nodeValue
|
|
|
|
: situationValue,
|
|
|
|
applicabilityEvaluation.missingVariables,
|
|
|
|
{
|
|
|
|
...rule,
|
|
|
|
...(situationValue?.nodeValue !== undefined && situationValue),
|
|
|
|
unit
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rule.defaultValue != null) {
|
2020-11-04 15:47:12 +00:00
|
|
|
const evaluation = this.evaluateNode(rule.defaultValue)
|
2020-10-19 15:25:12 +00:00
|
|
|
return cacheNode(evaluation.nodeValue ?? evaluation, {
|
|
|
|
...evaluation.missingVariables,
|
|
|
|
[dottedName]: 1
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rule.formule != null) {
|
2020-11-04 15:47:12 +00:00
|
|
|
const evaluation = this.evaluateNode(rule)
|
2020-10-19 15:25:12 +00:00
|
|
|
return cacheNode(
|
|
|
|
evaluation.nodeValue,
|
|
|
|
evaluation.missingVariables,
|
|
|
|
evaluation
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return cacheNode(null, { [dottedName]: 2 })
|
|
|
|
}
|
|
|
|
|
|
|
|
// This function is a wrapper that can apply :
|
|
|
|
// - unit transformations to the value of the variable.
|
|
|
|
// See the unité-temporelle.yaml test suite for details
|
|
|
|
// - filters on the variable to select one part of the variable's 'composantes'
|
|
|
|
|
2020-11-04 15:47:12 +00:00
|
|
|
export const evaluateReferenceTransforms: evaluationFunction = function(node) {
|
2020-10-19 15:25:12 +00:00
|
|
|
// Filter transformation
|
2020-11-04 15:47:12 +00:00
|
|
|
if (node.explanation.filter) {
|
|
|
|
this.cache._meta.filter = node.explanation.filter
|
|
|
|
}
|
|
|
|
const filteredNode = this.evaluateNode(node.explanation.originalNode)
|
|
|
|
if (node.explanation.filter) {
|
|
|
|
delete this.cache._meta.filter
|
2020-10-19 15:25:12 +00:00
|
|
|
}
|
|
|
|
const { explanation, nodeValue } = filteredNode
|
|
|
|
if (!explanation || nodeValue === null) {
|
|
|
|
return filteredNode
|
|
|
|
}
|
|
|
|
const unit = node.explanation.unit
|
|
|
|
if (unit) {
|
|
|
|
try {
|
|
|
|
return convertNodeToUnit(unit, filteredNode)
|
|
|
|
} catch (e) {
|
|
|
|
typeWarning(
|
2020-11-04 15:47:12 +00:00
|
|
|
this.cache._meta.contextRule,
|
2020-10-19 15:25:12 +00:00
|
|
|
`Impossible de convertir la reference '${filteredNode.name}'`,
|
|
|
|
e
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return filteredNode
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Statically filter out replacements from `replaceBy`.
|
|
|
|
* Note: whitelist and blacklist filtering are applicable to the replacement
|
|
|
|
* itself or any parent namespace.
|
|
|
|
*/
|
|
|
|
export const getApplicableReplacedBy = (contextRuleName, replacedBy) =>
|
|
|
|
replacedBy
|
|
|
|
.sort(
|
|
|
|
(replacement1, replacement2) =>
|
|
|
|
+!!replacement2.whiteListedNames - +!!replacement1.whiteListedNames
|
|
|
|
)
|
|
|
|
.filter(
|
|
|
|
({ whiteListedNames }) =>
|
|
|
|
!whiteListedNames ||
|
|
|
|
whiteListedNames.some(name => contextRuleName.startsWith(name))
|
|
|
|
)
|
|
|
|
.filter(
|
|
|
|
({ blackListedNames }) =>
|
|
|
|
!blackListedNames ||
|
|
|
|
blackListedNames.every(name => !contextRuleName.startsWith(name))
|
|
|
|
)
|
|
|
|
.filter(({ referenceNode }) => contextRuleName !== referenceNode.dottedName)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filter-out and apply all possible replacements at runtime.
|
|
|
|
*/
|
2020-11-04 15:47:12 +00:00
|
|
|
const getApplicableReplacements = function(
|
|
|
|
this: Engine<string>,
|
|
|
|
contextRuleName: string,
|
2020-10-19 15:25:12 +00:00
|
|
|
rule: ParsedRule
|
2020-11-04 15:47:12 +00:00
|
|
|
) {
|
2020-10-19 15:25:12 +00:00
|
|
|
let missingVariableList: Array<EvaluatedNode['missingVariables']> = []
|
|
|
|
if (contextRuleName.startsWith('[evaluation]')) {
|
|
|
|
return [[], []]
|
|
|
|
}
|
|
|
|
const applicableReplacements = getApplicableReplacedBy(
|
|
|
|
contextRuleName,
|
|
|
|
rule.replacedBy
|
|
|
|
)
|
|
|
|
// Remove remplacement defined in a not applicable node
|
|
|
|
.filter(({ referenceNode }) => {
|
2020-11-04 15:47:12 +00:00
|
|
|
const referenceRule = this.parsedRules[referenceNode.dottedName]
|
2020-10-19 15:25:12 +00:00
|
|
|
const {
|
|
|
|
nodeValue: isApplicable,
|
|
|
|
missingVariables
|
2020-11-04 15:47:12 +00:00
|
|
|
} = evaluateApplicability.call(this, referenceRule as any)
|
2020-10-19 15:25:12 +00:00
|
|
|
missingVariableList.push(missingVariables)
|
|
|
|
return isApplicable
|
|
|
|
})
|
|
|
|
// Remove remplacement defined in a node whose situation value is false
|
|
|
|
.filter(({ referenceNode }) => {
|
2020-11-04 15:47:12 +00:00
|
|
|
const referenceRule = this.parsedRules[referenceNode.dottedName]
|
|
|
|
const situationValue = this.parsedSituation[referenceRule.dottedName]
|
2020-10-19 15:25:12 +00:00
|
|
|
if (referenceNode.question && situationValue == null) {
|
|
|
|
missingVariableList.push({ [referenceNode.dottedName]: 1 })
|
|
|
|
}
|
2020-11-04 15:47:12 +00:00
|
|
|
return (situationValue as any)?.nodeValue !== false
|
2020-10-19 15:25:12 +00:00
|
|
|
})
|
|
|
|
// Remove remplacement defined in a boolean node whose evaluated value is false
|
|
|
|
.filter(({ referenceNode }) => {
|
2020-11-04 15:47:12 +00:00
|
|
|
const referenceRule = this.parsedRules[referenceNode.dottedName]
|
2020-10-19 15:25:12 +00:00
|
|
|
if (referenceRule.formule?.explanation?.operationType !== 'comparison') {
|
|
|
|
return true
|
|
|
|
}
|
2020-11-04 15:47:12 +00:00
|
|
|
const { nodeValue: isApplicable, missingVariables } = this.evaluateNode(
|
2020-10-19 15:25:12 +00:00
|
|
|
referenceRule
|
|
|
|
)
|
|
|
|
missingVariableList.push(missingVariables)
|
|
|
|
return isApplicable
|
|
|
|
})
|
|
|
|
.map(({ referenceNode, replacementNode }) =>
|
2020-11-04 15:47:12 +00:00
|
|
|
replacementNode != null ? replacementNode : referenceNode
|
2020-10-19 15:25:12 +00:00
|
|
|
)
|
|
|
|
.map(replacementNode => {
|
|
|
|
const replacedRuleUnit = rule.unit
|
|
|
|
if (!areUnitConvertible(replacementNode.unit, replacedRuleUnit)) {
|
|
|
|
typeWarning(
|
|
|
|
contextRuleName,
|
|
|
|
`L'unité de la règle de remplacement n'est pas compatible avec celle de la règle remplacée ${rule.dottedName}`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
...replacementNode,
|
|
|
|
unit: replacementNode.unit || replacedRuleUnit
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
missingVariableList = missingVariableList.filter(
|
|
|
|
missingVariables => !!Object.keys(missingVariables).length
|
|
|
|
)
|
|
|
|
|
|
|
|
return [applicableReplacements, missingVariableList]
|
|
|
|
}
|