195 lines
5.4 KiB
TypeScript
195 lines
5.4 KiB
TypeScript
import { groupBy } from 'ramda'
|
|
import { AST } from 'yaml'
|
|
import { traverseParsedRules, updateAST } from './AST'
|
|
import { ASTNode } from './AST/types'
|
|
import Variations from './components/mecanisms/Variations'
|
|
import { InternalError, warning } from './error'
|
|
import { defaultNode, makeJsx } from './evaluation'
|
|
import { VariationNode } from './mecanisms/variations'
|
|
import parse from './parse'
|
|
import { Context } from './parsePublicodes'
|
|
import { RuleNode } from './rule'
|
|
import { Rule } from './rule'
|
|
import { coerceArray } from './utils'
|
|
|
|
export type ReplacementNode = {
|
|
nodeKind: 'replacement'
|
|
definitionRule: ASTNode & { nodeKind: 'reference' }
|
|
replacedReference: ASTNode & { nodeKind: 'reference' }
|
|
replacementNode: ASTNode
|
|
whiteListedNames: Array<ASTNode & { nodeKind: 'reference' }>
|
|
jsx: any
|
|
rawNode: any
|
|
blackListedNames: Array<ASTNode & { nodeKind: 'reference' }>
|
|
}
|
|
|
|
export function parseReplacements(
|
|
replacements: Rule['remplace'],
|
|
context: Context
|
|
): Array<ReplacementNode> {
|
|
if (!replacements) {
|
|
return []
|
|
}
|
|
return coerceArray(replacements).map(replacement => {
|
|
if (typeof replacement === 'string') {
|
|
replacement = { règle: replacement }
|
|
}
|
|
|
|
const replacedReference = parse(replacement.règle, context)
|
|
let replacementNode = parse(replacement.par ?? context.dottedName, context)
|
|
|
|
const [whiteListedNames, blackListedNames] = [
|
|
replacement.dans ?? [],
|
|
replacement['sauf dans'] ?? []
|
|
]
|
|
.map(dottedName => coerceArray(dottedName))
|
|
.map(refs => refs.map(ref => parse(ref, context)))
|
|
|
|
return {
|
|
nodeKind: 'replacement',
|
|
rawNode: replacement,
|
|
definitionRule: parse(context.dottedName, context),
|
|
replacedReference,
|
|
replacementNode,
|
|
jsx: (node: ReplacementNode) => (
|
|
<span>
|
|
Remplace {makeJsx(node.replacedReference)}{' '}
|
|
{node.rawNode.par && <>par {makeJsx(node.replacementNode)}</>}
|
|
{node.rawNode.dans && (
|
|
<>dans {node.whiteListedNames.map(makeJsx).join(', ')}</>
|
|
)}
|
|
{node.rawNode['sauf dans'] && (
|
|
<>sauf dans {node.blackListedNames.map(makeJsx).join(', ')}</>
|
|
)}
|
|
</span>
|
|
),
|
|
whiteListedNames,
|
|
blackListedNames
|
|
} as ReplacementNode
|
|
})
|
|
}
|
|
|
|
export function parseRendNonApplicable(
|
|
rules: Rule['rend non applicable'],
|
|
context: Context
|
|
): Array<ReplacementNode> {
|
|
return parseReplacements(rules, context).map(
|
|
replacement =>
|
|
({
|
|
...replacement,
|
|
replacementNode: defaultNode(false)
|
|
} as ReplacementNode)
|
|
)
|
|
}
|
|
|
|
export function inlineReplacements(
|
|
parsedRules: Record<string, RuleNode>
|
|
): Record<string, RuleNode> {
|
|
const replacements: Record<string, Array<ReplacementNode>> = groupBy(
|
|
(r: ReplacementNode) => {
|
|
if (!r.replacedReference.dottedName) {
|
|
throw new InternalError(r)
|
|
}
|
|
return r.replacedReference.dottedName
|
|
},
|
|
Object.values(parsedRules).flatMap(rule => rule.replacements)
|
|
)
|
|
return traverseParsedRules(
|
|
updateAST(node => {
|
|
if (
|
|
node.nodeKind === 'replacement' ||
|
|
node.nodeKind === 'inversion' ||
|
|
node.nodeKind === 'une possibilité' ||
|
|
node.nodeKind === 'recalcul'
|
|
) {
|
|
// We don't want to replace references in replacements...
|
|
// Nor in ammended situation of recalcul and inversion (for now)
|
|
return false
|
|
}
|
|
if (node.nodeKind === 'reference') {
|
|
if (!node.dottedName) {
|
|
throw new InternalError(node)
|
|
}
|
|
return replace(node, replacements[node.dottedName] ?? [])
|
|
}
|
|
}),
|
|
parsedRules
|
|
) as Record<string, RuleNode>
|
|
}
|
|
|
|
function replace(
|
|
node: ASTNode & { nodeKind: 'reference' }, //& { dottedName: string },
|
|
replacements: Array<ReplacementNode>
|
|
): ASTNode {
|
|
// TODO : handle transitivité
|
|
|
|
const applicableReplacements = replacements
|
|
.filter(
|
|
({ definitionRule }) =>
|
|
definitionRule.dottedName !== node.contextDottedName
|
|
)
|
|
.filter(
|
|
({ whiteListedNames }) =>
|
|
!whiteListedNames.length ||
|
|
whiteListedNames.some(name =>
|
|
node.contextDottedName.startsWith(name.dottedName as string)
|
|
)
|
|
)
|
|
.filter(
|
|
({ blackListedNames }) =>
|
|
!blackListedNames.length ||
|
|
blackListedNames.every(
|
|
name => !node.contextDottedName.startsWith(name.dottedName as string)
|
|
)
|
|
)
|
|
.sort((r1, r2) => {
|
|
// Replacement with whitelist conditions have precedence over the others
|
|
const criterion1 =
|
|
(+!!r2.whiteListedNames.length as number) -
|
|
+!!r1.whiteListedNames.length
|
|
// Replacement with blacklist condition have precedence over the others
|
|
const criterion2 =
|
|
+!!r2.blackListedNames.length - +!!r1.blackListedNames.length
|
|
return criterion1 || criterion2
|
|
})
|
|
if (!applicableReplacements.length) {
|
|
return node
|
|
}
|
|
if (applicableReplacements.length > 1) {
|
|
warning(
|
|
node.contextDottedName,
|
|
`
|
|
Il existe plusieurs remplacements pour la référence '${node.dottedName}'.
|
|
Lors de l'execution, ils seront résolus dans l'odre suivant :
|
|
${applicableReplacements.map(
|
|
replacement =>
|
|
`\n\t- Celui définit dans la règle '${replacement.definitionRule.dottedName}'`
|
|
)}
|
|
`
|
|
)
|
|
}
|
|
|
|
return {
|
|
nodeKind: 'variations',
|
|
rawNode: node.rawNode,
|
|
jsx: Replacement,
|
|
explanation: [
|
|
...applicableReplacements.map(replacement => ({
|
|
condition: replacement.definitionRule,
|
|
consequence: replacement.replacementNode
|
|
})),
|
|
{
|
|
condition: defaultNode(true),
|
|
consequence: node
|
|
}
|
|
]
|
|
}
|
|
}
|
|
|
|
function Replacement(node: VariationNode) {
|
|
const applicableReplacement = node.explanation.find(ex => ex.satisfied)
|
|
?.consequence
|
|
const replacedNode = node.explanation.slice(-1)[0].consequence
|
|
return makeJsx(applicableReplacement || replacedNode)
|
|
}
|