196 lines
6.2 KiB
JavaScript
196 lines
6.2 KiB
JavaScript
|
|
// Séparation artificielle, temporaire, entre ces deux types de règles
|
|
import rawRules from './load-rules'
|
|
import R from 'ramda'
|
|
import possibleVariableTypes from './possibleVariableTypes.yaml'
|
|
import marked from './marked'
|
|
|
|
// console.log('rawRules', rawRules.map(({espace, nom}) => espace + nom))
|
|
/***********************************
|
|
Méthodes agissant sur une règle */
|
|
|
|
// Enrichissement de la règle avec des informations évidentes pour un lecteur humain
|
|
export let enrichRule = rule => {
|
|
let
|
|
type = possibleVariableTypes.find(t => R.has(t, rule)),
|
|
name = rule['nom'],
|
|
ns = rule['espace'],
|
|
dottedName = ns ? [
|
|
ns,
|
|
name
|
|
].join(' . ') : name,
|
|
subquestionMarkdown = rule['sous-question'],
|
|
subquestion = subquestionMarkdown && marked(subquestionMarkdown)
|
|
|
|
return {...rule, type, name, ns, dottedName, subquestion}
|
|
}
|
|
|
|
export let hasKnownRuleType = rule => rule && enrichRule(rule).type
|
|
|
|
export let
|
|
splitName = R.split(' . '),
|
|
joinName = R.join(' . ')
|
|
|
|
export let parentName = R.pipe(
|
|
splitName,
|
|
R.dropLast(1),
|
|
joinName
|
|
)
|
|
export let nameLeaf = R.pipe(
|
|
splitName,
|
|
R.last
|
|
)
|
|
|
|
export let encodeRuleName = name => name.replace(/\s/g, '-')
|
|
export let decodeRuleName = name => name.replace(/\-/g, ' ')
|
|
|
|
/* Les variables peuvent être exprimées dans la formule d'une règle relativement à son propre espace de nom, pour une plus grande lisibilité. Cette fonction résoud cette ambiguité.
|
|
*/
|
|
|
|
export let disambiguateRuleReference = (allRules, {ns, name}, partialName) => {
|
|
let
|
|
fragments = ns ? ns.split(' . ') : [], // ex. [CDD . événements . rupture]
|
|
pathPossibilities = // -> [ [CDD . événements . rupture], [CDD . événements], [CDD] ]
|
|
R.range(0, fragments.length + 1)
|
|
.map(nbEl => R.take(nbEl)(fragments))
|
|
.reverse(),
|
|
found = R.reduce((res, path) =>
|
|
R.when(
|
|
R.is(Object), R.reduced
|
|
)(findRuleByDottedName(allRules, [...path, partialName].join(' . ')))
|
|
, null, pathPossibilities)
|
|
|
|
return found && found.dottedName || do {throw `OUUUUPS la référence '${partialName}' dans la règle '${name}' est introuvable dans la base`}
|
|
}
|
|
|
|
// On enrichit la base de règles avec des propriétés dérivées de celles du YAML
|
|
export let rules = rawRules.map(enrichRule)
|
|
|
|
|
|
/****************************************
|
|
Méthodes de recherche d'une règle */
|
|
|
|
export let findRuleByName = (allRules, search) =>
|
|
allRules
|
|
.find( ({name}) =>
|
|
name.toLowerCase() === search.toLowerCase()
|
|
)
|
|
|
|
export let searchRules = searchInput =>
|
|
rules
|
|
.filter( rule =>
|
|
rule && hasKnownRuleType(rule) &&
|
|
JSON.stringify(rule).toLowerCase().indexOf(searchInput) > -1)
|
|
.map(enrichRule)
|
|
|
|
export let findRuleByDottedName = (allRules, dottedName) => {
|
|
let found = dottedName && allRules.find(rule => rule.dottedName.toLowerCase() == dottedName.toLowerCase()),
|
|
result = dottedName && dottedName.startsWith("sys .") ?
|
|
found || {dottedName: dottedName, nodeValue: null} :
|
|
found
|
|
|
|
return result
|
|
}
|
|
|
|
/*********************************
|
|
Autres */
|
|
|
|
let collectNodeMissingVariables = (root, source=root, results=[]) => {
|
|
if (
|
|
source.nodeValue != null ||
|
|
source.shortCircuit && source.shortCircuit(root)
|
|
) {
|
|
// console.log('nodev or shortcircuit root, source', root, source)
|
|
return []
|
|
}
|
|
|
|
if (source['missingVariables']) {
|
|
// console.log('root, source', root, source)
|
|
results.push(source['missingVariables'])
|
|
}
|
|
|
|
for (var prop in source) {
|
|
if (R.is(Object)(source[prop])) {
|
|
collectNodeMissingVariables(root, source[prop], results)
|
|
}
|
|
}
|
|
return results
|
|
}
|
|
|
|
// On peut travailler sur une somme, les objectifs sont alors les variables de cette somme.
|
|
// Ou sur une variable unique ayant une formule, elle est elle-même le seul objectif
|
|
export let getObjectives = analysedSituation => {
|
|
let formuleType = R.path(["formule", "explanation", "name"])(
|
|
analysedSituation
|
|
)
|
|
let result = formuleType == "somme"
|
|
? R.pluck(
|
|
"explanation",
|
|
R.path(["formule", "explanation", "explanation"])(analysedSituation)
|
|
)
|
|
: formuleType ? [analysedSituation] : null
|
|
|
|
return result ? R.reject(R.isNil)(result) : null;
|
|
}
|
|
|
|
export let collectMissingVariables = (groupMethod='groupByMissingVariable') => analysedSituation =>
|
|
R.pipe(
|
|
getObjectives,
|
|
R.chain( v =>
|
|
R.pipe(
|
|
collectNodeMissingVariables,
|
|
R.flatten,
|
|
R.map(mv => [v.dottedName, mv])
|
|
)(v)
|
|
),
|
|
//groupBy missing variable but remove mv from value, it's now in the key
|
|
R.groupBy(groupMethod == 'groupByMissingVariable' ? R.last : R.head),
|
|
R.map(R.map(groupMethod == 'groupByMissingVariable' ? R.head : R.last))
|
|
// below is a hand implementation of above... function composition can be nice sometimes :')
|
|
// R.reduce( (memo, [mv, dependencyOf]) => ({...memo, [mv]: [...(memo[mv] || []), dependencyOf] }), {})
|
|
)(analysedSituation)
|
|
|
|
let isVariant = R.path(['formule', 'une possibilité'])
|
|
|
|
export let findVariantsAndRecords = (allRules, names) => {
|
|
let tag = name => {
|
|
let parent = parentName(name),
|
|
gramps = parentName(parent),
|
|
findV = name => isVariant(findRuleByDottedName(allRules,name))
|
|
|
|
return findV(gramps) ? {type: "variantGroups", [gramps]:[name]}
|
|
: findV(parent) ? {type: "variantGroups", [parent]:[name]}
|
|
: {type: "recordGroups", [parent]:[name]}
|
|
}
|
|
|
|
let classify = R.map(tag),
|
|
groupByType = R.groupBy(R.prop("type")),
|
|
stripTypes = R.map(R.map(R.omit("type"))),
|
|
mergeLists = R.map(R.reduce(R.mergeWith(R.concat),{}))
|
|
|
|
return R.pipe(classify,groupByType,stripTypes,mergeLists)(names)
|
|
}
|
|
|
|
export let findVariantsAndRecords2 =
|
|
(allRules, {variantGroups, recordGroups}, dottedName, childDottedName) => {
|
|
let child = findRuleByDottedName(allRules, dottedName),
|
|
parentDottedName = parentName(dottedName),
|
|
parent = findRuleByDottedName(allRules, parentDottedName)
|
|
if (isVariant(parent)) {
|
|
let grandParentDottedName = parentName(parentDottedName),
|
|
grandParent = findRuleByDottedName(allRules, grandParentDottedName)
|
|
if (isVariant(grandParent))
|
|
return findVariantsAndRecords2(allRules, {variantGroups, recordGroups}, parentDottedName, childDottedName || dottedName)
|
|
else
|
|
return {
|
|
variantGroups: R.mergeWith(R.concat, variantGroups, {[parentDottedName]: [childDottedName || dottedName]}),
|
|
recordGroups
|
|
}
|
|
} else
|
|
return {
|
|
variantGroups,
|
|
recordGroups: R.mergeWith(R.concat, recordGroups, {[parentDottedName]: [dottedName]})
|
|
}
|
|
|
|
}
|