2020-09-08 15:32:10 +00:00
|
|
|
/* eslint-disable @typescript-eslint/ban-types */
|
2020-10-04 23:39:11 +00:00
|
|
|
import { map } from 'ramda'
|
2020-11-04 17:05:46 +00:00
|
|
|
import { ASTNode, EvaluationDecoration, NodeKind } from './AST/types'
|
|
|
|
import { evaluationFunctions } from './evaluationFunctions'
|
|
|
|
import parse from './parse'
|
|
|
|
import parsePublicodes, { disambiguateReference } from './parsePublicodes'
|
|
|
|
import { Rule, RuleNode } from './rule'
|
2020-11-04 15:47:12 +00:00
|
|
|
import * as utils from './ruleUtils'
|
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>
|
2020-11-04 17:05:46 +00:00
|
|
|
parentEvaluationStack?: Array<string>
|
2020-11-04 15:47:12 +00:00
|
|
|
inversionFail?:
|
|
|
|
| {
|
|
|
|
given: string
|
|
|
|
estimated: string
|
|
|
|
}
|
|
|
|
| true
|
|
|
|
inRecalcul?: boolean
|
|
|
|
filter?: string
|
2020-03-26 15:03:19 +00:00
|
|
|
}
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
2020-05-08 10:04:00 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
export type EvaluationOptions = Partial<{
|
|
|
|
unit: string
|
|
|
|
}>
|
|
|
|
|
2020-11-04 17:05:46 +00:00
|
|
|
// export { default as cyclesLib } from './AST/index'
|
2020-05-08 10:04:00 +00:00
|
|
|
export * from './components'
|
2020-06-14 14:12:06 +00:00
|
|
|
export { formatValue, serializeValue } from './format'
|
2020-03-30 16:24:18 +00:00
|
|
|
export { default as translateRules } from './translateRules'
|
2020-11-04 17:05:46 +00:00
|
|
|
export { parsePublicodes }
|
2020-09-01 23:47:30 +00:00
|
|
|
export { utils }
|
2020-05-08 10:04:00 +00:00
|
|
|
|
2020-11-04 17:05:46 +00:00
|
|
|
export type evaluationFunction<Kind extends NodeKind = NodeKind> = (
|
|
|
|
this: Engine,
|
|
|
|
node: ASTNode & { nodeKind: Kind }
|
|
|
|
) => ASTNode & { nodeKind: Kind } & EvaluationDecoration
|
|
|
|
type ParsedRules<Name extends string> = Record<
|
|
|
|
Name,
|
|
|
|
RuleNode & { dottedName: Name }
|
|
|
|
>
|
|
|
|
export default class Engine<Name extends string = string> {
|
|
|
|
parsedRules: ParsedRules<Name>
|
|
|
|
parsedSituation: Record<string, ASTNode> = {}
|
2020-11-04 15:47:12 +00:00
|
|
|
cache: Cache
|
2020-05-06 20:08:49 +00:00
|
|
|
private warnings: Array<string> = []
|
2020-03-10 10:52:53 +00:00
|
|
|
|
2020-11-04 17:05:46 +00:00
|
|
|
constructor(rules: string | Record<string, Rule> | Record<string, RuleNode>) {
|
2020-04-23 07:30:03 +00:00
|
|
|
this.cache = emptyCache()
|
2020-11-04 15:47:12 +00:00
|
|
|
this.resetCache()
|
2020-11-04 17:05:46 +00:00
|
|
|
if (typeof rules === 'string') {
|
|
|
|
this.parsedRules = parsePublicodes(rules) as ParsedRules<Name>
|
|
|
|
}
|
|
|
|
const firstRuleObject = Object.values(rules)[0] as Rule | RuleNode
|
|
|
|
if (
|
|
|
|
typeof firstRuleObject === 'object' &&
|
|
|
|
firstRuleObject != null &&
|
|
|
|
'nodeKind' in firstRuleObject
|
|
|
|
) {
|
|
|
|
this.parsedRules = rules as ParsedRules<Name>
|
|
|
|
return
|
|
|
|
}
|
|
|
|
this.parsedRules = parsePublicodes(
|
|
|
|
rules as Record<string, Rule>
|
|
|
|
) as ParsedRules<Name>
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private resetCache() {
|
2020-04-23 07:30:03 +00:00
|
|
|
this.cache = emptyCache()
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
setSituation(
|
2020-11-04 17:05:46 +00:00
|
|
|
situation: Partial<Record<Name, string | number | object>> = {}
|
2020-04-23 07:30:03 +00:00
|
|
|
) {
|
|
|
|
this.resetCache()
|
2020-11-04 17:05:46 +00:00
|
|
|
this.parsedSituation = map(value => {
|
|
|
|
return disambiguateReference(this.parsedRules)(
|
|
|
|
parse(value, {
|
|
|
|
dottedName: "'''situation",
|
|
|
|
parsedRules: {}
|
|
|
|
})
|
|
|
|
)
|
|
|
|
}, situation)
|
2020-04-23 07:30:03 +00:00
|
|
|
return this
|
|
|
|
}
|
2020-04-21 13:49:48 +00:00
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
evaluate(
|
2020-11-04 17:05:46 +00:00
|
|
|
expression: Name
|
|
|
|
): RuleNode & EvaluationDecoration & { dottedName: Name }
|
|
|
|
evaluate(expression: string): ASTNode & EvaluationDecoration {
|
|
|
|
/*
|
|
|
|
TODO
|
|
|
|
EN ATTENDANT d'AVOIR une meilleure gestion d'erreur, on va mocker console.warn
|
|
|
|
*/
|
|
|
|
const originalWarn = console.warn
|
|
|
|
|
|
|
|
console.warn = (warning: string) => {
|
|
|
|
this.warnings.push(warning)
|
|
|
|
originalWarn(warning)
|
2020-04-23 07:30:03 +00:00
|
|
|
}
|
2020-11-04 17:05:46 +00:00
|
|
|
if (this.parsedRules[expression]) {
|
|
|
|
// TODO : No replacement here. Is this what we want ?
|
|
|
|
return this.evaluateNode(this.parsedRules[expression])
|
2020-04-23 07:30:03 +00:00
|
|
|
}
|
2020-11-04 17:05:46 +00:00
|
|
|
const result = this.evaluateNode(
|
|
|
|
disambiguateReference(this.parsedRules)(
|
|
|
|
parse(expression, {
|
|
|
|
dottedName: "'''evaluation",
|
|
|
|
parsedRules: {}
|
|
|
|
})
|
|
|
|
)
|
|
|
|
)
|
|
|
|
console.warn = originalWarn
|
2020-04-23 07:30:03 +00:00
|
|
|
return result
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|
2020-05-06 20:08:49 +00:00
|
|
|
|
2020-05-26 14:12:16 +00:00
|
|
|
getWarnings() {
|
|
|
|
return this.warnings
|
|
|
|
}
|
|
|
|
|
2020-04-23 07:30:03 +00:00
|
|
|
inversionFail(): boolean {
|
|
|
|
return !!this.cache._meta.inversionFail
|
|
|
|
}
|
|
|
|
|
2020-11-04 17:05:46 +00:00
|
|
|
getParsedRules(): Record<string, RuleNode> {
|
2020-04-23 07:30:03 +00:00
|
|
|
return this.parsedRules
|
|
|
|
}
|
2020-11-04 15:47:12 +00:00
|
|
|
|
2020-11-04 17:05:46 +00:00
|
|
|
evaluateNode<N extends ASTNode = ASTNode>(node: N): N & EvaluationDecoration {
|
2020-11-04 15:47:12 +00:00
|
|
|
if (!node.nodeKind) {
|
|
|
|
throw Error('The provided node must have a "nodeKind" attribute')
|
|
|
|
} else if (!evaluationFunctions[node.nodeKind]) {
|
|
|
|
throw Error(`Unknown "nodeKind": ${node.nodeKind}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
return evaluationFunctions[node.nodeKind].call(this, node)
|
|
|
|
}
|
2020-03-10 10:52:53 +00:00
|
|
|
}
|