import { any, equals, is, map, max, mergeWith, min, path, pluck, reduce } from 'ramda' import React from 'react' import Allègement from './components/mecanisms/Allègement' import { Mecanism } from './components/mecanisms/common' import InversionNumérique from './components/mecanisms/InversionNumérique' import Product from './components/mecanisms/Product' import Recalcul from './components/mecanisms/Recalcul' import Somme from './components/mecanisms/Somme' import { RuleLinkWithContext } from './components/RuleLink' import { typeWarning } from './error' import { collectNodeMissing, defaultNode, evaluateArray, evaluateNode, evaluateObject, makeJsx, mergeAllMissing, parseObject } from './evaluation' import { decompose } from './mecanisms/utils' import variations from './mecanisms/variations' import { convertNodeToUnit } from './nodeUnits' import uniroot from './uniroot' import { areUnitConvertible, convertUnit, inferUnit, parseUnit, serializeUnit } from './units' export const mecanismOneOf = (recurse, k, v) => { if (!is(Array, v)) throw new Error('should be array') const explanation = map(recurse, v) const jsx = ({ nodeValue, explanation, unit }) => ( ) const evaluate = (cache, situation, parsedRules, node) => { const evaluateOne = child => evaluateNode(cache, situation, parsedRules, child) const explanation = map(evaluateOne, node.explanation) const anyTrue = explanation.find(e => e.nodeValue === true) const anyNull = explanation.find(e => e.nodeValue === null) const { nodeValue, missingVariables } = anyTrue ?? anyNull ?? { nodeValue: false, // Unlike most other array merges of missing variables this is a "flat" merge // because "one of these conditions" tend to be several tests of the same variable // (e.g. contract type is one of x, y, z) missingVariables: reduce( mergeWith(max), {}, map(collectNodeMissing, explanation) ) } return { ...node, nodeValue, explanation, missingVariables } } return { evaluate, jsx, explanation, category: 'mecanism', name: 'une de ces conditions', type: 'boolean' } } export const mecanismAllOf = (recurse, k, v) => { if (!is(Array, v)) throw new Error('should be array') const explanation = map(recurse, v) const jsx = ({ nodeValue, explanation, unit }) => ( ) const evaluate = (cache, situation, parsedRules, node) => { const evaluateOne = child => evaluateNode(cache, situation, parsedRules, child), explanation = map(evaluateOne, node.explanation) const anyFalse = explanation.find(e => e.nodeValue === false) // court-circuit const { nodeValue, missingVariables } = anyFalse ?? { nodeValue: explanation.some(e => e.nodeValue === null) ? null : true, missingVariables: mergeAllMissing(explanation) } return { ...node, nodeValue, explanation, missingVariables } } return { evaluate: evaluate, jsx, explanation, category: 'mecanism', name: 'toutes ces conditions', type: 'boolean' } } const evaluateInversion = (oldCache, situation, parsedRules, node) => { // TODO : take applicability into account here let inversedWith = node.explanation.inversionCandidates.find( n => situation[n.dottedName] != undefined ) if (!inversedWith) { return { ...node, missingVariables: { ...Object.fromEntries( node.explanation.inversionCandidates.map(n => [n.dottedName, 1]) ), [node.explanation.ruleToInverse]: 1 }, nodeValue: null } } inversedWith = evaluateNode(oldCache, situation, parsedRules, inversedWith) let inversionCache function resetInversionCache() { inversionCache = { _meta: { ...oldCache._meta } } return inversionCache } const evaluateWithValue = (n: number) => evaluateNode( resetInversionCache(), { ...situation, [inversedWith.dottedName]: undefined, [node.explanation.ruleToInverse]: { nodeValue: n, unit: parsedRules[node.explanation.ruleToInverse].unit } }, parsedRules, inversedWith ) // si fx renvoie null pour une valeur numérique standard, disons 2000, 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 const randomAttempt = evaluateWithValue(2000) const nodeValue = randomAttempt.nodeValue === null ? null : // cette fonction détermine l'inverse d'une fonction sans faire trop d'itérations uniroot( x => { const candidateNode = evaluateWithValue(x) return ( candidateNode.nodeValue - // TODO: convertNodeToUnit migth return null or false (convertNodeToUnit(candidateNode.unit, inversedWith) .nodeValue as number) ) }, node.explanation.negativeValuesAllowed ? -1000000 : 0, 100000000, 0.1, 10 ) if (nodeValue === undefined) { oldCache._meta.inversionFail = true } else { // For performance reason, we transfer the inversion cache Object.entries(inversionCache).forEach(([k, value]) => { oldCache[k] = value }) } return { ...node, nodeValue: nodeValue ?? null, explanation: { ...node.explanation, inversionFail: nodeValue === undefined, inversedWith }, missingVariables: randomAttempt.missingVariables } } export const mecanismInversion = dottedName => (recurse, k, v) => { if (!v.avec) { throw new Error( "Une formule d'inversion doit préciser _avec_ quoi on peut inverser la variable" ) } return { evaluate: evaluateInversion, unit: v.unité && parseUnit(v.unité), explanation: { ruleToInverse: dottedName, inversionCandidates: v.avec.map(recurse), negativeValuesAllowed: v['valeurs négatives possibles'] === 'oui' }, jsx: InversionNumérique, category: 'mecanism', name: 'inversion numérique', type: 'numeric' } } export const mecanismRecalcul = dottedNameContext => (recurse, k, v) => { const evaluate = (cache, situation, parsedRules, node) => { if (cache._meta.inRecalcul) { return defaultNode(false) } const amendedSituation = Object.fromEntries( node.explanation.amendedSituation .map(([originRule, replacement]) => [ evaluateNode(cache, situation, parsedRules, originRule), evaluateNode(cache, situation, parsedRules, replacement) ]) .filter( ([originRule, replacement]) => originRule.nodeValue !== replacement.nodeValue || serializeUnit(originRule.unit) !== serializeUnit(replacement.unit) ) .map(([originRule, replacement]) => [ originRule.dottedName, replacement ]) ) // Optimisation : no need for recalcul if situation is the same const recalculCache = Object.keys(amendedSituation).length ? { _meta: { ...cache._meta, inRecalcul: true } } // Create an empty cache : cache const evaluatedNode = evaluateNode( recalculCache, { ...situation, ...amendedSituation }, parsedRules, node.explanation.recalcul ) return { ...node, nodeValue: evaluatedNode.nodeValue, ...(evaluatedNode.temporalValue && { temporalValue: evaluatedNode.temporalValue }), unit: evaluatedNode.unit, explanation: { recalcul: evaluatedNode, amendedSituation } } } const amendedSituation = Object.keys(v.avec).map(dottedName => [ recurse(dottedName), recurse(v.avec[dottedName]) ]) const defaultRuleToEvaluate = dottedNameContext const nodeToEvaluate = recurse(v.règle ?? defaultRuleToEvaluate) return { explanation: { recalcul: nodeToEvaluate, amendedSituation }, jsx: Recalcul, evaluate } } export const mecanismSum = (recurse, k, v) => { const explanation = v.map(recurse) const evaluate = evaluateArray( (x, y) => (x === false && y === false ? false : x + y), false ) return { evaluate, jsx: Somme, explanation, category: 'mecanism', name: 'somme', type: 'numeric', unit: inferUnit( '+', explanation.map(r => r.unit) ) } } export const mecanismReduction = (recurse, k, v) => { const objectShape = { assiette: false, abattement: defaultNode(0), plafond: defaultNode(Infinity), franchise: defaultNode(0) } const effect = ( { assiette, abattement, plafond, franchise, décote }, cache ) => { const assietteValue = assiette.nodeValue if (assietteValue == null) return { nodeValue: null } if (assiette.unit) { try { franchise = convertNodeToUnit(assiette.unit, franchise) plafond = convertNodeToUnit(assiette.unit, plafond) if (serializeUnit(abattement.unit) !== '%') { abattement = convertNodeToUnit(assiette.unit, abattement) } if (décote) { décote.plafond = convertNodeToUnit(assiette.unit, décote.plafond) décote.taux = convertNodeToUnit(parseUnit(''), décote.taux) } } catch (e) { typeWarning( cache._meta.contextRule, "Impossible de convertir les unités de l'allègement entre elles", e ) } } const montantFranchiséDécoté = franchise.nodeValue && assietteValue < franchise.nodeValue ? 0 : décote ? (function() { const plafondDécote = décote.plafond.nodeValue, taux = décote.taux.nodeValue return assietteValue > plafondDécote ? assietteValue : max(0, (1 + taux) * assietteValue - taux * plafondDécote) })() : assietteValue const nodeValue = abattement ? abattement.nodeValue == null ? montantFranchiséDécoté === 0 ? 0 : null : serializeUnit(abattement.unit) === '%' ? max( 0, montantFranchiséDécoté - min( plafond.nodeValue, (abattement.nodeValue / 100) * montantFranchiséDécoté ) ) : max( 0, montantFranchiséDécoté - min(plafond.nodeValue, abattement.nodeValue) ) : montantFranchiséDécoté return { nodeValue, unit: assiette.unit, explanation: { franchise, plafond, abattement } } } const base = parseObject(recurse, objectShape, v), explanation = v.décote ? { ...base, décote: map(recurse, v.décote) } : base, evaluate = evaluateObject(objectShape, effect) return { evaluate, jsx: Allègement, explanation, category: 'mecanism', name: 'allègement', type: 'numeric', unit: explanation?.assiette?.unit } } export const mecanismProduct = (recurse, k, v) => { if (v.composantes) { //mécanisme de composantes. Voir mécanismes.md/composantes return decompose(recurse, k, v) } if (v.variations) { return variations(recurse, k, v, true) } const objectShape = { assiette: false, taux: defaultNode(1), facteur: defaultNode(1), plafond: defaultNode(Infinity) } const effect = ({ assiette, taux, facteur, plafond }, cache) => { if (assiette.unit) { try { plafond = convertNodeToUnit(assiette.unit, plafond) } catch (e) { typeWarning( cache._meta.contextRule, "Impossible de convertir l'unité du plafond du produit dans celle de l'assiette", e ) } } const mult = (base, rate, facteur, plafond) => Math.min(base, plafond === false ? Infinity : plafond) * rate * facteur let nodeValue = [taux, assiette, facteur].some(n => n.nodeValue === false) ? false : [taux, assiette, facteur].some(n => n.nodeValue === 0) ? 0 : [taux, assiette, facteur].some(n => n.nodeValue === null) ? null : mult( assiette.nodeValue, taux.nodeValue, facteur.nodeValue, plafond.nodeValue ) let unit = inferUnit( '*', [assiette, taux, facteur].map(el => el.unit) ) if (areUnitConvertible(unit, assiette.unit)) { nodeValue = convertUnit(unit, assiette.unit, nodeValue) unit = assiette.unit } return { nodeValue, unit, explanation: { plafondActif: assiette.nodeValue > plafond.nodeValue } } } const explanation = parseObject(recurse, objectShape, v), evaluate = evaluateObject(objectShape, effect) return { evaluate, jsx: Product, explanation, category: 'mecanism', name: 'produit', type: 'numeric', unit: inferUnit( '*', [explanation.assiette, explanation.taux, explanation.facteur].map( el => el.unit ) ) } } export const mecanismMax = (recurse, k, v) => { const explanation = v.map(recurse) const max = (a, b) => { if (a === false) { return b } if (b === false) { return a } if (a === null || b === null) { return null } return Math.max(a, b) } const evaluate = evaluateArray(max, Number.NEGATIVE_INFINITY) const jsx = ({ nodeValue, explanation, unit }) => ( ) return { evaluate, jsx, explanation, type: 'numeric', category: 'mecanism', name: 'le maximum de', unit: explanation[0].unit } } export const mecanismMin = (recurse, k, v) => { const explanation = v.map(recurse) const evaluate = evaluateArray(min, Infinity) const jsx = ({ nodeValue, explanation, unit }) => ( ) return { evaluate, jsx, explanation, type: 'numeric', category: 'mecanism', name: 'le minimum de', unit: explanation[0].unit } } export const mecanismSynchronisation = (recurse, k, v) => { const evaluate = (cache, situation, parsedRules, node) => { const APIExplanation = evaluateNode( cache, situation, parsedRules, node.explanation.API ) const valuePath = v.chemin.split(' . ') const nodeValue = APIExplanation.nodeValue == null ? null : path(valuePath, APIExplanation.nodeValue) // If the API gave a non null value, then some of its props may be null (the API can be composed of multiple API, some failing). Then this prop will be set to the default value defined in the API's rule const safeNodeValue = nodeValue == null && APIExplanation.nodeValue != null ? path(valuePath, APIExplanation.explanation.defaultValue) : nodeValue const missingVariables = { ...APIExplanation.missingVariables, ...(APIExplanation.nodeValue === null ? { [APIExplanation.dottedName]: 1 } : {}) } const explanation = { ...v, API: APIExplanation } return { ...node, nodeValue: safeNodeValue, explanation, missingVariables } } return { explanation: { ...v, API: recurse(v.API) }, evaluate, jsx: function Synchronisation({ explanation }) { return (

Obtenu à partir de la saisie{' '}

) }, category: 'mecanism', name: 'synchronisation' } } export const mecanismOnePossibility = dottedName => (recurse, k, v) => ({ ...v, 'une possibilité': 'oui', evaluate: (cache, situation, parsedRules, node) => ({ ...node, missingVariables: { [dottedName]: 1 } }) })