2020-03-26 15:03:19 +00:00
|
|
|
import { evaluateControls } from 'Engine/controls'
|
2020-04-23 07:30:03 +00:00
|
|
|
import { convertNodeToUnit, simplifyNodeUnit } from 'Engine/nodeUnits'
|
|
|
|
import { parse } from 'Engine/parse'
|
|
|
|
import { EvaluatedNode, EvaluatedRule, ParsedRules, Rules } from 'Engine/types'
|
|
|
|
import { parseUnit } from 'Engine/units'
|
|
|
|
import { mapObjIndexed } from 'ramda'
|
2020-03-10 10:52:53 +00:00
|
|
|
import { Simulation } from 'Reducers/rootReducer'
|
2020-04-23 07:30:03 +00:00
|
|
|
import { evaluationError, warning } from './error'
|
|
|
|
import { collectDefaults, evaluateNode } from './evaluation'
|
2020-03-26 15:03:19 +00:00
|
|
|
import parseRules from './parseRules'
|
2020-03-10 10:52:53 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
const emptyCache = () => ({
|
|
|
|
_meta: { contextRule: [] }
|
|
|
|
})
|
2020-03-10 10:52:53 +00:00
|
|
|
|
2020-03-26 15:03:19 +00:00
|
|
|
type Cache = {
|
|
|
|
_meta: {
|
|
|
|
contextRule: Array<string>
|
|
|
|
inversionFail?: {
|
|
|
|
given: string
|
|
|
|
estimated: string
|
|
|
|
}
|
|
|
|
}
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
export type EvaluationOptions = Partial<{
|
|
|
|
unit: string
|
|
|
|
useDefaultValues: boolean
|
|
|
|
}>
|
|
|
|
|
2020-03-30 16:24:18 +00:00
|
|
|
export { default as translateRules } from './translateRules'
|
2020-03-26 15:03:19 +00:00
|
|
|
export { parseRules }
|
2020-03-30 17:14:03 +00:00
|
|
|
export default class Engine<Names extends string> {
|
|
|
|
parsedRules: ParsedRules<Names>
|
2020-03-10 10:52:53 +00:00
|
|
|
defaultValues: Simulation['situation']
|
|
|
|
situation: Simulation['situation'] = {}
|
2020-04-23 07:30:03 +00:00
|
|
|
cache: Cache
|
|
|
|
cacheWithoutDefault: Cache
|
2020-03-10 10:52:53 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
constructor(rules: string | Rules<Names> | ParsedRules<Names>) {
|
|
|
|
this.cache = emptyCache()
|
|
|
|
this.cacheWithoutDefault = emptyCache()
|
2020-03-26 15:03:19 +00:00
|
|
|
this.parsedRules =
|
2020-03-30 17:14:03 +00:00
|
|
|
typeof rules === 'string' || !(Object.values(rules)[0] as any)?.dottedName
|
|
|
|
? parseRules(rules)
|
|
|
|
: (rules as ParsedRules<Names>)
|
2020-04-23 07:30:03 +00:00
|
|
|
|
|
|
|
this.defaultValues = mapObjIndexed(
|
|
|
|
(value, name) =>
|
|
|
|
typeof value === 'string'
|
|
|
|
? this.evaluateExpression(value, `[valeur par défaut] ${name}`, false)
|
|
|
|
: value,
|
|
|
|
collectDefaults(this.parsedRules)
|
|
|
|
)
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private resetCache() {
|
2020-04-23 07:30:03 +00:00
|
|
|
this.cache = emptyCache()
|
|
|
|
this.cacheWithoutDefault = emptyCache()
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
private situationGate(useDefaultValues = true) {
|
|
|
|
return dottedName =>
|
|
|
|
this.situation[dottedName] ??
|
|
|
|
(useDefaultValues ? this.defaultValues[dottedName] : null)
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
private evaluateExpression(
|
|
|
|
expression: string,
|
|
|
|
context: string,
|
2020-04-30 15:13:45 +00:00
|
|
|
useDefaultValues = true
|
2020-04-23 07:30:03 +00:00
|
|
|
): EvaluatedRule<Names> {
|
|
|
|
const result = simplifyNodeUnit(
|
|
|
|
evaluateNode(
|
|
|
|
useDefaultValues ? this.cache : this.cacheWithoutDefault,
|
|
|
|
this.situationGate(useDefaultValues),
|
|
|
|
this.parsedRules,
|
|
|
|
parse(
|
|
|
|
this.parsedRules,
|
|
|
|
{ dottedName: context },
|
|
|
|
this.parsedRules
|
|
|
|
)(expression)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
if (Object.keys(result.defaultValue?.missingVariable ?? {}).length) {
|
2020-04-29 14:19:20 +00:00
|
|
|
throw evaluationError(
|
2020-04-23 07:30:03 +00:00
|
|
|
context,
|
|
|
|
"Impossible d'évaluer l'expression car celle ci fait appel à des variables manquantes"
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return result
|
2020-04-02 16:54:41 +00:00
|
|
|
}
|
2020-03-10 10:52:53 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
setSituation(
|
|
|
|
situation: Partial<Record<Names, string | number | object>> = {}
|
|
|
|
) {
|
|
|
|
this.resetCache()
|
|
|
|
this.situation = mapObjIndexed(
|
|
|
|
(value, name) =>
|
|
|
|
typeof value === 'string'
|
|
|
|
? this.evaluateExpression(value, `[situation] ${name}`, true)
|
|
|
|
: value,
|
|
|
|
situation
|
|
|
|
)
|
|
|
|
return this
|
|
|
|
}
|
2020-04-21 13:49:48 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
evaluate(expression: Names, options?: EvaluationOptions): EvaluatedRule<Names>
|
|
|
|
evaluate(
|
|
|
|
expression: string,
|
|
|
|
options?: EvaluationOptions
|
|
|
|
): EvaluatedNode<Names>
|
|
|
|
evaluate(
|
|
|
|
expression: string,
|
|
|
|
options?: EvaluationOptions
|
|
|
|
): EvaluatedNode<Names> {
|
|
|
|
let result = this.evaluateExpression(
|
|
|
|
expression,
|
|
|
|
`[evaluation] ${expression}`,
|
|
|
|
options?.useDefaultValues ?? true
|
2020-03-26 15:03:19 +00:00
|
|
|
)
|
2020-04-23 07:30:03 +00:00
|
|
|
if (result.category === 'reference' && result.explanation) {
|
2020-04-23 09:55:51 +00:00
|
|
|
result = {
|
|
|
|
nodeValue: result.nodeValue,
|
|
|
|
unit: result.unit,
|
|
|
|
...('temporalValue' in result && {
|
|
|
|
temporalValue: result.temporalValue
|
|
|
|
}),
|
|
|
|
...result.explanation
|
|
|
|
}
|
2020-04-23 07:30:03 +00:00
|
|
|
}
|
|
|
|
if (options?.unit) {
|
|
|
|
try {
|
|
|
|
return convertNodeToUnit(
|
|
|
|
parseUnit(options.unit),
|
|
|
|
result as EvaluatedNode<Names, number>
|
|
|
|
)
|
|
|
|
} catch (e) {
|
|
|
|
warning(
|
|
|
|
`[evaluation] ${expression}`,
|
|
|
|
"L'unité demandée est incompatible avec l'expression évaluée"
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
2020-03-26 15:03:19 +00:00
|
|
|
controls() {
|
2020-04-23 07:30:03 +00:00
|
|
|
return evaluateControls(this.cache, this.situationGate(), this.parsedRules)
|
2020-03-26 15:03:19 +00:00
|
|
|
}
|
2020-04-23 07:30:03 +00:00
|
|
|
|
|
|
|
inversionFail(): boolean {
|
|
|
|
return !!this.cache._meta.inversionFail
|
|
|
|
}
|
|
|
|
|
|
|
|
getParsedRules(): ParsedRules<Names> {
|
|
|
|
return this.parsedRules
|
|
|
|
}
|
|
|
|
|
2020-03-26 15:03:19 +00:00
|
|
|
// TODO : this should be private
|
|
|
|
getCache(): Cache {
|
|
|
|
return this.cache
|
|
|
|
}
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|