2017-01-26 10:27:24 +00:00
|
|
|
// Séparation artificielle, temporaire, entre ces deux types de règles
|
|
|
|
import rawRules from './load-rules'
|
2018-01-08 15:07:26 +00:00
|
|
|
import {
|
|
|
|
has,
|
|
|
|
pipe,
|
|
|
|
toPairs,
|
|
|
|
map,
|
|
|
|
fromPairs,
|
|
|
|
split,
|
|
|
|
join,
|
|
|
|
dropLast,
|
|
|
|
take,
|
|
|
|
reduce,
|
|
|
|
when,
|
|
|
|
is,
|
|
|
|
props,
|
|
|
|
identity,
|
|
|
|
path,
|
|
|
|
reject,
|
|
|
|
reduced,
|
|
|
|
range,
|
|
|
|
last
|
|
|
|
} from 'ramda'
|
2017-01-23 18:06:46 +00:00
|
|
|
import possibleVariableTypes from './possibleVariableTypes.yaml'
|
2017-03-28 16:52:01 +00:00
|
|
|
import marked from './marked'
|
2018-01-03 15:54:19 +00:00
|
|
|
import { capitalise0 } from '../utils'
|
2017-11-24 17:55:15 +00:00
|
|
|
import formValueTypes from 'Components/conversation/formValueTypes'
|
2017-01-26 10:27:24 +00:00
|
|
|
|
2017-09-18 07:29:26 +00:00
|
|
|
// TODO - should be in UI, not engine
|
2017-10-02 19:29:07 +00:00
|
|
|
import taux_versement_transport from '../../règles/rémunération-travail/cotisations/ok/liste-taux.json'
|
2017-09-18 07:29:26 +00:00
|
|
|
|
2018-01-03 15:54:19 +00:00
|
|
|
// console.log('rawRules', rawRules.map(({espace, nom}) => espace + nom))
|
2017-01-26 10:27:24 +00:00
|
|
|
/***********************************
|
|
|
|
Méthodes agissant sur une règle */
|
|
|
|
|
2017-02-10 14:12:00 +00:00
|
|
|
// Enrichissement de la règle avec des informations évidentes pour un lecteur humain
|
2017-09-18 07:29:26 +00:00
|
|
|
export let enrichRule = (rule, sharedData = {}) => {
|
2018-01-08 15:07:26 +00:00
|
|
|
let type = possibleVariableTypes.find(t => has(t, rule) || rule.type === t),
|
2017-04-24 18:03:38 +00:00
|
|
|
name = rule['nom'],
|
2017-11-13 14:06:42 +00:00
|
|
|
title = capitalise0(rule['titre'] || name),
|
2017-04-24 18:03:38 +00:00
|
|
|
ns = rule['espace'],
|
2017-09-18 07:29:26 +00:00
|
|
|
data = rule['données'] ? sharedData[rule['données']] : null,
|
2018-01-03 15:54:19 +00:00
|
|
|
dottedName = ns ? [ns, name].join(' . ') : name,
|
2017-02-16 14:08:50 +00:00
|
|
|
subquestionMarkdown = rule['sous-question'],
|
2017-11-20 16:25:08 +00:00
|
|
|
subquestion = subquestionMarkdown && marked(subquestionMarkdown),
|
|
|
|
defaultValue = rule['par défaut']
|
2017-01-26 10:27:24 +00:00
|
|
|
|
2018-01-03 15:54:19 +00:00
|
|
|
return {
|
|
|
|
...rule,
|
|
|
|
type,
|
|
|
|
name,
|
|
|
|
title,
|
|
|
|
ns,
|
|
|
|
data,
|
|
|
|
dottedName,
|
|
|
|
subquestion,
|
|
|
|
defaultValue
|
|
|
|
}
|
2017-01-26 10:27:24 +00:00
|
|
|
}
|
|
|
|
|
2018-01-03 15:54:19 +00:00
|
|
|
export let disambiguateExampleSituation = (rules, rule) =>
|
2018-01-08 15:07:26 +00:00
|
|
|
pipe(
|
|
|
|
toPairs,
|
|
|
|
map(([k, v]) => [disambiguateRuleReference(rules, rule, k), v]),
|
|
|
|
fromPairs
|
2018-01-03 15:54:19 +00:00
|
|
|
)
|
2017-12-08 14:12:41 +00:00
|
|
|
|
2017-01-26 10:27:24 +00:00
|
|
|
export let hasKnownRuleType = rule => rule && enrichRule(rule).type
|
|
|
|
|
2018-01-08 15:07:26 +00:00
|
|
|
export let splitName = split(' . '),
|
|
|
|
joinName = join(' . ')
|
2017-01-26 10:27:24 +00:00
|
|
|
|
2018-01-08 15:07:26 +00:00
|
|
|
export let parentName = pipe(splitName, dropLast(1), joinName)
|
|
|
|
export let nameLeaf = pipe(splitName, last)
|
2017-01-26 10:27:24 +00:00
|
|
|
|
2017-05-09 14:33:35 +00:00
|
|
|
export let encodeRuleName = name => name.replace(/\s/g, '-')
|
2018-01-08 15:07:26 +00:00
|
|
|
export let decodeRuleName = name => name.replace(/-/g, ' ')
|
2017-05-09 14:33:35 +00:00
|
|
|
|
2017-04-24 18:03:38 +00:00
|
|
|
/* 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é.
|
2017-03-16 18:30:30 +00:00
|
|
|
*/
|
2017-06-27 17:53:37 +00:00
|
|
|
|
2018-01-03 15:54:19 +00:00
|
|
|
export let disambiguateRuleReference = (
|
|
|
|
allRules,
|
|
|
|
{ ns, name },
|
|
|
|
partialName
|
|
|
|
) => {
|
|
|
|
let fragments = ns ? ns.split(' . ') : [], // ex. [CDD . événements . rupture]
|
2018-01-08 15:07:26 +00:00
|
|
|
pathPossibilities = range(0, fragments.length + 1) // -> [ [CDD . événements . rupture], [CDD . événements], [CDD] ]
|
|
|
|
.map(nbEl => take(nbEl)(fragments))
|
2017-04-24 18:03:38 +00:00
|
|
|
.reverse(),
|
2018-01-08 15:07:26 +00:00
|
|
|
found = reduce(
|
2018-01-03 15:54:19 +00:00
|
|
|
(res, path) =>
|
2018-01-08 15:07:26 +00:00
|
|
|
when(is(Object), reduced)(
|
2018-01-03 15:54:19 +00:00
|
|
|
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`
|
|
|
|
}
|
|
|
|
)
|
2017-03-16 18:30:30 +00:00
|
|
|
}
|
|
|
|
|
2018-01-08 15:07:26 +00:00
|
|
|
export let collectDefaults = pipe(
|
|
|
|
map(props(['dottedName', 'defaultValue'])),
|
|
|
|
reject(([, v]) => v === undefined),
|
|
|
|
fromPairs
|
2017-11-22 09:57:07 +00:00
|
|
|
)
|
2017-11-22 09:17:22 +00:00
|
|
|
|
2017-01-26 10:27:24 +00:00
|
|
|
// On enrichit la base de règles avec des propriétés dérivées de celles du YAML
|
2018-01-03 15:54:19 +00:00
|
|
|
export let rules = rawRules.map(rule =>
|
|
|
|
enrichRule(rule, { taux_versement_transport })
|
|
|
|
)
|
2017-01-26 10:27:24 +00:00
|
|
|
|
|
|
|
/****************************************
|
|
|
|
Méthodes de recherche d'une règle */
|
2016-12-07 18:08:10 +00:00
|
|
|
|
2017-06-27 17:53:37 +00:00
|
|
|
export let findRuleByName = (allRules, search) =>
|
2018-01-03 15:54:19 +00:00
|
|
|
allRules.find(({ name }) => name === search)
|
2016-12-07 18:08:10 +00:00
|
|
|
|
|
|
|
export let searchRules = searchInput =>
|
|
|
|
rules
|
2018-01-03 15:54:19 +00:00
|
|
|
.filter(
|
|
|
|
rule =>
|
|
|
|
rule &&
|
|
|
|
hasKnownRuleType(rule) &&
|
|
|
|
JSON.stringify(rule)
|
|
|
|
.toLowerCase()
|
|
|
|
.indexOf(searchInput) > -1
|
|
|
|
)
|
2017-01-24 15:22:40 +00:00
|
|
|
.map(enrichRule)
|
2016-12-07 18:08:10 +00:00
|
|
|
|
2017-11-23 10:25:24 +00:00
|
|
|
export let findRuleByDottedName = (allRules, dottedName) => {
|
2017-11-27 12:57:11 +00:00
|
|
|
return allRules.find(rule => rule.dottedName == dottedName)
|
2017-11-23 10:25:24 +00:00
|
|
|
}
|
2017-03-01 19:27:35 +00:00
|
|
|
|
|
|
|
/*********************************
|
2018-01-03 15:54:19 +00:00
|
|
|
Autres */
|
2017-03-01 19:27:35 +00:00
|
|
|
|
2018-01-08 15:07:26 +00:00
|
|
|
let isVariant = path(['formule', 'une possibilité'])
|
2017-11-24 17:55:15 +00:00
|
|
|
|
|
|
|
export let formatInputs = (flatRules, formValueSelector) => state => name => {
|
|
|
|
// Our situationGate retrieves data from the "conversation" form
|
|
|
|
// The search below is to apply input conversions such as replacing "," with "."
|
|
|
|
if (name.startsWith('sys.')) return null
|
|
|
|
|
|
|
|
let rule = findRuleByDottedName(flatRules, name),
|
|
|
|
format = rule ? formValueTypes[rule.format] : null,
|
2018-01-08 15:07:26 +00:00
|
|
|
pre = format && format.validator.pre ? format.validator.pre : identity,
|
2017-11-24 17:55:15 +00:00
|
|
|
value = formValueSelector('conversation')(state, name)
|
|
|
|
|
|
|
|
return value && pre(value)
|
|
|
|
}
|