2018-06-29 09:13:05 +00:00
import { treatString , treatNumber , treatObject , treatOther } from './treat'
2017-03-14 10:42:44 +00:00
import React from 'react'
2017-10-24 17:33:55 +00:00
import {
findRuleByDottedName ,
disambiguateRuleReference ,
2018-09-19 13:16:04 +00:00
findRule ,
ruleParents ,
joinName
2017-10-24 17:33:55 +00:00
} from './rules'
2018-06-19 08:25:53 +00:00
import {
2018-01-08 15:07:26 +00:00
curry ,
chain ,
cond ,
evolve ,
path ,
map ,
2018-04-12 13:43:02 +00:00
merge ,
2018-01-08 15:07:26 +00:00
keys ,
is ,
2018-09-19 13:16:04 +00:00
T ,
tail
2018-01-08 15:07:26 +00:00
} from 'ramda'
2018-06-19 08:25:53 +00:00
import { Node } from './mecanismViews/common'
2017-10-24 17:33:55 +00:00
import {
evaluateNode ,
rewriteNode ,
2018-04-12 13:43:02 +00:00
makeJsx ,
mergeMissing ,
2018-04-13 16:41:45 +00:00
bonus
2017-10-24 17:33:55 +00:00
} from './evaluation'
2018-04-26 12:24:47 +00:00
import { anyNull , val , undefOrTrue } from './traverse-common-functions'
2017-03-07 17:25:25 +00:00
2017-03-06 16:35:30 +00:00
/ *
Dans ce fichier , les règles YAML sont parsées .
Elles expriment un langage orienté expression , les expressions étant
- préfixes quand elles sont des 'mécanismes' ( des mot - clefs représentant des calculs courants dans la loi )
- infixes pour les feuilles : des tests d 'égalité, d' inclusion , des comparaisons sur des variables ou tout simplement la variable elle - même , ou une opération effectuée sur la variable
* /
2017-02-22 16:55:36 +00:00
/ *
- > Notre règle est naturellement un AST ( car notation préfixe dans le YAML )
- > préliminaire : les expression infixes devront être parsées ,
par exemple ainsi : https : //github.com/Engelberg/instaparse#transforming-the-tree
- > Notre règle entière est un AST , qu ' il faut maintenant traiter :
- faire le calcul ( déterminer les valeurs de chaque noeud )
- trouver les branches complètes pour déterminer les autres branches courtcircuitées
- ex . rule . formule est courtcircuitée si rule . non applicable est vrai
2017-04-24 18:03:38 +00:00
- les feuilles de 'une de ces conditions' sont courtcircuitées si l 'une d' elle est vraie
2017-02-22 16:55:36 +00:00
- les feuilles de "toutes ces conditions" sont courtcircuitées si l 'une d' elle est fausse
- ...
( - bonus : utiliser ces informations pour l ' ordre de priorité des variables inconnues )
- si une branche est incomplète et qu 'elle est de type numérique, déterminer les bornes si c' est possible .
Ex . - pour une multiplication , si l 'assiette est connue mais que l ' applicabilité est inconnue ,
les bornes seront [ 0 , multiplication . value = assiette * taux ]
- si taux = effectif entreprise >= 20 ? 1 % : 2 % et que l ' applicabilité est connue ,
bornes = [ assiette * 1 % , assiette * 2 % ]
- transformer l ' arbre en JSX pour afficher le calcul * et son état en prenant en compte les variables renseignées et calculées * de façon sympathique dans un butineur Web tel que Mozilla Firefox .
- surement plein d ' autres applications ...
* /
2018-06-29 09:13:05 +00:00
export let treat = ( rules , rule ) => rawNode => {
2018-01-08 15:07:26 +00:00
let onNodeType = cond ( [
2018-06-29 09:13:05 +00:00
[ is ( String ) , treatString ( rules , rule ) ] ,
2018-01-08 15:07:26 +00:00
[ is ( Number ) , treatNumber ] ,
2018-06-29 09:13:05 +00:00
[ is ( Object ) , treatObject ( rules , rule ) ] ,
2018-01-08 15:07:26 +00:00
[ T , treatOther ]
2017-06-27 21:09:03 +00:00
] )
2017-07-09 16:48:00 +00:00
2017-12-12 19:10:22 +00:00
let defaultEvaluate = ( cache , situationGate , parsedRules , node ) => node
2017-07-09 16:48:00 +00:00
let parsedNode = onNodeType ( rawNode )
2017-10-24 17:33:55 +00:00
return parsedNode . evaluate
? parsedNode
: { ... parsedNode , evaluate : defaultEvaluate }
2017-02-22 16:55:36 +00:00
}
2017-10-24 13:57:46 +00:00
export let computeRuleValue = ( formuleValue , isApplicable ) =>
isApplicable === true
2017-10-03 12:35:18 +00:00
? formuleValue
2018-04-26 12:24:47 +00:00
: isApplicable === false
2018-11-15 15:21:53 +00:00
? 0
: formuleValue == 0
? 0
: null
2017-05-07 17:45:44 +00:00
2017-07-17 17:39:49 +00:00
export let treatRuleRoot = ( rules , rule ) => {
2017-11-28 11:45:06 +00:00
/ *
La fonction treatRuleRoot va descendre l 'arbre de la règle `rule` et produire un AST, un objet contenant d' autres objets contenant d ' autres objets ...
2018-09-18 09:14:09 +00:00
Aujourd ' hui , une règle peut avoir ( comme propriétés à parser ) ` non applicable si ` , ` applicable si ` et ` formule ` ,
2017-11-28 11:45:06 +00:00
qui ont elles - mêmes des propriétés de type mécanisme ( ex . barème ) ou des expressions en ligne ( ex . maVariable + 3 ) .
Ces mécanismes où variables sont descendues à leur tour grâce à ` treat() ` .
2018-09-18 09:14:09 +00:00
Lors de ce traitement , des fonctions 'evaluate' et ` jsx ` sont attachés aux objets de l 'AST. Elles seront exécutées à l' évaluation .
2017-11-28 11:45:06 +00:00
* /
2018-04-10 12:13:37 +00:00
let evaluate = ( cache , situationGate , parsedRules , node ) => {
2018-04-26 12:24:47 +00:00
// console.log((cache.op || ">").padStart(cache.parseLevel),rule.dottedName)
2018-04-13 09:44:49 +00:00
cache . parseLevel ++
2018-04-12 10:25:01 +00:00
2018-01-08 15:07:26 +00:00
let evolveRule = curry ( evaluateNode ) ( cache , situationGate , parsedRules ) ,
evaluated = evolve (
2017-10-24 17:33:55 +00:00
{
formule : evolveRule ,
2018-09-19 13:16:04 +00:00
parentDependency : evolveRule ,
2017-10-24 17:33:55 +00:00
'non applicable si' : evolveRule ,
'applicable si' : evolveRule
} ,
2018-04-10 12:13:37 +00:00
node
2017-10-24 17:33:55 +00:00
) ,
2018-09-19 13:16:04 +00:00
parentValue = val ( evaluated [ 'parentDependency' ] ) ,
2017-10-24 13:57:46 +00:00
formuleValue = val ( evaluated [ 'formule' ] ) ,
2017-10-24 17:33:55 +00:00
isApplicable = do {
let e = evaluated
2018-09-19 13:16:04 +00:00
parentValue === false
? false
: val ( e [ 'non applicable si' ] ) === true
2017-10-24 13:57:46 +00:00
? false
: val ( e [ 'applicable si' ] ) === false
2018-11-15 15:21:53 +00:00
? false
2018-09-19 13:16:04 +00:00
: anyNull ( [ e [ 'non applicable si' ] , e [ 'applicable si' ] , parentValue ] )
2018-11-15 15:21:53 +00:00
? null
: ! val ( e [ 'non applicable si' ] ) && undefOrTrue ( val ( e [ 'applicable si' ] ) )
2017-10-24 13:57:46 +00:00
} ,
nodeValue = computeRuleValue ( formuleValue , isApplicable )
2017-11-07 10:07:02 +00:00
let {
formule ,
2018-09-19 13:16:04 +00:00
parentDependency ,
2017-11-07 10:07:02 +00:00
'non applicable si' : notApplicable ,
2017-11-20 12:16:18 +00:00
'applicable si' : applicable
2018-04-10 12:13:37 +00:00
} = evaluated
2017-11-07 10:07:02 +00:00
2017-11-07 18:46:40 +00:00
let condMissing =
2017-10-24 17:47:16 +00:00
val ( notApplicable ) === true
2018-04-12 13:43:02 +00:00
? { }
2017-10-24 17:47:16 +00:00
: val ( applicable ) === false
2018-11-15 15:21:53 +00:00
? { }
: merge (
2018-09-19 13:16:04 +00:00
( parentDependency && parentDependency . missingVariables ) || { } ,
2018-11-15 15:21:53 +00:00
( notApplicable && notApplicable . missingVariables ) || { } ,
( applicable && applicable . missingVariables ) || { }
) ,
2017-10-24 13:57:46 +00:00
collectInFormule = isApplicable !== false ,
2018-05-09 16:41:57 +00:00
formMissing =
( collectInFormule && formule && formule . missingVariables ) || { } ,
2018-04-13 16:41:45 +00:00
// On veut abaisser le score des conséquences par rapport aux conditions,
// mais seulement dans le cas où une condition est effectivement présente
hasCondition = keys ( condMissing ) . length > 0 ,
2018-04-26 12:24:47 +00:00
missingVariables = mergeMissing (
bonus ( condMissing , hasCondition ) ,
formMissing
)
2017-10-24 17:47:16 +00:00
2018-04-12 10:25:01 +00:00
cache . parseLevel --
2018-04-26 12:24:47 +00:00
// if (keys(condMissing).length) console.log("".padStart(cache.parseLevel-1),{conditions:condMissing, formule:formMissing})
// else console.log("".padStart(cache.parseLevel-1),{formule:formMissing})
2018-04-10 12:13:37 +00:00
return { ... evaluated , nodeValue , isApplicable , missingVariables }
2017-07-09 14:39:31 +00:00
}
2018-09-19 13:16:04 +00:00
// A parent dependency means that one of a rule's parents is a boolean question
// When the question is resolved to false, then the whole branch under it is disactivate, non applicable
// It lets those children omit parent applicability tests
let parentDependencies = ruleParents ( rule . dottedName )
. reverse ( )
. map ( joinName ) ,
parentDependency = parentDependencies . find (
parent => rules . find ( r => r . dottedName === parent ) ? . booleanNamespace
)
2018-01-08 15:07:26 +00:00
let parsedRoot = evolve ( {
2017-10-24 13:57:46 +00:00
// Voilà les attributs d'une règle qui sont aujourd'hui dynamiques, donc à traiter
// Les métadonnées d'une règle n'en font pas aujourd'hui partie
2017-02-22 16:55:36 +00:00
2017-10-24 13:57:46 +00:00
// condition d'applicabilité de la règle
2018-09-19 13:16:04 +00:00
parentDependency : parent => {
let evaluate = ( cache , situationGate , parsedRules , node ) => {
let explanation = evaluateNode (
cache ,
situationGate ,
parsedRules ,
node . explanation
) ,
[ nodeValue , missingVariables ] =
explanation . nodeValue === null
? [ null , { [ explanation . dottedName ] : 1 } ]
: explanation . nodeValue === false
? [ false , { } ]
: [ true , { } ]
return rewriteNode ( node , nodeValue , explanation , missingVariables )
}
let child = treat ( rules , rule ) ( parent )
let jsx = ( nodeValue , explanation ) => (
< div > Dépendance : { makeJsx ( explanation ) } < / d i v >
)
return {
evaluate ,
jsx ,
category : 'ruleProp' ,
rulePropType : 'formula' ,
name : 'formule' ,
type : 'numeric' ,
explanation : child
}
} ,
2017-10-24 13:57:46 +00:00
'non applicable si' : evolveCond ( 'non applicable si' , rule , rules ) ,
'applicable si' : evolveCond ( 'applicable si' , rule , rules ) ,
2018-09-19 13:16:04 +00:00
// formule de calcul
2017-10-24 17:33:55 +00:00
formule : value => {
2017-12-12 19:10:22 +00:00
let evaluate = ( cache , situationGate , parsedRules , node ) => {
2018-01-03 15:54:19 +00:00
let explanation = evaluateNode (
cache ,
2017-10-24 17:33:55 +00:00
situationGate ,
parsedRules ,
node . explanation
) ,
2018-04-10 11:23:06 +00:00
nodeValue = explanation . nodeValue ,
2018-04-10 12:13:37 +00:00
missingVariables = explanation . missingVariables
2018-04-10 11:23:06 +00:00
2018-04-10 12:13:37 +00:00
return rewriteNode ( node , nodeValue , explanation , missingVariables )
2017-07-09 16:48:00 +00:00
}
2017-07-17 17:39:49 +00:00
let child = treat ( rules , rule ) ( value )
2017-07-11 14:00:10 +00:00
2017-10-24 17:33:55 +00:00
let jsx = ( nodeValue , explanation ) => makeJsx ( explanation )
2017-07-13 19:55:59 +00:00
2017-03-16 18:30:30 +00:00
return {
2017-07-09 16:48:00 +00:00
evaluate ,
2017-07-13 19:55:59 +00:00
jsx ,
2017-03-16 18:30:30 +00:00
category : 'ruleProp' ,
rulePropType : 'formula' ,
name : 'formule' ,
type : 'numeric' ,
2017-07-13 19:55:59 +00:00
explanation : child
2017-03-16 18:30:30 +00:00
}
2017-03-01 10:21:53 +00:00
}
2018-09-19 13:16:04 +00:00
} ) ( { ... rule , ... ( parentDependency ? { parentDependency } : { } ) } )
2017-03-20 11:17:49 +00:00
2018-06-29 10:36:42 +00:00
let controls =
rule [ 'contrôles' ] &&
rule [ 'contrôles' ] . map ( control => {
let testExpression = treatString ( rules , rule ) ( control . si )
2018-06-29 13:46:42 +00:00
if ( ! testExpression . explanation )
throw new Error (
'Ce contrôle ne semble pas être compris :' + control [ 'si' ]
)
let otherVariables = testExpression . explanation . filter (
node =>
node . category === 'variable' && node . dottedName !== rule . dottedName
)
2018-09-11 14:14:46 +00:00
let isInputControl = ! otherVariables . length ,
level = control [ 'niveau' ]
2018-06-29 13:46:42 +00:00
2018-09-11 14:14:46 +00:00
if ( level === 'bloquant' && ! isInputControl ) {
throw new Error (
'Un contrôle ne peut être bloquant et invoquer des calculs de variables' +
control [ 'si' ] +
level
)
}
2018-07-03 10:19:19 +00:00
2018-06-29 13:46:42 +00:00
return {
2018-09-12 17:12:59 +00:00
dottedName : rule . dottedName ,
2018-06-29 13:46:42 +00:00
level : control [ 'niveau' ] ,
test : control [ 'si' ] ,
2018-06-29 16:14:00 +00:00
message : control [ 'message' ] ,
2018-06-29 13:46:42 +00:00
testExpression ,
2018-09-12 17:12:59 +00:00
solution : control [ 'solution' ] ,
isInputControl
2018-06-29 13:46:42 +00:00
}
2018-06-29 10:36:42 +00:00
} )
2018-06-29 09:13:05 +00:00
2017-07-09 14:39:31 +00:00
return {
2017-07-31 14:30:05 +00:00
// Pas de propriété explanation et jsx ici car on est parti du (mauvais) principe que 'non applicable si' et 'formule' sont particuliers, alors qu'ils pourraient être rangé avec les autres mécanismes
2017-07-09 14:39:31 +00:00
... parsedRoot ,
2017-07-11 14:00:10 +00:00
evaluate ,
2018-06-29 10:36:42 +00:00
parsed : true ,
controls
2017-03-16 18:30:30 +00:00
}
2017-07-09 14:39:31 +00:00
}
2017-02-22 16:55:36 +00:00
2017-10-24 13:57:46 +00:00
let evolveCond = ( name , rule , rules ) => value => {
2017-12-12 19:10:22 +00:00
let evaluate = ( cache , situationGate , parsedRules , node ) => {
2018-01-03 15:54:19 +00:00
let explanation = evaluateNode (
cache ,
2017-10-24 17:33:55 +00:00
situationGate ,
parsedRules ,
node . explanation
) ,
2018-04-10 11:23:06 +00:00
nodeValue = explanation . nodeValue ,
2018-04-10 12:13:37 +00:00
missingVariables = explanation . missingVariables
return rewriteNode ( node , nodeValue , explanation , missingVariables )
2017-10-24 13:57:46 +00:00
}
let child = treat ( rules , rule ) ( value )
let jsx = ( nodeValue , explanation ) => (
< Node
classes = "ruleProp mecanism cond"
name = { name }
value = { nodeValue }
child = {
explanation . category === 'variable' ? (
< div className = "node" > { makeJsx ( explanation ) } < / d i v >
) : (
makeJsx ( explanation )
)
}
/ >
)
return {
evaluate ,
jsx ,
category : 'ruleProp' ,
rulePropType : 'cond' ,
name ,
type : 'boolean' ,
explanation : child
}
}
2017-11-07 18:46:40 +00:00
export let getTargets = ( target , rules ) => {
2018-01-08 15:07:26 +00:00
let multiSimulation = path ( [ 'simulateur' , 'objectifs' ] ) ( target )
2017-11-07 18:46:40 +00:00
let targets = multiSimulation
? // On a un simulateur qui définit une liste d'objectifs
2018-03-14 15:58:12 +00:00
multiSimulation
2018-01-03 15:54:19 +00:00
. map ( n => disambiguateRuleReference ( rules , target , n ) )
. map ( n => findRuleByDottedName ( rules , n ) )
2017-11-07 18:46:40 +00:00
: // Sinon on est dans le cas d'une simple variable d'objectif
2018-03-14 15:58:12 +00:00
[ target ]
2017-11-07 18:46:40 +00:00
return targets
2017-07-09 22:14:22 +00:00
}
2017-11-28 11:45:06 +00:00
export let parseAll = flatRules => {
let treatOne = rule => treatRuleRoot ( flatRules , rule )
2018-01-08 15:07:26 +00:00
return map ( treatOne , flatRules )
2017-11-28 11:45:06 +00:00
}
2018-11-15 15:21:53 +00:00
let evaluateControls = blocking => ( cache , parsedRules , situationGate ) => {
2018-10-05 15:48:54 +00:00
return chain ( ( { controls , dottedName } ) =>
2018-11-20 14:39:00 +00:00
( controls || [ ] )
. filter ( ( { level } ) =>
2018-11-15 15:21:53 +00:00
blocking
? level === 'bloquant' && situationGate ( dottedName ) != undefined
: level !== 'bloquant'
2018-10-02 13:40:12 +00:00
)
. map ( control => ( {
... control ,
evaluated : evaluateNode (
2018-11-15 15:21:53 +00:00
cache ,
2018-10-02 13:40:12 +00:00
situationGate ,
parsedRules ,
control . testExpression
2018-06-29 13:46:42 +00:00
)
2018-10-02 13:40:12 +00:00
} ) )
. filter ( ( { evaluated : { nodeValue } } ) => nodeValue )
2018-09-11 14:14:46 +00:00
) ( parsedRules ) . filter ( found => found )
2018-06-29 13:46:42 +00:00
}
2017-11-28 11:45:06 +00:00
export let analyseMany = ( parsedRules , targetNames ) => situationGate => {
2018-01-03 15:54:19 +00:00
// TODO: we should really make use of namespaces at this level, in particular
// setRule in Rule.js needs to get smarter and pass dottedName
2018-04-26 12:24:47 +00:00
let cache = { parseLevel : 0 }
2017-12-12 19:10:22 +00:00
2018-09-11 14:14:46 +00:00
// These controls do not trigger the evaluation of variables of the system : they are input controls
// This is necessary because our evaluation implementation is not yet fast enough to not freeze slow mobile devices
// They could be implemented directly at the redux-form level, but they should also be triggered by the engine used as a library
2018-11-15 15:21:53 +00:00
let blockingInputControls = evaluateControls ( true ) (
cache ,
parsedRules ,
situationGate
)
2018-09-11 16:45:54 +00:00
if ( blockingInputControls . length )
return {
blockingInputControls
}
2018-11-15 15:21:53 +00:00
let nonBlockingControls = evaluateControls ( false ) (
cache ,
parsedRules ,
situationGate
)
2018-06-29 13:46:42 +00:00
2018-09-26 13:00:05 +00:00
let parsedTargets = targetNames . map ( t => {
let parsedTarget = findRule ( parsedRules , t )
if ( ! parsedTarget )
throw new Error (
` L'objectif de calcul " ${ t } " ne semble pas exister dans la base de règles `
)
return parsedTarget
} ) ,
2018-11-15 15:21:53 +00:00
targets = chain ( pt => getTargets ( pt , parsedRules ) , parsedTargets ) . map (
t =>
cache [ t . dottedName ] || // This check exists because it is not done in treatRuleRoot's eval, while it is in treatVariable. This should be merged : we should probably call treatVariable here : targetNames could be expressions (hence with filters) TODO
evaluateNode ( cache , situationGate , parsedRules , t )
2017-11-07 18:46:40 +00:00
)
2018-09-11 14:14:46 +00:00
return { targets , cache , controls : nonBlockingControls }
2017-11-28 11:45:06 +00:00
}
export let analyse = ( parsedRules , target ) => {
return analyseMany ( parsedRules , [ target ] )
2017-07-09 14:39:31 +00:00
}