mon-entreprise/publicodes
Johan Girod 0d50e03777 Ajoute la possibilité d'importer publicodes en ESModule natif
Pour cela, la solution la plus simple est de faire un wrapper autour de l'api exposée.
Il ne faudra pas oublier de le mettre à jour.

[A lire pour en savoir plus](https://redfin.engineering/node-modules-at-war-why-commonjs-and-es-modules-cant-get-along-9617135eeca1)

fix #1239
2020-12-11 13:39:42 +01:00
..
docs Apply suggestions from code review 2020-12-10 17:16:43 +01:00
esm Ajoute la possibilité d'importer publicodes en ESModule natif 2020-12-11 13:39:42 +01:00
source 🎨 Améliore l'affichage des noms des règles imbriquées 2020-12-10 16:54:21 +01:00
test ⚙️ améliore la résolution des noms dans les règles 2020-12-10 16:31:09 +01:00
README.md Apply suggestions from code review 2020-12-10 17:16:43 +01:00
package.json Ajoute la possibilité d'importer publicodes en ESModule natif 2020-12-11 13:39:42 +01:00
tsconfig.json
webpack.config.js Prettier s'occupe aussi des scripts 2020-12-10 12:23:16 +01:00
webpack.test.js Prettier s'occupe aussi des scripts 2020-12-10 12:23:16 +01:00

README.md

Publicode est un langage déclaratif pour encoder les algorithmes d'intérêt public. Il permet de réaliser des calculs généraux tout en fournissant une explication permettant de comprendre et de documenter ces calculs.

Publicode est adapté pour modéliser des domaines métiers complexes pouvant être décomposés en règles élémentaires simples (comme la législation socio-fiscale, un bilan carbone, un estimateur de rendement locatif, etc.).

Il permet de générer facilement des simulateurs web interactifs où l'on peut affiner progressivement le résultat affiché, et d'exposer une documentation du calcul explorable.

Projets phares

  • mon-entreprise.fr utilise publicodes pour spécifier l'ensemble des calculs relatifs à la législation socio-fiscale en France. Le site permet entre autre de simuler une fiche de paie complète, de calculer les cotisations sociales pour un indépendant ou encore connaître le montant du chômage partiel.
  • futur.eco utilise publicodes pour calculer les bilans carbone d'un grand nombre d'activités, plats, transports ou biens.
  • Nos Gestes Climat utilise publicodes pour proposer un calculateur d'empreinte climat personnel de référence complètement ouvert

Principes de base

La syntaxe de Publicode est basée sur le langage Yaml.

Un fichier Publicode contient une liste de règles identifiées par leur nom et possédant une valeur :

prix d'un repas: 10 

Une formule de calcul peut faire référence à d'autres règles. Dans l'exemple suivant la règle prix total aura pour valeur 50 (= 5 * 10)

prix d'un repas: 10 
prix total: 5 * prix d'un repas

Il s'agit d'un langage déclaratif : comme dans une formule d'un tableur le prix total sera recalculé automatiquement si le prix d'un repas change. L'ordre de définition des règles n'a pas d'importance.

Unités

Pour fiabiliser les calculs et faciliter leur compréhension, on peut préciser l'unité des valeurs littérales :

prix d'un repas: 10 €/repas
nombre de repas: 5 repas

prix total: nombre de repas * prix d'un repas

Le calcul est inchangé mais on a indiqué que le "prix d'un repas" s'exprime en €/repas et que le "nombre de repas" est un nombre de repas. L'unité du prix total est inférée automatiquement comme étant en . (€/repas * repas = )

Ce système d'unité permet de typer les formules de calcul et de rejeter automatiquement des formules incohérentes :

prix d'un repas: 10 €/repas
nombre de repas: 5 repas
frais de réservation: 1 €/repas

prix total: nombre de repas * prix d'un repas + frais de réservation
# Erreur:
# La formule de "prix total" est invalide.

Dans l'exemple ci-dessus Publicode détecte une erreur car les termes de l'addition ont des unités incompatibles : d'un côté on a des et de l'autre des €/repas.

Cette incohérence d'unité témoigne d'une erreur de logique. Ici une manière de corriger l'erreur peut être de factoriser la variable "nombre de repas" dans la formule du "prix total".

prix total: nombre de repas * (prix d'un repas + frais de réservation)

Attention : Il ne faut pas insérer d'espace autour de la barre oblique dans les unités, l'unité € / mois doit être notée €/mois

Publicode convertit automatiquement les unités si besoin.

salaire: 1500 €/mois
prime faible salaire applicable: salaire < 20 k€/an

NB : On peut forcer la conversion des unités via le mécanisme unité

Types de base disponibles pour la conversion :

  • jour / mois / an
  • / k€

Mécanismes

Il existe une autre manière d'écrire des formules de calcul : les mécanismes. Au lieu de définir la formule sur une ligne, celle-ci prends la forme d'un objet sur plusieurs lignes.

Par exemple, la formule suivante :

prix total: 5 repas * prix d'un repas

Peut également s'écrire en utilisant le mécanisme produit :

prix total:
    produit:
        assiette: prix d'un repas
        facteur: 5 repas

Un des avantages de cette écriture est que la syntaxe hiérarchique de Yaml permet d'imbriquer les mécanismes :

prix TTC:
    somme:
        - prix d'un repas
        - produit:
              assiette: prix d'un repas
              taux: TVA

Mécanismes chaînés

Certains mécanismes peuvent apparaître au même niveau d'indentation. Dans ce cas, le moteur appliquera les transformations dans un ordre préetabli.

remboursement repas:
    valeur: nombre de repas * remboursement forfaitaire
    plafond: 500 €/an
    unité: €/an
    arrondi: oui

Pour en savoir plus sur les mécanismes

Pages d'explications

L'explication des règles est un des objectifs fondamentaux de Publicodes.

Chaque règle se voit générer automatiquement une page explicatives correspondante dans le front-end, contenant une information facilement digeste mise en regard des calculs eux-mêmes.

Plusieurs propriétés sont reprises dans ces pages d'explications :

  • le titre, qui s'affiche en haut de la page. Par défaut on utilise le nom de la règle, mais la propriété titre permet de choisir un titre plus approprié ;
  • la description qui peut être rédigée en Markdown et est généralement affichée comme paragraphe d'introduction sur la page. On utilise le caractère | pour indiquer au parseur Yaml que la description est sur plusieurs lignes ;
  • les références externes (documentation utile) affichées en bas de page et qui sont constituées d'une liste de liens avec une description.
ticket resto:
    titre: Prise en charge des titres-restaurants
    formule: 4 €/repas
    description: |
        L'employeur peut remettre des titres restaurants sous plusieurs formats:
        - ticket *papier*
        - carte à *puce*
        - appli *mobile*        
    références:
        Fiche service public: https://www.service-public.fr/professionnels-entreprises/vosdroits/F21059
        Fiche Urssaf: https://www.urssaf.fr/portail/home/taux-et-baremes/frais-professionnels/les-titres-restaurant.html

Conditions booléennes

Publicode supporte des opérateurs booléens basiques.

âge: 17 ans
mineur émancipé: non
nationalité française: oui

droit de vote:
    toutes ces conditions:
        - nationalité française
        - une de ces conditions:
              - âge >= 18 ans
              - mineur émancipé

Il est possible de faire des branchements conditionnels via le mécanisme variations

Applicabilité

On peut définir des conditions d'applicabilité pour des valeurs :

date de début: 12/02/2020

ancienneté en fin d'année:
    durée:
        depuis: date de début
        jusqu'à: 31/12/2020

prime de vacances:
    applicable si: ancienneté en fin d'année > 1 an
    valeur: 200

Ici si l'ancienneté est inférieure à un an la prime de vacances ne sera pas applicable. Les variables non applicables sont égale à non. Elles sont ignorées au niveau des mécanismes numériques (par exemple le mécanisme somme comptera une prime non applicable comme valant zéro, voir la page spécifique aux mécanismes).

La syntaxe suivante est également valable:

assimilé salarié:
    valeur: oui
    rend non applicable: convention collective

Espaces de noms

Les espaces de noms sont utiles pour organiser un grand nombre de règles. On utilise le . pour exprimer la hiérarchie des noms.

prime de vacances:
    formule: taux * 1000 €

prime de vacances . taux:
    formule: 6%

Ici prime de vacances est à la fois une règle et un espace de noms. La variable taux est définie dans cet espace de noms et c'est elle qui est référencée dans la formule de calcul de la règle prime de vacances.

La règle prime de vacances est elle-même définie dans l'espace de nom racine.

On pourrait avoir une autre variable taux dans un espace de nom différent, sans que cela entre en conflit:

# Ceci n'entre pas dans le calcul de `prime de vacances` définie plus haut
autre prime . taux:
    formule: 19%

On dit que la formule de la règle prime de vacances fait référence à la règle prime de vacances . taux via le nom raccourci taux.

Pour faire référence à une règle hors de son espace de nom, on peut écrire le nom complet de cette règle:

prime de vacances v2:
    formule: autre prime . taux * 1000 €

Dans le cas d'espaces de noms imbriqués (à plus qu'un étage), le nom inscrit dans une règle donnée est recherché de plus en plus haut dans la hiérarchie des espaces de nom jusqu'à la racine.

contrat salarié . rémunération . primes . prime de vacances:
    formule: taux générique * 1000 €

contrat salarié . rémunération . taux générique:
    formule: 10%

Ici contrat salarié . rémunération . primes . prime de vacances va faire référence à contrat salarié . rémunération . taux générique trouvé deux espaces de noms plus haut, et va donc valoir 100 €.

Désactivation de branche

Il est possible de désactiver l'ensemble des règles définies dans un espace de nom.

Toutes les règles possèdent une dépendance implicite à leur parent. Si ce dernier est égal à non alors leur valeur est également non.

CDD: non
CDD . indemnité de précarité: 10% * 1500€/mois * 6 mois

indemnités:
    somme:
        - 100 
        - CDD . indemnité de précarité # non

Remplacement

Certaines règles ne s'appliquent parfois que dans quelques situations particulières et modifier la définition des règles générales pour prendre en compte ces particularités pose des problème de maintenabilité de la base de règle.

Publicode dispose d'un mécanisme de remplacement qui permet d'amender n'importe quelle règle existante sans avoir besoin de la modifier :

frais de repas: 5 €/repas

convention hôtels cafés restaurants: oui
convention hôtels cafés restaurants . frais de repas:
    remplace: frais de repas
    valeur: 6 €/repas

montant repas mensuels: 20 repas * frais de repas

On peut également choisir de remplacer dans un contexte donné:

temps de préparation: 20 min
temps de cuisson: 20 min

robot de cuisine:
    remplace:
        - règle: temps de préparation
          sauf dans: temps original
          par: 10 min
    valeur: oui

temps original:
    formule: a + b

temps modifié:
    formule: a + b

En savoir plus sur les remplacements

Définition de règle imbriquée

Si le mécanisme de remplacement permet de faire des substitutions de règles complètes, il est parfois utile de ne modifier qu'un seul paramètre d'une règle existante, par exemple modifier le facteur d'une multiplication tout en conservant le reste de sa définition inchangée.

Une première manière de faire consiste à extraire le paramètre en question dans une règle indépendante, le rendant ainsi accessible et modifiable depuis l'extérieur :

prime:
    formule:
        multiplication:
            assiette: 1000
            taux: taux

prime . taux:
    formule: 5%

super-prime:
    remplace: prime . taux
    formule: 10%

Ce code fonctionne mais il nous oblige a créer une règle prime . taux qui n'est pas pertinente en tant qu'entité autonome (avec sa propre page de documentation, etc.), uniquement pour pouvoir la modifier avec un remplace. On a aussi introduit une indirection dans la définition de la prime en remplaçant une ligne explicite taux: 5% par une référence vers une règle tierce taux: taux, qui est loin d'être aussi claire.

Pour ce cas d'usage il est possible de définir une règle imbriquée. On garde la définition de la prime inchangée et on annote la valeur à laquelle on veut accéder depuis l'extérieur avec le mot clé nom :

prime:
    formule:
        multiplication:
            assiette: 1000
            taux:
                nom: taux
                valeur: 5%

super-prime:
    remplace: prime . taux
    formule: 10%

Évaluation

Lors de l'évaluation, les variables dont les valeurs ne sont pas renseignées sont remontées afin que ces dernières puissent être complétées par l'utilisateur (dans le cas d'un simulateur par exemple).

Il est possible de donner une valeur par défaut. Les variables manquantes seront quand même remontée, et le moteur utilisera la valeur par défaut pour le calcul.

durée:
    par défaut: 2 mois

salaire brut:
    par défaut: 1500 €/mois

indemnité de CDD: 10 % * salaire brut * durée