2020-03-30 17:14:03 +00:00
import { dropLast , last , pipe , propEq , range , take } from 'ramda'
2020-03-26 15:03:19 +00:00
import { coerceArray } from '../utils'
2020-04-23 07:30:03 +00:00
import { EvaluatedRule , Rule , Rules } from './types'
2020-03-26 15:03:19 +00:00
2020-04-29 14:19:20 +00:00
export const splitName = ( str : string ) = > str . split ( ' . ' )
2020-03-30 17:14:03 +00:00
export const joinName = strs = > strs . join ( ' . ' )
export const parentName = pipe (
splitName ,
dropLast ( 1 ) as ( a : string [ ] ) = > string [ ] ,
joinName
)
2020-03-26 15:03:19 +00:00
export const nameLeaf = pipe ( splitName , last )
export const encodeRuleName = name = >
encodeURI (
name
. replace ( /\s\.\s/g , '/' )
. replace ( /-/g , '\u2011' ) // replace with a insecable tiret to differenciate from space
. replace ( /\s/g , '-' )
)
2020-04-30 15:13:45 +00:00
export const decodeRuleName = name = >
2020-03-26 15:03:19 +00:00
decodeURI (
name
. replace ( /\//g , ' . ' )
. replace ( /-/g , ' ' )
. replace ( /\u2011/g , '-' )
)
2020-03-30 17:14:03 +00:00
export function ruleParents < Names extends string > (
dottedName : Names
) : Array < Names > {
2020-04-30 15:13:45 +00:00
const fragments = splitName ( dottedName ) // dottedName ex. [CDD . événements . rupture]
2020-03-26 15:03:19 +00:00
return range ( 1 , fragments . length )
2020-03-30 17:14:03 +00:00
. map ( nbEl = > take ( nbEl , fragments ) )
2020-03-26 15:03:19 +00:00
. map ( joinName ) // -> [ [CDD . événements . rupture], [CDD . événements], [CDD
. reverse ( )
}
2020-03-30 17:14:03 +00:00
export function disambiguateRuleReference < Names extends string > (
rules : Rules < Names > ,
contextName : Names ,
partialName : string
) {
2020-03-26 15:03:19 +00:00
const possibleDottedName = [
contextName ,
. . . ruleParents ( contextName ) ,
''
] . map ( x = > ( x ? x + ' . ' + partialName : partialName ) )
const dottedName = possibleDottedName . find ( name = > name in rules )
if ( ! dottedName ) {
throw new Error ( ` La référence ' ${ partialName } ' est introuvable.
V é rifiez que l 'orthographe et l' espace de nom sont corrects ` )
}
return dottedName
}
/ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
Autres
* /
2020-03-30 17:14:03 +00:00
export function findParentDependencies < Names extends string > (
rules : Rules < Names > ,
name : Names
) : Array < Names > {
2020-03-26 15:03:19 +00:00
// A parent dependency means that one of a rule's parents is not just a namespace holder, it is a boolean question. E.g. is it a fixed-term contract, yes / no
// When it is resolved to false, then the whole branch under it is disactivated (non applicable)
// It lets those children omit obvious and repetitive parent applicability tests
2020-03-30 17:14:03 +00:00
return ruleParents ( name )
. map ( parent = > [ parent , rules [ parent ] ] as [ Names , Rule ] )
. filter ( ( [ _ , rule ] ) = > ! ! rule )
. filter (
( [ _ , { question , unit é , formule } ] ) = >
//Find the first "calculable" parent
( question && ! unit é && ! formule ) ||
2020-03-26 15:03:19 +00:00
( question && formule ? . [ 'une possibilité' ] !== undefined ) ||
( typeof formule === 'string' && formule . includes ( ' = ' ) ) ||
formule === 'oui' ||
formule === 'non' ||
formule ? . [ 'une de ces conditions' ] ||
formule ? . [ 'toutes ces conditions' ]
2020-03-30 17:14:03 +00:00
)
. map ( ( [ name , _ ] ) = > name )
2020-03-26 15:03:19 +00:00
}
2020-04-30 15:13:45 +00:00
export const getRuleFromAnalysis = analysis = > < Names extends string > (
2020-03-30 17:14:03 +00:00
dottedName : Names
) : EvaluatedRule < Names > = > {
2020-03-26 15:03:19 +00:00
if ( ! analysis ) {
throw new Error ( "[getRuleFromAnalysis] The analysis can't be nil !" )
}
2020-04-30 15:13:45 +00:00
const rule = coerceArray ( analysis ) // In some simulations, there are multiple "branches" : the analysis is run with e.g. 3 different input situations
2020-03-26 15:03:19 +00:00
. map (
analysis = >
analysis . cache [ dottedName ] ? . explanation || // the cache stores a reference to a variable, the variable is contained in the 'explanation' attribute
analysis . targets . find ( propEq ( 'dottedName' , dottedName ) )
)
. filter ( Boolean ) [ 0 ]
if ( process . env . NODE_ENV !== 'production' && ! rule ) {
console . warn ( ` [getRuleFromAnalysis] Unable to find the rule ${ dottedName } ` )
}
return rule
}