2020-03-26 15:03:19 +00:00
|
|
|
import parseRule from 'Engine/parseRule'
|
2020-04-25 13:27:06 +00:00
|
|
|
import yaml from 'yaml'
|
2020-03-25 17:06:07 +00:00
|
|
|
import { lensPath, set } from 'ramda'
|
|
|
|
import { compilationError } from './error'
|
2020-03-26 15:03:19 +00:00
|
|
|
import { parseReference } from './parseReference'
|
2020-03-30 17:14:03 +00:00
|
|
|
import { ParsedRules, Rules } from './types'
|
2020-03-26 15:03:19 +00:00
|
|
|
|
2020-03-30 17:14:03 +00:00
|
|
|
export default function parseRules<Names extends string>(
|
|
|
|
rawRules: Rules<Names> | string
|
|
|
|
): ParsedRules<Names> {
|
|
|
|
const rules =
|
|
|
|
typeof rawRules === 'string'
|
2020-04-25 13:27:06 +00:00
|
|
|
? (yaml.parse(rawRules.replace(/\t/g, ' ')) as Rules<Names>)
|
2020-03-25 17:06:07 +00:00
|
|
|
: { ...rawRules }
|
|
|
|
|
|
|
|
extractInlinedNames(rules)
|
2020-03-26 15:03:19 +00:00
|
|
|
|
|
|
|
/* First we parse each rule one by one. When a mechanism is encountered, it is
|
|
|
|
recursively parsed. When a reference to a variable is encountered, a
|
|
|
|
'variable' node is created, we don't parse variables recursively. */
|
2020-04-30 15:13:45 +00:00
|
|
|
const parsedRules = {}
|
2020-03-26 15:03:19 +00:00
|
|
|
|
|
|
|
/* A rule `A` can disable a rule `B` using the rule `rend non applicable: B`
|
|
|
|
in the definition of `A`. We need to map these exonerations to be able to
|
|
|
|
retreive them from `B` */
|
2020-04-30 15:13:45 +00:00
|
|
|
const nonApplicableMapping: Record<string, any> = {}
|
|
|
|
const replacedByMapping: Record<string, any> = {}
|
2020-03-30 17:14:03 +00:00
|
|
|
;(Object.keys(rules) as Names[]).map(dottedName => {
|
|
|
|
const parsedRule = parseRule(rules, dottedName, parsedRules)
|
2020-03-26 15:03:19 +00:00
|
|
|
|
2020-03-30 17:14:03 +00:00
|
|
|
if (parsedRule['rend non applicable']) {
|
|
|
|
nonApplicableMapping[parsedRule.dottedName] =
|
|
|
|
parsedRule['rend non applicable']
|
2020-03-26 15:03:19 +00:00
|
|
|
}
|
|
|
|
|
2020-03-30 17:14:03 +00:00
|
|
|
const replaceDescriptors = parsedRule['remplace']
|
2020-03-26 15:03:19 +00:00
|
|
|
if (replaceDescriptors) {
|
|
|
|
replaceDescriptors.forEach(
|
|
|
|
descriptor =>
|
|
|
|
(replacedByMapping[descriptor.referenceName] = [
|
|
|
|
...(replacedByMapping[descriptor.referenceName] ?? []),
|
2020-03-30 17:14:03 +00:00
|
|
|
{ ...descriptor, referenceName: parsedRule.dottedName }
|
2020-03-26 15:03:19 +00:00
|
|
|
])
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
Object.entries(nonApplicableMapping).forEach(([a, b]) => {
|
|
|
|
b.forEach(ruleName => {
|
|
|
|
parsedRules[ruleName].isDisabledBy.push(
|
|
|
|
parseReference(rules, parsedRules[ruleName], parsedRules)(a)
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
Object.entries(replacedByMapping).forEach(([a, b]) => {
|
|
|
|
parsedRules[a].replacedBy = b.map(({ referenceName, ...other }) => ({
|
|
|
|
referenceNode: parseReference(
|
|
|
|
rules,
|
|
|
|
parsedRules[referenceName],
|
|
|
|
parsedRules
|
|
|
|
)(referenceName),
|
|
|
|
...other
|
|
|
|
}))
|
|
|
|
})
|
|
|
|
|
2020-03-30 17:14:03 +00:00
|
|
|
return parsedRules as ParsedRules<Names>
|
2020-03-26 15:03:19 +00:00
|
|
|
}
|
2020-03-25 17:06:07 +00:00
|
|
|
|
|
|
|
// We recursively traverse the YAML tree in order to extract named parameters
|
|
|
|
// into their own dedicated rules, and replace the inline definition with a
|
|
|
|
// reference to the newly created rule.
|
2020-04-30 15:13:45 +00:00
|
|
|
function extractInlinedNames(rules: Record<string, Record<string, any>>) {
|
2020-03-25 17:06:07 +00:00
|
|
|
const extractNamesInRule = (dottedName: string) => {
|
|
|
|
rules[dottedName] !== null &&
|
|
|
|
Object.entries(rules[dottedName]).forEach(
|
|
|
|
extractNamesInObject(dottedName)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
const extractNamesInObject = (
|
|
|
|
dottedName: string,
|
|
|
|
context: Array<string | number> = []
|
2020-04-30 15:13:45 +00:00
|
|
|
) => ([key, value]: [string, Record<string, any>]) => {
|
|
|
|
const match = /\[ref( (.+))?\]$/.exec(key)
|
2020-03-25 17:06:07 +00:00
|
|
|
if (match) {
|
|
|
|
const argumentType = key.replace(match[0], '').trim()
|
|
|
|
const argumentName = match[2]?.trim() || argumentType
|
|
|
|
const extractedReferenceName = `${dottedName} . ${argumentName}`
|
|
|
|
|
|
|
|
if (typeof rules[extractedReferenceName] !== 'undefined') {
|
|
|
|
compilationError(
|
|
|
|
dottedName,
|
|
|
|
`Le paramètre [ref] ${argumentName} entre en conflit avec la règle déjà existante ${extractedReferenceName}`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
rules[extractedReferenceName] = {
|
|
|
|
formule: value,
|
|
|
|
// TODO: The `virtualRule` parameter should be used to avoid creating a
|
|
|
|
// dedicated documentation page.
|
|
|
|
virtualRule: true
|
|
|
|
}
|
|
|
|
rules[dottedName] = set(
|
|
|
|
lensPath([...context, argumentType]),
|
|
|
|
extractedReferenceName,
|
|
|
|
rules[dottedName]
|
|
|
|
)
|
|
|
|
extractNamesInRule(extractedReferenceName)
|
|
|
|
} else if (Array.isArray(value)) {
|
2020-04-30 15:13:45 +00:00
|
|
|
value.forEach((content: Record<string, any>, i) =>
|
2020-03-25 17:06:07 +00:00
|
|
|
Object.entries(content).forEach(
|
|
|
|
extractNamesInObject(dottedName, [...context, key, i])
|
|
|
|
)
|
|
|
|
)
|
|
|
|
} else if (value && typeof value === 'object') {
|
|
|
|
Object.entries(value).forEach(
|
|
|
|
extractNamesInObject(dottedName, [...context, key])
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Object.keys(rules).forEach(extractNamesInRule)
|
|
|
|
}
|