2017-03-14 10:42:44 +00:00
import React from 'react'
2017-10-24 17:33:55 +00:00
import {
findRuleByDottedName ,
disambiguateRuleReference ,
findRuleByName
} from './rules'
import { evaluateVariable } from './variables'
2017-02-22 16:55:36 +00:00
import R from 'ramda'
2017-03-03 09:43:43 +00:00
import knownMecanisms from './known-mecanisms.yaml'
2017-03-07 17:25:25 +00:00
import { Parser } from 'nearley'
import Grammar from './grammar.ne'
2017-10-24 17:33:55 +00:00
import { Node , Leaf } from './mecanismViews/common'
2017-07-30 21:50:36 +00:00
import {
2017-10-24 17:33:55 +00:00
mecanismOneOf ,
mecanismAllOf ,
mecanismNumericalSwitch ,
mecanismSum ,
mecanismProduct ,
mecanismScale ,
mecanismMax ,
mecanismMin ,
mecanismError ,
mecanismComplement ,
2017-09-18 22:17:02 +00:00
mecanismSelection
2017-10-24 17:33:55 +00:00
} from './mecanisms'
import {
evaluateNode ,
rewriteNode ,
collectNodeMissing ,
makeJsx
} from './evaluation'
import {
anyNull ,
val ,
undefOrTrue ,
applyOrEmpty
} from './traverse-common-functions'
2017-03-07 17:25:25 +00:00
2017-11-06 19:12:08 +00:00
import uniroot from './uniroot'
2017-03-08 16:47:12 +00:00
let nearley = ( ) => new Parser ( Grammar . ParserRules , Grammar . ParserStart )
2017-01-09 17:17:51 +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 ...
* /
2017-07-17 17:39:49 +00:00
let fillFilteredVariableNode = ( rules , rule ) => ( filter , parseResult ) => {
2017-07-16 21:58:06 +00:00
let evaluateFiltered = originalEval => ( situation , parsedRules , node ) => {
2017-10-24 17:33:55 +00:00
let newSituation = name => ( name == 'sys.filter' ? filter : situation ( name ) )
2017-07-16 21:58:06 +00:00
return originalEval ( newSituation , parsedRules , node )
}
2017-07-17 17:39:49 +00:00
let node = fillVariableNode ( rules , rule ) ( parseResult )
2017-07-16 21:58:06 +00:00
return {
... node ,
evaluate : evaluateFiltered ( node . evaluate )
}
}
2017-07-31 15:32:59 +00:00
// TODO: dirty, dirty
// ne pas laisser trop longtemps cette "optimisation" qui tue l'aspect fonctionnel de l'algo
2017-10-24 17:33:55 +00:00
var dict
2017-07-31 15:32:59 +00:00
2017-10-24 17:33:55 +00:00
export let clearDict = ( ) => ( dict = { } )
2017-08-22 09:38:11 +00:00
2017-10-24 17:33:55 +00:00
let fillVariableNode = ( rules , rule ) => parseResult => {
2017-07-09 14:39:31 +00:00
let evaluate = ( situation , parsedRules , node ) => {
let dottedName = node . dottedName ,
2017-08-22 08:53:38 +00:00
// On va vérifier dans le cache courant, dict, si la variable n'a pas été déjà évaluée
// En effet, l'évaluation dans le cas d'une variable qui a une formule, est coûteuse !
2017-10-24 17:33:55 +00:00
filter = situation ( 'sys.filter' ) ,
cacheName = dottedName + ( filter ? '.' + filter : '' ) ,
2017-09-23 10:54:09 +00:00
cached = dict [ cacheName ] ,
2017-08-22 08:53:38 +00:00
// make parsedRules a dict object, that also serves as a cache of evaluation ?
2017-10-24 17:33:55 +00:00
variable = cached
? cached
: findRuleByDottedName ( parsedRules , dottedName ) ,
2017-09-08 10:06:53 +00:00
variableIsCalculable = variable . formule != null ,
2017-10-24 17:33:55 +00:00
parsedRule =
variableIsCalculable &&
( cached ? cached : evaluateNode ( situation , parsedRules , variable ) ) ,
2017-08-22 08:53:38 +00:00
// evaluateVariable renvoit la valeur déduite de la situation courante renseignée par l'utilisateur
2017-07-16 21:58:06 +00:00
situationValue = evaluateVariable ( situation , dottedName , variable ) ,
2017-10-24 17:33:55 +00:00
nodeValue =
situationValue != null
? situationValue // cette variable a été directement renseignée
2017-07-09 14:39:31 +00:00
: ! variableIsCalculable
2017-08-22 08:53:38 +00:00
? null // pas moyen de calculer car il n'y a pas de formule, elle restera donc nulle
: parsedRule . nodeValue , // la valeur du calcul fait foi
2017-07-09 14:39:31 +00:00
explanation = parsedRule ,
2017-09-08 10:28:35 +00:00
missingVariables = variableIsCalculable ? [ ] : [ dottedName ]
2017-07-09 14:39:31 +00:00
2017-10-24 17:33:55 +00:00
let collectMissing = node =>
nodeValue != null // notamment si situationValue != null
? [ ]
: variableIsCalculable
? collectNodeMissing ( parsedRule )
: node . missingVariables
let result = cached
? cached
: {
... rewriteNode ( node , nodeValue , explanation , collectMissing ) ,
missingVariables
2017-07-09 14:39:31 +00:00
}
2017-10-24 17:33:55 +00:00
dict [ cacheName ] = result
2017-07-31 15:32:59 +00:00
2017-10-24 17:33:55 +00:00
return result
2017-07-09 14:39:31 +00:00
}
2017-10-24 17:33:55 +00:00
let { fragments } = parseResult ,
2017-03-08 16:49:22 +00:00
variablePartialName = fragments . join ( ' . ' ) ,
2017-07-09 14:39:31 +00:00
dottedName = disambiguateRuleReference ( rules , rule , variablePartialName )
2017-03-08 16:49:22 +00:00
2017-11-07 18:46:40 +00:00
let jsx = nodeValue => (
2017-10-24 17:33:55 +00:00
< Leaf classes = "variable" name = { fragments . join ( ' . ' ) } value = { nodeValue } / >
)
2017-07-13 19:55:59 +00:00
2017-03-08 16:49:22 +00:00
return {
2017-07-09 14:39:31 +00:00
evaluate ,
2017-07-13 19:55:59 +00:00
jsx ,
2017-07-09 22:14:22 +00:00
name : variablePartialName ,
2017-03-08 16:49:22 +00:00
category : 'variable' ,
2017-07-13 19:55:59 +00:00
fragments ,
2017-05-09 15:14:13 +00:00
dottedName ,
2017-07-13 19:55:59 +00:00
type : 'boolean | numeric'
2017-03-08 16:49:22 +00:00
}
}
2017-05-09 13:39:42 +00:00
let buildNegatedVariable = variable => {
2017-07-13 10:12:19 +00:00
let evaluate = ( situation , parsedRules , node ) => {
2017-07-17 17:39:49 +00:00
let explanation = evaluateNode ( situation , parsedRules , node . explanation ) ,
2017-07-13 10:12:19 +00:00
nodeValue = explanation . nodeValue == null ? null : ! explanation . nodeValue
let collectMissing = node => collectNodeMissing ( node . explanation )
2017-10-24 17:33:55 +00:00
return rewriteNode ( node , nodeValue , explanation , collectMissing )
2017-07-13 10:12:19 +00:00
}
2017-10-24 17:33:55 +00:00
let jsx = ( nodeValue , explanation ) => (
2017-07-13 19:55:59 +00:00
< Node
2017-05-09 13:39:42 +00:00
classes = "inlineExpression negation"
2017-07-29 08:00:49 +00:00
value = { nodeValue }
2017-05-09 13:39:42 +00:00
child = {
< span className = "nodeContent" >
< span className = "operator" > ¬ < / s p a n >
2017-07-13 19:55:59 +00:00
{ makeJsx ( explanation ) }
2017-05-09 13:39:42 +00:00
< / s p a n >
}
/ >
2017-10-24 17:33:55 +00:00
)
2017-07-13 19:55:59 +00:00
return {
evaluate ,
jsx ,
category : 'mecanism' ,
name : 'négation' ,
type : 'boolean' ,
explanation : variable
2017-05-09 13:39:42 +00:00
}
}
2017-07-17 17:39:49 +00:00
let treat = ( rules , rule ) => rawNode => {
2017-06-27 21:09:03 +00:00
// inner functions
2017-10-24 17:33:55 +00:00
let reTreat = treat ( rules , rule ) ,
2017-06-27 21:09:03 +00:00
treatString = rawNode => {
2017-07-31 14:30:05 +00:00
/ * O n a a f f a i r e à u n s t r i n g , d o n c à u n e e x p r e s s i o n i n f i x e .
Elle sera traité avec le parser obtenu grâce à NearleyJs et notre grammaire ` grammar.ne ` .
2017-06-27 21:09:03 +00:00
On obtient un objet de type Variable ( avec potentiellement un 'modifier' , par exemple temporel ( TODO ) ) , CalcExpression ou Comparison .
Cet objet est alors rebalancé à 'treat' .
* /
2017-03-08 16:47:12 +00:00
2017-06-27 21:09:03 +00:00
let [ parseResult , ... additionnalResults ] = nearley ( ) . feed ( rawNode ) . results
if ( additionnalResults && additionnalResults . length > 0 )
2017-10-24 17:33:55 +00:00
throw 'Attention ! L\'expression <' +
rawNode +
'> ne peut être traitée de façon univoque'
if (
! R . contains ( parseResult . category ) ( [
'variable' ,
'calcExpression' ,
'filteredVariable' ,
'comparison' ,
'negatedVariable' ,
'percentage'
] )
)
throw 'Attention ! Erreur de traitement de l\'expression : ' + rawNode
2017-06-27 21:09:03 +00:00
if ( parseResult . category == 'variable' )
2017-07-17 17:39:49 +00:00
return fillVariableNode ( rules , rule ) ( parseResult )
2017-06-29 10:03:01 +00:00
if ( parseResult . category == 'filteredVariable' ) {
2017-10-24 17:33:55 +00:00
return fillFilteredVariableNode ( rules , rule ) (
parseResult . filter ,
parseResult . variable
)
2017-06-29 10:03:01 +00:00
}
2017-06-27 21:09:03 +00:00
if ( parseResult . category == 'negatedVariable' )
return buildNegatedVariable (
2017-07-17 17:39:49 +00:00
fillVariableNode ( rules , rule ) ( parseResult . variable )
2017-06-27 21:09:03 +00:00
)
2017-09-19 11:17:43 +00:00
// We don't need to handle category == 'value' because YAML then returns it as
// numerical value, not a String: it goes to treatNumber
if ( parseResult . category == 'percentage' ) {
return {
nodeValue : parseResult . nodeValue ,
2017-10-24 17:33:55 +00:00
jsx : ( ) => < span className = "percentage" > { rawNode } < / s p a n >
2017-09-19 11:17:43 +00:00
}
}
2017-10-24 17:33:55 +00:00
if (
parseResult . category == 'calcExpression' ||
parseResult . category == 'comparison'
) {
2017-07-09 16:48:00 +00:00
let evaluate = ( situation , parsedRules , node ) => {
2017-10-24 17:33:55 +00:00
let operatorFunctionName = {
2017-07-09 16:48:00 +00:00
'*' : 'multiply' ,
'/' : 'divide' ,
'+' : 'add' ,
2017-07-28 09:30:23 +00:00
'-' : 'subtract' ,
'<' : 'lt' ,
'<=' : 'lte' ,
'>' : 'gt' ,
2017-09-07 14:52:14 +00:00
'>=' : 'gte' ,
2017-09-07 21:26:31 +00:00
'=' : 'equals' ,
'!=' : 'equals'
2017-07-09 16:48:00 +00:00
} [ node . operator ] ,
2017-10-24 17:33:55 +00:00
explanation = R . map (
R . curry ( evaluateNode ) ( situation , parsedRules ) ,
node . explanation
) ,
2017-07-11 14:00:10 +00:00
value1 = explanation [ 0 ] . nodeValue ,
value2 = explanation [ 1 ] . nodeValue ,
2017-10-24 17:33:55 +00:00
operatorFunction =
node . operator == '!='
? ( a , b ) => ! R . equals ( a , b )
: R [ operatorFunctionName ] ,
nodeValue =
value1 == null || value2 == null
? null
: operatorFunction ( value1 , value2 )
let collectMissing = node =>
R . chain ( collectNodeMissing , node . explanation )
return rewriteNode ( node , nodeValue , explanation , collectMissing )
2017-07-09 16:48:00 +00:00
}
2017-10-24 17:33:55 +00:00
let fillFiltered = parseResult =>
fillFilteredVariableNode ( rules , rule ) (
parseResult . filter ,
parseResult . variable
)
2017-07-17 17:39:49 +00:00
let fillVariable = fillVariableNode ( rules , rule ) ,
2017-06-27 21:09:03 +00:00
filledExplanation = parseResult . explanation . map (
R . cond ( [
2017-06-29 10:03:01 +00:00
[ R . propEq ( 'category' , 'variable' ) , fillVariable ] ,
[ R . propEq ( 'category' , 'filteredVariable' ) , fillFiltered ] ,
2017-10-24 17:33:55 +00:00
[
R . propEq ( 'category' , 'value' ) ,
node => ( {
2017-09-19 11:17:43 +00:00
nodeValue : node . nodeValue ,
2017-10-24 17:33:55 +00:00
jsx : nodeValue => < span className = "value" > { nodeValue } < / s p a n >
2017-07-09 16:48:00 +00:00
} )
2017-09-19 11:17:43 +00:00
] ,
2017-10-24 17:33:55 +00:00
[
R . propEq ( 'category' , 'percentage' ) ,
node => ( {
2017-09-19 11:17:43 +00:00
nodeValue : node . nodeValue ,
2017-10-24 17:33:55 +00:00
jsx : nodeValue => (
< span className = "value" > { nodeValue * 100 } % < / s p a n >
)
2017-09-19 11:17:43 +00:00
} )
2017-06-27 21:09:03 +00:00
]
] )
) ,
2017-07-09 16:48:00 +00:00
operator = parseResult . operator
2017-06-27 21:09:03 +00:00
2017-10-24 17:33:55 +00:00
let jsx = ( nodeValue , explanation ) => (
2017-07-13 20:45:54 +00:00
< Node
2017-10-24 17:33:55 +00:00
classes = { 'inlineExpression ' + parseResult . category }
2017-07-13 19:55:59 +00:00
value = { nodeValue }
2017-06-27 21:09:03 +00:00
child = {
< span className = "nodeContent" >
2017-07-13 19:55:59 +00:00
{ makeJsx ( explanation [ 0 ] ) }
2017-06-27 21:09:03 +00:00
< span className = "operator" > { parseResult . operator } < / s p a n >
2017-07-13 19:55:59 +00:00
{ makeJsx ( explanation [ 1 ] ) }
2017-06-27 21:09:03 +00:00
< / s p a n >
}
/ >
2017-10-24 17:33:55 +00:00
)
2017-07-13 19:55:59 +00:00
return {
evaluate ,
jsx ,
operator ,
text : rawNode ,
2017-07-28 09:30:23 +00:00
category : parseResult . category ,
2017-10-24 17:33:55 +00:00
type :
parseResult . category == 'calcExpression' ? 'numeric' : 'boolean' ,
2017-07-13 19:55:59 +00:00
explanation : filledExplanation
2017-06-27 21:09:03 +00:00
}
}
} ,
treatNumber = rawNode => {
2017-03-09 15:42:52 +00:00
return {
2017-10-24 17:33:55 +00:00
text : '' + rawNode ,
2017-06-27 21:09:03 +00:00
category : 'number' ,
nodeValue : rawNode ,
2017-03-09 15:42:52 +00:00
type : 'numeric' ,
2017-10-24 17:33:55 +00:00
jsx : < span className = "number" > { rawNode } < / s p a n >
2017-03-09 15:42:52 +00:00
}
2017-06-27 21:09:03 +00:00
} ,
treatOther = rawNode => {
console . log ( ) // eslint-disable-line no-console
2017-10-24 17:33:55 +00:00
throw 'Cette donnée : ' +
rawNode +
' doit être un Number, String ou Object'
2017-06-27 21:09:03 +00:00
} ,
2017-06-27 23:02:36 +00:00
treatObject = rawNode => {
let mecanisms = R . intersection ( R . keys ( rawNode ) , R . keys ( knownMecanisms ) )
if ( mecanisms . length != 1 ) {
2017-11-07 18:46:40 +00:00
console . log (
// eslint-disable-line no-console
2017-10-24 17:33:55 +00:00
'Erreur : On ne devrait reconnaître que un et un seul mécanisme dans cet objet' ,
mecanisms ,
rawNode
)
2017-06-27 23:02:36 +00:00
throw 'OUPS !'
2017-06-27 21:09:03 +00:00
}
2017-04-24 18:03:38 +00:00
2017-06-27 23:02:36 +00:00
let k = R . head ( mecanisms ) ,
v = rawNode [ k ]
2017-06-28 07:06:52 +00:00
let dispatch = {
2017-10-24 17:33:55 +00:00
'une de ces conditions' : mecanismOneOf ,
'toutes ces conditions' : mecanismAllOf ,
'aiguillage numérique' : mecanismNumericalSwitch ,
somme : mecanismSum ,
multiplication : mecanismProduct ,
barème : mecanismScale ,
'le maximum de' : mecanismMax ,
'le minimum de' : mecanismMin ,
complément : mecanismComplement ,
sélection : mecanismSelection ,
'une possibilité' : R . always ( {
'une possibilité' : 'oui' ,
collectMissing : ( ) => [ rule . dottedName ]
} )
2017-06-28 07:06:52 +00:00
} ,
2017-07-31 14:30:05 +00:00
action = R . propOr ( mecanismError , k , dispatch )
2017-04-06 16:34:00 +00:00
2017-10-24 17:33:55 +00:00
return action ( reTreat , k , v )
2017-03-03 09:43:43 +00:00
}
2017-02-22 16:55:36 +00:00
2017-06-27 21:09:03 +00:00
let onNodeType = R . cond ( [
2017-10-24 17:33:55 +00:00
[ R . is ( String ) , treatString ] ,
[ R . is ( Number ) , treatNumber ] ,
[ R . is ( Object ) , treatObject ] ,
[ R . T , treatOther ]
2017-06-27 21:09:03 +00:00
] )
2017-07-09 16:48:00 +00:00
2017-07-11 15:39:50 +00:00
let defaultEvaluate = ( 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
2017-10-24 17:33:55 +00:00
: isApplicable === false ? 0 : formuleValue == 0 ? 0 : null
2017-05-07 17:45:44 +00:00
2017-11-07 10:07:02 +00:00
export let findInversion = ( situationGate , rules , rule ) => {
let inversions = rule [ 'inversions possibles' ]
if ( ! inversions ) return null
/ *
Quelle variable d ' inversion possible a sa valeur renseignée dans la situation courante ?
Ex . s 'il nous est demandé de calculer le salaire de base, est-ce qu' un candidat à l ' inversion , comme
le salaire net , a été renseigné ?
* /
let fixedObjective = inversions . find ( name => situationGate ( name ) != undefined ) //TODO ça va foirer avec des espaces de nom
if ( fixedObjective == null ) return null
//par exemple, fixedObjective = 'salaire net', et v('salaire net') == 2000
return {
fixedObjective ,
fixedObjectiveValue : situationGate ( fixedObjective ) ,
fixedObjectiveRule : findRuleByName ( rules , fixedObjective )
}
}
2017-11-06 18:36:14 +00:00
2017-07-17 17:39:49 +00:00
export let treatRuleRoot = ( rules , rule ) => {
2017-11-07 10:07:02 +00:00
let evaluate = ( situationGate , parsedRules , r ) => {
let inversion = findInversion ( situationGate , parsedRules , r )
if ( inversion ) {
2017-11-07 18:46:40 +00:00
let { fixedObjectiveValue , fixedObjectiveRule } = inversion
let fx = x =>
evaluateNode (
n => ( r . name === n || n === 'sys.filter' ? x : situationGate ( n ) ) , //TODO pourquoi doit-on nous préoccuper de sys.filter ?
parsedRules ,
fixedObjectiveRule
) . nodeValue ,
2017-11-07 10:07:02 +00:00
tolerancePercentage = 0.00001 ,
// cette fonction détermine la racine d'une fonction sans faire trop d'itérations
nodeValue = uniroot (
x => fx ( x ) - fixedObjectiveValue ,
0 ,
1000000000 ,
tolerancePercentage * fixedObjectiveValue ,
100
)
2017-11-06 18:36:14 +00:00
2017-11-07 18:46:40 +00:00
// si fx renvoie null pour une valeur numérique standard, disons 1000, on peut
// considérer que l'inversion est impossible du fait de variables manquantes
// TODO fx peut être null pour certains x, et valide pour d'autres : on peut implémenter ici le court-circuit
return fx ( 1000 ) == null
? {
... r ,
nodeValue : null ,
inversionMissingVariables : collectNodeMissing (
evaluateNode (
n =>
r . name === n || n === 'sys.filter' ? 1000 : situationGate ( n ) , //TODO pourquoi doit-on nous préoccuper de sys.filter ?
parsedRules ,
fixedObjectiveRule
)
)
}
: { ... r , nodeValue }
2017-11-06 18:36:14 +00:00
}
2017-10-24 17:33:55 +00:00
let evolveRule = R . curry ( evaluateNode ) ( situationGate , parsedRules ) ,
evaluated = R . evolve (
{
formule : evolveRule ,
'non applicable si' : evolveRule ,
'applicable si' : evolveRule
} ,
r
) ,
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
2017-10-24 13:57:46 +00:00
val ( e [ 'non applicable si' ] ) === true
? false
: val ( e [ 'applicable si' ] ) === false
? false
2017-10-24 17:33:55 +00:00
: anyNull ( [ e [ 'non applicable si' ] , e [ 'applicable si' ] ] )
2017-10-24 13:57:46 +00:00
? null
2017-10-24 17:33:55 +00:00
: ! val ( e [ 'non applicable si' ] ) &&
undefOrTrue ( val ( e [ 'applicable si' ] ) )
2017-10-24 13:57:46 +00:00
} ,
nodeValue = computeRuleValue ( formuleValue , isApplicable )
2017-10-24 17:33:55 +00:00
return { ... evaluated , nodeValue , isApplicable }
2017-07-11 14:00:10 +00:00
}
2017-10-24 17:47:16 +00:00
2017-11-07 18:46:40 +00:00
let collectMissing = rule => {
2017-11-07 10:07:02 +00:00
let {
formule ,
isApplicable ,
'non applicable si' : notApplicable ,
'applicable si' : applicable ,
inversionMissingVariables
} = rule
if ( inversionMissingVariables ) {
return inversionMissingVariables
}
2017-11-07 18:46:40 +00:00
let condMissing =
2017-10-24 17:47:16 +00:00
val ( notApplicable ) === true
? [ ]
: val ( applicable ) === false
? [ ]
: [
... applyOrEmpty ( collectNodeMissing ) ( notApplicable ) ,
... applyOrEmpty ( collectNodeMissing ) ( applicable )
] ,
2017-10-24 13:57:46 +00:00
collectInFormule = isApplicable !== false ,
2017-10-24 17:33:55 +00:00
formMissing = applyOrEmpty ( ( ) =>
applyOrEmpty ( collectNodeMissing ) ( formule )
2017-10-24 13:57:46 +00:00
) ( collectInFormule )
2017-10-24 17:47:16 +00:00
2017-10-24 17:33:55 +00:00
return R . concat ( condMissing , formMissing )
2017-07-09 14:39:31 +00:00
}
2017-10-24 13:57:46 +00:00
let parsedRoot = R . evolve ( {
// 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
'non applicable si' : evolveCond ( 'non applicable si' , rule , rules ) ,
'applicable si' : evolveCond ( 'applicable si' , rule , rules ) ,
2017-10-24 17:33:55 +00:00
formule : value => {
2017-07-09 16:48:00 +00:00
let evaluate = ( situationGate , parsedRules , node ) => {
2017-07-13 10:25:46 +00:00
let collectMissing = node => collectNodeMissing ( node . explanation )
2017-10-24 17:33:55 +00:00
let explanation = evaluateNode (
situationGate ,
parsedRules ,
node . explanation
) ,
2017-07-11 14:00:10 +00:00
nodeValue = explanation . nodeValue
2017-10-24 17:33:55 +00:00
return rewriteNode ( node , nodeValue , explanation , collectMissing )
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
}
2017-07-09 14:39:31 +00:00
} ) ( rule )
2017-03-20 11:17:49 +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 ,
collectMissing
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 => {
let evaluate = ( situationGate , parsedRules , node ) => {
let collectMissing = node => collectNodeMissing ( node . explanation )
2017-10-24 17:33:55 +00:00
let explanation = evaluateNode (
situationGate ,
parsedRules ,
node . explanation
) ,
2017-10-24 13:57:46 +00:00
nodeValue = explanation . nodeValue
2017-10-24 17:33:55 +00:00
return rewriteNode ( node , nodeValue , explanation , collectMissing )
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 ) => {
let multiSimulation = R . path ( [ 'simulateur' , 'objectifs' ] ) ( target )
let targets = multiSimulation
? // On a un simulateur qui définit une liste d'objectifs
multiSimulation
. map ( n => disambiguateRuleReference ( rules , target , n ) )
. map ( n => findRuleByDottedName ( rules , n ) )
: // Sinon on est dans le cas d'une simple variable d'objectif
[ target ]
return targets
2017-07-09 22:14:22 +00:00
}
2017-11-07 18:46:40 +00:00
export let analyse = ( rules , targetName ) => situationGate => {
2017-08-22 09:38:11 +00:00
clearDict ( )
2017-10-24 17:33:55 +00:00
let / *
2017-07-31 14:30:05 +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 ...
Aujourd ' hui , une règle peut avoir ( comme propriétés à parser ) ` non applicable si ` et ` formule ` ,
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() ` .
Lors de ce traitement , des fonctions 'evaluate' , ` collectMissingVariables ` et ` jsx ` sont attachés aux objets de l ' AST
* /
treatOne = rule => treatRuleRoot ( rules , rule ) ,
//On fait ainsi pour chaque règle de la base.
2017-10-24 17:33:55 +00:00
parsedRules = R . map ( treatOne , rules ) ,
2017-09-29 20:59:20 +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
2017-11-07 18:46:40 +00:00
parsedTarget = findRuleByName ( parsedRules , targetName ) ,
2017-07-31 14:30:05 +00:00
/ *
Ce n 'est que dans cette nouvelle étape que l' arbre est vraiment évalué .
Auparavant , l 'évaluation était faite lors de la construction de l' AST .
* /
2017-11-07 18:46:40 +00:00
targets = getTargets ( parsedTarget , parsedRules ) . map ( t =>
evaluateNode ( situationGate , parsedRules , t )
)
2017-01-26 12:19:04 +00:00
2017-07-09 22:14:22 +00:00
return {
2017-11-07 18:46:40 +00:00
targets ,
2017-07-09 22:14:22 +00:00
parsedRules
}
2017-07-09 14:39:31 +00:00
}