))}
}
/>
)
let evaluate = (situationGate, parsedRules, node) => {
let evaluateOne = child => evaluateNode(situationGate, parsedRules, child),
explanation = R.map(evaluateOne, node.explanation),
values = R.pluck('nodeValue', explanation),
nodeValue = R.any(R.equals(false), values)
? false // court-circuit
: R.any(R.equals(null), values) ? null : true
let collectMissing = node =>
node.nodeValue == null
? R.chain(collectNodeMissing, node.explanation)
: []
return rewriteNode(node, nodeValue, explanation, collectMissing)
}
return {
evaluate: evaluate,
jsx,
explanation,
category: 'mecanism',
name: 'toutes ces conditions',
type: 'boolean'
}
}
export let mecanismNumericalSwitch = (recurse, k, v) => {
// Si "l'aiguillage" est une constante ou une référence directe à une variable;
// l'utilité de ce cas correspond à un appel récursif au mécanisme
if (R.is(String, v)) return recurse(v)
if (!R.is(Object, v) || R.keys(v).length == 0) {
throw 'Le mécanisme "aiguillage numérique" et ses sous-logiques doivent contenir au moins une proposition'
}
// les termes sont les couples (condition, conséquence) de l'aiguillage numérique
let terms = R.toPairs(v)
// la conséquence peut être un 'string' ou un autre aiguillage numérique
let parseCondition = ([condition, consequence]) => {
let conditionNode = recurse(condition), // can be a 'comparison', a 'variable', TODO a 'negation'
consequenceNode = mecanismNumericalSwitch(recurse, condition, consequence)
let evaluate = (situationGate, parsedRules, node) => {
let collectMissing = node => {
let missingOnTheLeft = collectNodeMissing(node.explanation.condition),
investigate = node.explanation.condition.nodeValue !== false,
missingOnTheRight = investigate
? collectNodeMissing(node.explanation.consequence)
: []
return R.concat(missingOnTheLeft, missingOnTheRight)
}
let explanation = R.evolve(
{
condition: R.curry(evaluateNode)(situationGate, parsedRules),
consequence: R.curry(evaluateNode)(situationGate, parsedRules)
},
node.explanation
)
return {
...node,
collectMissing,
explanation,
nodeValue: explanation.consequence.nodeValue,
condValue: explanation.condition.nodeValue
}
}
let jsx = (nodeValue, { condition, consequence }) => (
{makeJsx(condition)}
{makeJsx(consequence)}
)
return {
evaluate,
jsx,
explanation: { condition: conditionNode, consequence: consequenceNode },
category: 'condition',
text: condition,
condition: conditionNode,
type: 'boolean'
}
}
let evaluateTerms = (situationGate, parsedRules, node) => {
let evaluateOne = child => evaluateNode(situationGate, parsedRules, child),
explanation = R.map(evaluateOne, node.explanation),
nonFalsyTerms = R.filter(node => node.condValue !== false, explanation),
getFirst = prop => R.pipe(R.head, R.prop(prop))(nonFalsyTerms),
nodeValue =
// voilà le "numérique" dans le nom de ce mécanisme : il renvoie zéro si aucune condition n'est vérifiée
R.isEmpty(nonFalsyTerms)
? 0
: // c'est un 'null', on renvoie null car des variables sont manquantes
getFirst('condValue') == null
? null
: // c'est un true, on renvoie la valeur de la conséquence
getFirst('nodeValue')
let collectMissing = node => {
let choice = R.find(node => node.condValue, node.explanation)
return choice
? collectNodeMissing(choice)
: R.chain(collectNodeMissing, node.explanation)
}
return rewriteNode(node, nodeValue, explanation, collectMissing)
}
let explanation = R.map(parseCondition, terms)
let jsx = (nodeValue, explanation) => (
{explanation.map(item => (
{makeJsx(item)}
))}
}
/>
)
return {
evaluate: evaluateTerms,
jsx,
explanation,
category: 'mecanism',
name: 'aiguillage numérique',
type: 'boolean || numeric' // lol !
}
}
export let findInversion = (situationGate, rules, v, dottedName) => {
let inversions = v.avec
if (!inversions)
throw 'Une formule d\'inversion doit préciser _avec_ quoi on peut inverser la variable'
/*
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
.map(i => disambiguateRuleReference(rules, rules.find(R.propEq('dottedName', dottedName)), i))
.find(name => situationGate(name) != undefined)
if (fixedObjective == null) return {inversionChoiceNeeded: true}
//par exemple, fixedObjective = 'salaire net', et v('salaire net') == 2000
return {
fixedObjective,
fixedObjectiveValue: situationGate(fixedObjective),
fixedObjectiveRule: findRuleByDottedName(rules, fixedObjective)
}
}
let doInversion = (situationGate, parsedRules, v, dottedName) => {
let inversion = findInversion(situationGate, parsedRules, v, dottedName)
if (inversion.inversionChoiceNeeded) return {
inversionMissingVariables: [dottedName],
nodeValue: null
}
let { fixedObjectiveValue, fixedObjectiveRule } = inversion
let fx = x =>
console.log('fx') || clearDict() && evaluateNode(
n => dottedName === n ? x : situationGate(n),
parsedRules,
fixedObjectiveRule
).nodeValue
// 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
if (fx(1000) == null)
return {
nodeValue: null,
inversionMissingVariables: collectNodeMissing(
evaluateNode(
n =>
dottedName === n ? 1000 : situationGate(n),
parsedRules,
fixedObjectiveRule
)
)
}
console.log('uniroot', dottedName, inversion.fixedObjective)
let 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
)
return {
nodeValue,
inversionMissingVariables: []
}
}
export let mecanismInversion = dottedName => (recurse, k, v) => {
let evaluate = (situationGate, parsedRules, node) => {
let inversion =
// avoid the inversion loop !
situationGate(dottedName) == undefined &&
doInversion(situationGate, parsedRules, v, dottedName)
let
collectMissing = () => inversion.inversionMissingVariables,
nodeValue = inversion.nodeValue
return rewriteNode(node, nodeValue, null, collectMissing)
}
return {
evaluate,
jsx: () => (