12 KiB
Api
Publicode est distribué sous la forme d'une librairie Node.js permettant de compiler un jeu de règles publicodes et de l'évaluer dans une situation donnée.
Installation
Le paquet est disponble sur NPM :
$ npm install publicodes
# installation des peer dependancies
$ npm install react react-router-dom react-router-hash-link
Note : publicodes requiert l'installation de ses peerDependancy pour fonctionner. La raison est qu'en l'état actuel des choses, la documentation interactive (en react) n'a pas été extraite du coeur du moteur (calcul). Cela est temporaire et nous avons prévu d'y remedier dans une prochaine version, en créant deux paquets séparés : publicodes et publicodes/react
Utilisation
Introduction
La libraire exporte un objet Engine
qui permet d'instancier le moteur avec un
jeu de règles publicodes :
import Engine from 'publicodes'
// On définit une liste de règles publicodes
const rules = `
prix . carottes: 2€/kg
prix . champignons: 5€/kg
prix . avocat: 2€/avocat
dépenses primeur:
formule:
somme:
- prix . carottes * 1.5 kg
- prix . champignons * 500g
- prix . avocat * 3 avocat
`
// On initialise un moteur en lui donnant le publicode.
// Ce publicode va être parsé
const engine = new Engine(rules)
La variable engine
permet en ensuite de calculer la valeur d'une règle avec la
méthode evaluate
.
console.log(engine.evaluate('dépenses primeur'))
La valeur du nœud est disponible dans l'attribut nodeValue
, son
unité est disponible dans l'attribut unit
. Mais pour un formattage sans
effort, on préfèrera utiliser la fonction formatValue
import Engine, { formatValue } from 'publicodes'
const dépenses = engine.evaluate('dépenses primeur')
console.log(`j'ai dépensé ${formatValue(dépenses)} chez le primeur`)
La méthode setSituation
permet de forcer la valeur d'une liste de règle. Elle
utile pour pour présier les paramètres spécifiques à une simulation.
// Ici on change le prix des avocats
engine.setSituation({
'prix . avocat': '3€/avocat'
})
La valeur de dépenses primeur
se base maintenant sur un avocat à 3€ :
// On ré-évalue la règle dans la nouvelle situation
console.log(
`Nouveau prix ! ${formatValue(engine.evaluate('dépenses primeur'))}`
)
Évaluation d'expressions
La fonction evaluate
permet d'évaluer des expressions publicode complètes
// On va au marché une fois par semaine, amortissons la dépense sur 7 jours
engine.evaluate('dépenses primeur / 7 jours')
Conversion d'unité
Publicode permet de réaliser des conversions d'unités. Pour celà il faut
indiquer l'unité désirée comme paramètre à la méthode evaluate
:
// on va au marché une fois par semaine en moyenne, combien dépense-t-on par mois ?
engine.evaluate('dépenses primeurs / 7 jours', { unit: '€/mois' })
➡ en savoir plus sur les unités
Variables manquantes
Publicode calcule automatiquement les dépendances de chaque règle. Si une la
valeur d'une dépendance est manquante et ne permet pas de faire le calcul elle
apparaîtra dans l'attribut missingVariables
const engine = new Engine(`
x: y + 5
y:
`)
console.log(engine.evaluate('x').missingVariables)
Cette information est utile pour intégrer publicode à votre application.
Il est aussi possible d'utiliser des valeurs par défaut. Dans ce cas la règle
sera calculée avec la valeur par défaut de sa dépendance, mais cette dernière
apparaîtra tout de même dans les missingVariables
. Cette fonctionnalité est
utile pour réaliser des simulateurs où l'on veut proposer un résultat sans
attendre que l'utilisateur n'ait répondu à l'intégralité des questions tout en
utilisant la liste des variables manquantes pour déterminer les questions
restant à poser.
Les variables manquantes sont calculées lors de l'évaluation. Si une variable
apparaît dans la formule de calcul d'une règle elle ne sera rapportée que si
elle est effectivement nécessaire au calcul. Si elle est présente dans une
portion non active de l'évaluation (par exemple dans un bloc condition non
actif, ou la tranche d'un barème non actif) elle sera filtrée et n'apparaîtra
pas dans les missingVariables
Documentation intéractive
Publicodes génère également pour vous une documentation interactive, très facilement intégrable dans une app react. Pour cela, il vous suffit d'importer le composant dédié, et passer l'engine à afficher dans les props.
import { Documentation } from 'publicodes'
function MonApp() {
return (
<ReactRouter>
<Documentation engine={engine} documentationPath={'/documentation'} />
{/* Composants de l'app */}
</ReactRouter>
)
}
Vous pourrez ensuite faire un lien vers la documentation avec le composant
RuleLink
.
import { RuleLink } from 'publicodes'
function MesDépenses() {
return (
<p>
Accéder aux{' '}
<RuleLink
dottedName={'dépenses primeur'}
engine={engine}
documentationPath={'/documentation'}
/>
</p>
)
}
Note : ces composants n'ont pas vocation à rester dans la bibliothèque coeur, et seront prochainement disponible dans un paquet séparé, ce afin de ne pas avoir
react
dans les dépendances de publicodes.
Référence
new Engine(rules)
Crée un moteur d'évaluation avec les règles publicodes données en argument.
Arguments
rules
: les règles publicodes utilisée. Ces dernières peuvent-être sous la forme d'une chaine de caractèreyaml
publicodes, ou d'un object javascript correspondant.
Retourne Un moteur d'évaluation qui expose les fonctions suivantes :
- setSituation
- evaluate
- getParsedRules
method engine.setSituation(situation)
Permet de spécifier une situation en entrée. Toutes les prochaines évaluations seront effectuées en se basant sur ces valeurs plutôt que les valeurs présentes dans la base de règle.
C'est le point d'entrée principal pour adapter les calculs de règles générales à
une situation particulière. La situation est gardée en mémoire, et chaque appel
à setSituation
remplace la situation précédente. Le moteur
contient donc un état interne. Cela permet d'obtenir de meilleure performance,
avec une gestion plus fine du cache de calcul. En revanche, cela peut-être une
source de bug si l'état interne est modifié lors d'effet de bord non prévus.
Arguments
situation
: un objet javascript qui associe le nom complet d'une règle à sa valeur dans la situation. Cette valeur peut être de typenumber
, oustring
. Dans ce dernier cas, elle sera évaluée par le moteur. Cela permet de spécifier des nombre avec unité, des expressions, ou encore des références vers d'autres règles.
Retourne
L'objet engine (this
) sur lequel la fonction a été appelée, afin de pouvoir
utiliser une écriture chaînée (engine.setSituation(situation).evaluate()
)
method engine.evaluate(expression, [options])
Évalue l'expression dans le contexte du moteur (règle et situation).
Pour des raisons de performance, les résultats intermédiaires sont enregistrés dans un cache. Par conséquent, les prochains appels seront plus rapides.
Arguments
-
expression
: la formule à évaluer (typestring
). Cela peut-être une référence vers une règle, une expression arithmétique, tout ce que la grammaire publicode permet. -
options
: un objet de configuration pour l'évaluationunit
: spécifie l'unité dans laquelle le résultat doit être retourné. Si la valeur retournée par le calcul est un nombre, ce dernier sera converti dans l'unité demandée. Ainsievaluate('prix', {unit: '€'})
équivaut àevaluate('prix [€]')
. Une erreur est levée si l'unité n'est pas compatible avec la formule.useDefaultValues
(par défauttrue
): option pour forcer l'utilisation des valeurs par défaut des règles. Si sa valeur est àfalse
et qu'il manque des valeurs dans la situation pour que le calcul soit effectué, ces dernières seront remontée dans lesmissingsVariables
de l'objet retourné, et la valeur seranull
.
Retourne
Un objet javascript de type EvaluatedNode
contenant la valeur calculée.
Attention ! Il est déconseillé d'utiliser directement les valeurs présentes dans l'objet retourné, étant donné que leur forme va très probablement changer au cours des prochaines versions du moteur. Utilisez la fonction
formatNode(evaluationResult)
autant que possible pour afficher la valeur retournée.
missingVariables
: contient les valeur manquante lorsqueuseDefaultValues
est mis àfalse
.nodeValue
: la valeur calculéeisApplicable
: si l'expression évaluée est une référence à une règle, alors ce booléen indique si la règle est applicable ou non
function formatValue(evaluatedNode, [options])
Formate la valeur evaluée.
Arguments
-
evaluatedNode
: l'objet retourné lors de l'appel à la fonction d'évaluation du moteurevaluate(expression)
-
options
: configuration pour le formatagelanguage
: le langage utilisé pour le formatage (par défautfr
)precision
: le nombre de chiffre après la virgule pour le formatage des nombres (par défaut2
)displayedUnit
: l'unité à afficher pour le formatage des nombres. Outrepasse l'unité définie dans le calcul (on peut donc forcer l'unité affichée à une autre que celle retournée par le calcul, même si elle ne sont pas compatibles)
Retourne
La chaîne de caractère correspondant à la valeur bien formatée.
Composants react
Publicodes exporte des composants react permettant d'afficher une documentation explorable des calculs. Cette documentation est auto-générée en s'appuyant sur les données descriptives contenues dans les règles publicodes (description, références, titre, note, etc.) et en affichant pour chaque règle les étapes intermédiaires qui permettent d'aboutir au résultat affiché.
Voir un exemple sur mon-entreprise.fr
Composant react permettant d'afficher une documentation explorable d'une base de règles publicodes. Se base sur react-router pour créer une arborescence de pages correspondant aux espaces de noms existants dans les règles.
Voir le bac à sable pour voir le composant en action (il est affiché sur l'écran de droite).
Props
engine
: l'objet moteur dont on veut afficher les calculs.documentationPath
: (string
) le chemin de base sur lequel la documentation sera montée. Par exemple, si c'est/documentation
l'url de la règle 'rémunération . primes' sera/documentation/rémunération/primes
language
: le language dans lequel afficher la documentation (pour l'instant, seulfr
eten
sont supportés)
Note : les valeurs des règles
par défaut
ne sont pas utilisée dans la doc. Si l'on souhaite afficher la documentation avec les calculs utilisant les valeurs par défaut, il suffit d'ajouter la cléuseDefaultValues: true
dans lestate
de l'objetlocation
du navigateur. On peut également utiliser RuleLink (ci-dessous) qui s'en occupe pour nous.
Composant react permettant de faire un lien vers une page de la documentation. Par défaut, le texte affiché est le nom de la règle.
Props
engine
: l'objet moteur dont on veut afficher la règle.documentationPath
: (string
) le chemin de base sur lequel la documentation est montée. Doit correspondre à celui précisé pour le composant<Documentation />
dottedName
: le nom de la règle à afficherdisplayIcon
: affiche l'icône de la règle dans le lien (par défaut àfalse
)useDefaultValues
: utilise les valeurspar défaut
des règles (par défaut àfalse
)children
: N'importe quel noeud react. Par défaut, c'est le nom de la règle qui est utilisé.