/** * Note: all here is strictly based on duck typing. * We don't exepect the parent rule to explain the type of the contained formula, for example. */ import * as R from 'ramda' import { ArrondiExplanation } from './mecanisms/arrondi' import { ParsedRule, ParsedRules } from './types' type OnOff = 'oui' | 'non' export function isOnOff(a: string): a is OnOff { return a === 'oui' || a === 'non' } // Note: to build type-guards, we would need to have a `isNames` guard. That's // pretty cumbersome, so for now we rely on this. type WannabeDottedName = string export function isWannabeDottedName(a: string): a is WannabeDottedName { return typeof a === 'string' } type ASTNode = { [_: string]: {} | undefined } type RuleNode = ASTNode & ParsedRule type RuleProp = ASTNode & { category: 'ruleProp' rulePropType: string } export function isRuleProp(node: ASTNode): node is RuleProp { return ( (node as RuleProp).category === 'ruleProp' && typeof (node as RuleProp).rulePropType === 'string' ) } type CondRuleProp = RuleProp & { rulePropType: 'cond' } export function isCondRuleProp(node: ASTNode): node is CondRuleProp { return isRuleProp(node) && (node as CondRuleProp).rulePropType === 'cond' } type ApplicableSi = CondRuleProp & { dottedName: 'applicable si' explanation: ASTNode } export function isApplicableSi(node: ASTNode): node is ApplicableSi { const applicableSi = node as ApplicableSi return ( isCondRuleProp(node) && applicableSi.dottedName === 'applicable si' && applicableSi.explanation !== undefined ) } type NonApplicableSi = CondRuleProp & { dottedName: 'non applicable si' explanation: ASTNode } export function isNonApplicableSi(node: ASTNode): node is NonApplicableSi { const nonApplicableSi = node as NonApplicableSi return ( isCondRuleProp(node) && nonApplicableSi.dottedName === 'non applicable si' && nonApplicableSi.explanation !== undefined ) } type Formule = RuleProp & { name: 'formule' rulePropType: 'formula' explanation: FormuleExplanation } export function isFormule( node: ASTNode ): node is Formule { const formule = node as Formule return ( isRuleProp(node) && formule.name === 'formule' && formule.rulePropType === 'formula' && isFormuleExplanation(formule.explanation) ) } type FormuleExplanation = | Value | Operation | Possibilities | Possibilities2 | Reference | AnyMechanism export function isFormuleExplanation( node: ASTNode ): node is FormuleExplanation { return ( isValue(node) || isOperation(node) || isReference(node) || isPossibilities(node) || isPossibilities2(node) || isAnyMechanism(node) ) } type Value = ASTNode & { nodeValue: number | string | boolean } export function isValue(node: ASTNode): node is Value { const value = node as Value return ( typeof value.nodeValue === 'string' || typeof value.nodeValue === 'number' || typeof value.nodeValue === 'boolean' ) } type Operation = ASTNode & { operationType: 'comparison' | 'calculation' explanation: Array } export function isOperation(node: ASTNode): node is Operation { return R.includes((node as Operation).operationType, [ 'comparison', 'calculation' ]) } type Possibilities = ASTNode & { possibilités: Array 'choix obligatoire'?: OnOff 'une possibilité': OnOff } export function isPossibilities(node: ASTNode): node is Possibilities { const possibilities = node as Possibilities return ( possibilities.possibilités instanceof Array && possibilities.possibilités.every(it => typeof it === 'string') && (possibilities['choix obligatoire'] === undefined || isOnOff(possibilities['choix obligatoire'])) && isOnOff(possibilities['une possibilité']) ) } type Possibilities2 = ASTNode & { [index: number]: string // short dotted name 'choix obligatoire'?: OnOff 'une possibilité': OnOff } export function isPossibilities2(node: ASTNode): node is Possibilities2 { const possibilities2 = node as Possibilities2 return ( Object.entries(possibilities2).every( ([k, v]) => isNaN(parseInt(k, 10)) || typeof v === 'string' ) && (possibilities2['choix obligatoire'] === undefined || isOnOff(possibilities2['choix obligatoire'])) && isOnOff(possibilities2['une possibilité']) ) } type Reference = ASTNode & { category: 'reference' name: Names partialReference: Names dottedName: Names } export function isReference( node: ASTNode ): node is Reference { const reference = node as Reference return ( reference.category === 'reference' && isWannabeDottedName(reference.name) && isWannabeDottedName(reference.partialReference) && isWannabeDottedName(reference.dottedName) ) } type AbstractMechanism = ASTNode & { category: 'mecanism' name: string } export function isAbstractMechanism(node: ASTNode): node is AbstractMechanism { return ( (node as AbstractMechanism).category === 'mecanism' && typeof (node as AbstractMechanism).name === 'string' ) } type RecalculMech = AbstractMechanism & { explanation: { recalcul: Reference amendedSituation: Record> } } export function isRecalculMech( node: ASTNode ): node is RecalculMech { const recalculMech = node as RecalculMech const isReferenceSpec = isReference as ( node: ASTNode ) => node is Reference return ( typeof recalculMech.explanation === 'object' && typeof recalculMech.explanation.recalcul === 'object' && isReferenceSpec(recalculMech.explanation.recalcul as ASTNode) && typeof recalculMech.explanation.amendedSituation === 'object' ) } type EncadrementMech = AbstractMechanism & { name: 'encadrement' explanation: { valeur: ASTNode plafond: ASTNode plancher: ASTNode } } export function isEncadrementMech(node: ASTNode): node is EncadrementMech { const encadrementMech = node as EncadrementMech return ( isAbstractMechanism(encadrementMech) && encadrementMech.name == 'encadrement' && typeof encadrementMech.explanation === 'object' && encadrementMech.explanation.valeur !== undefined && encadrementMech.explanation.plafond !== undefined && encadrementMech.explanation.plancher !== undefined ) } type SommeMech = AbstractMechanism & { name: 'somme' explanation: Array } export function isSommeMech(node: ASTNode): node is SommeMech { const sommeMech = node as SommeMech return ( isAbstractMechanism(sommeMech) && sommeMech.name === 'somme' && sommeMech.explanation instanceof Array ) } type ProduitMech = AbstractMechanism & { name: 'produit' explanation: { assiette: ASTNode plafond: ASTNode facteur: ASTNode taux: ASTNode } } export function isProduitMech(node: ASTNode): node is ProduitMech { const produitMech = node as ProduitMech return ( isAbstractMechanism(produitMech) && produitMech.name === 'produit' && typeof produitMech.explanation === 'object' && typeof produitMech.explanation.assiette === 'object' && typeof produitMech.explanation.plafond === 'object' && typeof produitMech.explanation.facteur === 'object' && typeof produitMech.explanation.taux === 'object' ) } type VariationsMech = AbstractMechanism & { name: 'variations' explanation: { condition: ASTNode consequence: ASTNode }[] } export function isVariationsMech(node: ASTNode): node is VariationsMech { const variationsMech = node as VariationsMech return ( isAbstractMechanism(variationsMech) && variationsMech.name === 'variations' && variationsMech.explanation instanceof Array && variationsMech.explanation.every( variation => typeof variation === 'object' && variation.condition !== undefined && variation.consequence !== undefined ) ) } type AllegementMech = AbstractMechanism & { name: 'allègement' explanation: { abattement: ASTNode assiette: ASTNode décote: | undefined | { plafond: ASTNode taux: ASTNode } franchise: ASTNode plafond: ASTNode } } export function isAllegementMech(node: ASTNode): node is AllegementMech { const allegementMech = node as AllegementMech return ( isAbstractMechanism(allegementMech) && allegementMech.name === 'allègement' && typeof allegementMech.explanation === 'object' && allegementMech.explanation.abattement !== undefined && allegementMech.explanation.assiette !== undefined && allegementMech.explanation.franchise !== undefined && allegementMech.explanation.plafond !== undefined && (allegementMech.explanation.décote === undefined || (typeof allegementMech.explanation.décote === 'object' && allegementMech.explanation.décote.plafond !== undefined && allegementMech.explanation.décote.taux !== undefined)) ) } type BaremeMech = AbstractMechanism & { name: 'barème' explanation: { assiette: ASTNode multiplicateur: ASTNode tranches: { plafond: ASTNode taux: ASTNode }[] } } export function isBaremeMech(node: ASTNode): node is BaremeMech { const baremeMech = node as BaremeMech return ( isAbstractMechanism(baremeMech) && baremeMech.name === 'barème' && typeof baremeMech.explanation === 'object' && baremeMech.explanation.assiette !== undefined && baremeMech.explanation.multiplicateur !== undefined && baremeMech.explanation.tranches instanceof Array && baremeMech.explanation.tranches.every( tranche => typeof tranche === 'object' && tranche.plafond !== undefined && tranche.taux !== undefined ) ) } type InversionNumMech = AbstractMechanism & { name: 'inversion numérique' explanation: { inversionCandidates: Array> } } export function isInversionNumMech( node: ASTNode ): node is InversionNumMech { const inversionNumMech = node as InversionNumMech const isReferenceSpec = isReference as ( node: ASTNode ) => node is Reference return ( isAbstractMechanism(inversionNumMech) && inversionNumMech.name === 'inversion numérique' && typeof inversionNumMech.explanation === 'object' && inversionNumMech.explanation.inversionCandidates instanceof Array && inversionNumMech.explanation.inversionCandidates.every(isReferenceSpec) ) } type ArrondiMech = AbstractMechanism & { name: 'arrondi' explanation: Record } export function isArrondiMech(node: ASTNode): node is ArrondiMech { const arrondiMech = node as ArrondiMech return ( isAbstractMechanism(arrondiMech) && arrondiMech.name === 'arrondi' && typeof arrondiMech.explanation === 'object' && arrondiMech.explanation.decimals !== undefined && arrondiMech.explanation.value !== undefined ) } type MaxMech = AbstractMechanism & { name: 'le maximum de' explanation: Array } export function isMaxMech(node: ASTNode): node is MaxMech { const maxMech = node as MaxMech return ( isAbstractMechanism(maxMech) && maxMech.name === 'le maximum de' && maxMech.explanation instanceof Array ) } type MinMech = AbstractMechanism & { name: 'le minimum de' explanation: Array } export function isMinMech(node: ASTNode): node is MinMech { const minMech = node as MinMech return ( isAbstractMechanism(minMech) && minMech.name === 'le minimum de' && minMech.explanation instanceof Array ) } type ComposantesMech = AbstractMechanism & { name: 'composantes' explanation: Array } export function isComposantesMech(node: ASTNode): node is ComposantesMech { const composantesMech = node as ComposantesMech return ( isAbstractMechanism(composantesMech) && composantesMech.name === 'composantes' && composantesMech.explanation instanceof Array ) } type UneConditionsMech = AbstractMechanism & { name: 'une de ces conditions' explanation: Array } export function isUneConditionsMech(node: ASTNode): node is UneConditionsMech { const uneConditionsMech = node as UneConditionsMech return ( isAbstractMechanism(uneConditionsMech) && uneConditionsMech.name === 'une de ces conditions' && uneConditionsMech.explanation instanceof Array ) } type ToutesConditionsMech = AbstractMechanism & { name: 'toutes ces conditions' explanation: Array } export function isToutesConditionsMech( node: ASTNode ): node is ToutesConditionsMech { const toutesConditionsMech = node as ToutesConditionsMech return ( isAbstractMechanism(toutesConditionsMech) && toutesConditionsMech.name === 'toutes ces conditions' && toutesConditionsMech.explanation instanceof Array ) } type SyncMech = AbstractMechanism & { name: 'synchronisation' API: {} } export function isSyncMech(node: ASTNode): node is SyncMech { const syncMech = node as SyncMech return isAbstractMechanism(syncMech) && syncMech.name === 'synchronisation' } type GrilleMech = AbstractMechanism & { name: 'grille' explanation: { assiette: ASTNode multiplicateur: ASTNode tranches: { montant: ASTNode plafond: ASTNode }[] } } export function isGrilleMech(node: ASTNode): node is GrilleMech { const grilleMech = node as GrilleMech return ( isAbstractMechanism(grilleMech) && grilleMech.name === 'grille' && typeof grilleMech.explanation === 'object' && grilleMech.explanation.assiette !== undefined && grilleMech.explanation.multiplicateur !== undefined && grilleMech.explanation.tranches instanceof Array && grilleMech.explanation.tranches.every( tranche => typeof tranche === 'object' && tranche.montant !== undefined && tranche.plafond !== undefined ) ) } type TauxProgMech = AbstractMechanism & { name: 'taux progressif' explanation: { assiette: ASTNode multiplicateur: ASTNode tranches: { plafond: ASTNode taux: ASTNode }[] } } export function isTauxProgMech(node: ASTNode): node is TauxProgMech { const tauxProgMech = node as TauxProgMech return ( isAbstractMechanism(tauxProgMech) && tauxProgMech.name === 'taux progressif' && typeof tauxProgMech.explanation === 'object' && tauxProgMech.explanation.assiette !== undefined && tauxProgMech.explanation.multiplicateur !== undefined && tauxProgMech.explanation.tranches instanceof Array && tauxProgMech.explanation.tranches.every( tranche => typeof tranche === 'object' && tranche.plafond !== undefined && tranche.taux !== undefined ) ) } type DureeMech = AbstractMechanism & { name: 'Durée' explanation: { depuis: ASTNode "jusqu'à": ASTNode } } export function isDureeMech(node: ASTNode): node is DureeMech { const dureeMech = node as DureeMech return ( isAbstractMechanism(dureeMech) && dureeMech.name === 'Durée' && typeof dureeMech.explanation === 'object' && dureeMech.explanation.depuis !== undefined && dureeMech.explanation["jusqu'à"] !== undefined ) } type AnyMechanism = | RecalculMech | EncadrementMech | SommeMech | ProduitMech | VariationsMech | AllegementMech | BaremeMech | InversionNumMech | ArrondiMech | MaxMech | MinMech | ComposantesMech | UneConditionsMech | ToutesConditionsMech | SyncMech | GrilleMech | TauxProgMech | DureeMech export function isAnyMechanism( node: ASTNode ): node is AnyMechanism { return ( isRecalculMech(node) || isEncadrementMech(node) || isSommeMech(node) || isProduitMech(node) || isVariationsMech(node) || isAllegementMech(node) || isBaremeMech(node) || isInversionNumMech(node) || isArrondiMech(node) || isMaxMech(node) || isMinMech(node) || isComposantesMech(node) || isUneConditionsMech(node) || isToutesConditionsMech(node) || isSyncMech(node) || isGrilleMech(node) || isTauxProgMech(node) || isDureeMech(node) ) } enum DependencyType { formule, replacedBy, disabledBy } type RuleDependency = [Names, DependencyType] type RuleDependencies = Array> export function ruleDepsOfNode( ruleName: Names, node: ASTNode ): RuleDependencies { function ruleDepsOfApplicableSi( applicableSi: ApplicableSi ): RuleDependencies { return ruleDepsOfNode(ruleName, applicableSi.explanation) } function ruleDepsOfNonApplicableSi( nonApplicableSi: NonApplicableSi ): RuleDependencies { return ruleDepsOfNode(ruleName, nonApplicableSi.explanation) } function ruleDepsOfFormule(formule: Formule): RuleDependencies { return ruleDepsOfNode(ruleName, formule.explanation) } function ruleDepsOfValue(value: Value): RuleDependencies { return [] } function ruleDepsOfOperation(operation: Operation): RuleDependencies { return operation.explanation.flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) } function ruleDepsOfPossibilities( possibilities: Possibilities ): RuleDependencies { return [] } function ruleDepsOfPossibilities2( possibilities: Possibilities2 ): RuleDependencies { return [] } function ruleDepsOfReference( reference: Reference ): RuleDependencies { return [[reference.dottedName, DependencyType.formule]] } function ruleDepsOfRecalculMech( recalculMech: RecalculMech ): RuleDependencies { const ruleReference = recalculMech.explanation.recalcul.partialReference return ruleReference === ruleName ? [] : [[ruleReference, DependencyType.formule]] } function ruleDepsOfEncadrementMech( encadrementMech: EncadrementMech ): RuleDependencies { const result = [ encadrementMech.explanation.plafond, encadrementMech.explanation.plancher, encadrementMech.explanation.valeur ].flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfSommeMech(sommeMech: SommeMech): RuleDependencies { const result = sommeMech.explanation.flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfProduitMech( produitMech: ProduitMech ): RuleDependencies { const result = [ produitMech.explanation.assiette, produitMech.explanation.plafond, produitMech.explanation.facteur, produitMech.explanation.taux ].flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfVariationsMech( variationsMech: VariationsMech ): RuleDependencies { function ruleOfVariation({ condition, consequence }: { condition: ASTNode consequence: ASTNode }): RuleDependencies { return R.concat( ruleDepsOfNode(ruleName, condition), ruleDepsOfNode(ruleName, consequence) ) } const result = variationsMech.explanation.flatMap(ruleOfVariation) return result } function ruleDepsOfAllegementMech( allegementMech: AllegementMech ): RuleDependencies { const subNodes = R.concat( [ allegementMech.explanation.abattement, allegementMech.explanation.assiette, allegementMech.explanation.franchise, allegementMech.explanation.plafond ], allegementMech.explanation.décote ? [ allegementMech.explanation.décote.plafond, allegementMech.explanation.décote.taux ] : [] ) const result = subNodes.flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfBaremeMech( baremeMech: BaremeMech ): RuleDependencies { const tranchesNodes = baremeMech.explanation.tranches.flatMap( ({ plafond, taux }) => [plafond, taux] ) const result = R.concat( [baremeMech.explanation.assiette, baremeMech.explanation.multiplicateur], tranchesNodes ).flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } /** * Returns 0 dependency for _inversion numérique_ as it's not creating a logical dependency. */ function ruleDepsOfInversionNumMech( inversionNumMech: InversionNumMech ): RuleDependencies { return [] } function ruleDepsOfArrondiMech( arrondiMech: ArrondiMech ): RuleDependencies { const result = [ arrondiMech.explanation.decimals, arrondiMech.explanation.value ].flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfMaxMech(maxMech: MaxMech): RuleDependencies { const result = maxMech.explanation.flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfMinMech(minMech: MinMech): RuleDependencies { const result = minMech.explanation.flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfComposantesMech( composantesMech: ComposantesMech ): RuleDependencies { const result = composantesMech.explanation.flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfUneConditionsMech( uneConditionsMech: UneConditionsMech ): RuleDependencies { const result = uneConditionsMech.explanation.flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfToutesConditionsMech( toutesConditionsMech: ToutesConditionsMech ): RuleDependencies { const result = toutesConditionsMech.explanation.flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfSyncMech(_: SyncMech): RuleDependencies { return [] } function ruleDepsOfGrilleMech( grilleMech: GrilleMech ): RuleDependencies { const tranchesNodes = grilleMech.explanation.tranches.flatMap( ({ montant, plafond }) => [montant, plafond] ) const result = R.concat( [grilleMech.explanation.assiette, grilleMech.explanation.multiplicateur], tranchesNodes ).flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfTauxProgMech( tauxProgMech: TauxProgMech ): RuleDependencies { const tranchesNodes = tauxProgMech.explanation.tranches.flatMap( ({ plafond, taux }) => [plafond, taux] ) const result = R.concat( [ tauxProgMech.explanation.assiette, tauxProgMech.explanation.multiplicateur ], tranchesNodes ).flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } function ruleDepsOfDureeMech(dureeMech: DureeMech): RuleDependencies { const result = [ dureeMech.explanation.depuis, dureeMech.explanation["jusqu'à"] ].flatMap( R.partial>(ruleDepsOfNode, [ ruleName ]) ) return result } let result if (isApplicableSi(node)) { result = ruleDepsOfApplicableSi(node) } else if (isNonApplicableSi(node)) { result = ruleDepsOfNonApplicableSi(node) } else if (isFormule(node)) { result = ruleDepsOfFormule(node) } else if (isValue(node)) { result = ruleDepsOfValue(node) } else if (isOperation(node)) { result = ruleDepsOfOperation(node) } else if (isReference(node)) { result = ruleDepsOfReference(node) } else if (isPossibilities(node)) { result = ruleDepsOfPossibilities(node) } else if (isPossibilities2(node)) { result = ruleDepsOfPossibilities2(node) } else if (isRecalculMech(node)) { result = ruleDepsOfRecalculMech(node) } else if (isEncadrementMech(node)) { result = ruleDepsOfEncadrementMech(node) } else if (isSommeMech(node)) { result = ruleDepsOfSommeMech(node) } else if (isProduitMech(node)) { result = ruleDepsOfProduitMech(node) } else if (isVariationsMech(node)) { result = ruleDepsOfVariationsMech(node) } else if (isAllegementMech(node)) { result = ruleDepsOfAllegementMech(node) } else if (isBaremeMech(node)) { result = ruleDepsOfBaremeMech(node) } else if (isInversionNumMech(node)) { result = ruleDepsOfInversionNumMech(node) } else if (isArrondiMech(node)) { result = ruleDepsOfArrondiMech(node) } else if (isMaxMech(node)) { result = ruleDepsOfMaxMech(node) } else if (isMinMech(node)) { result = ruleDepsOfMinMech(node) } else if (isComposantesMech(node)) { result = ruleDepsOfComposantesMech(node) } else if (isUneConditionsMech(node)) { result = ruleDepsOfUneConditionsMech(node) } else if (isToutesConditionsMech(node)) { result = ruleDepsOfToutesConditionsMech(node) } else if (isSyncMech(node)) { result = ruleDepsOfSyncMech(node) } else if (isGrilleMech(node)) { result = ruleDepsOfGrilleMech(node) } else if (isTauxProgMech(node)) { result = ruleDepsOfTauxProgMech(node) } else if (isDureeMech(node)) { result = ruleDepsOfDureeMech(node) } if (result === undefined) { throw new Error( `This node doesn't have a visitor method defined: ${JSON.stringify( node, null, 4 )}` ) } return result } function ruleDepsOfRuleNode( rule: RuleNode ): RuleDependencies { const subNodes = [ rule.formule, rule['applicable si'], rule['non applicable si'] ].filter(x => x !== undefined) as Array const subNodesDeps = subNodes .map(x => ruleDepsOfNode(rule.dottedName, x)) .flat(1) const isDisabledByDependencies: RuleDependencies = rule.isDisabledBy.map( x => [x.dottedName, DependencyType.disabledBy] ) const replacedByDependencies: RuleDependencies = rule.replacedBy.map( x => [x.referenceNode.dottedName, DependencyType.replacedBy] ) return [subNodesDeps, isDisabledByDependencies, replacedByDependencies].flat( 1 ) } export function buildRulesDependencies( parsedRules: ParsedRules ): Array<[Names, RuleDependencies]> { // This stringPairs thing is necessary because `toPairs` is strictly considering that // object keys are strings (same for `Object.entries`). Maybe we should build our own // `toPairs`? const stringPairs: Array<[string, RuleNode]> = Object.entries( parsedRules ) const pairs: Array<[Names, RuleNode]> = stringPairs as Array< [Names, RuleNode] > return pairs.map(([dottedName, ruleNode]: [Names, RuleNode]): [ Names, RuleDependencies ] => [dottedName, ruleDepsOfRuleNode(ruleNode)]) }