8.9 KiB
Contribuer au cœur de YunoHost
Vous souhaitez ajouter une nouvelle fonctionnalité au cœur de YunoHost, mais ne savez pas comment procéder ? Ce guide parcourt les étapes du développement et du processus de contribution.
Si vous cherchez quelque chose à implémenter ou un bug à réparer, le bug tracker est ici !
Venez dire coucou sur le salon de dev !
Mettre en place un environnement de développement
-
Utilisez ynh-dev (voir le README) pour mettre en place un environnement de développement - en local sur une machine virtuelle, ou bien sur un VPS. Ceci installera une instance fonctionelle de YunoHost, en utilisant directement les dépôts git à l'aide de liens symboliques. De cette façon, il vous sera possible de modifier les fichiers, de tester les changements en temps réel, et de commiter et push/pull directement depuis cet environnement.
-
Implémentez et testez votre fonctionnalité. Suivant ce sur quoi vous voulez travailler :
- Cœur Python/ligne de commande : allez dans
/ynh-dev/yunohost/
- Interface d'administration web : allez dans
/ynh-dev/yunohost-admin/
- Vous pouvez aussi travailler sur les autres projets liés sur lesquels s'appuie YunoHost (SSOwat, moulinette) de façon similaire.
- Cœur Python/ligne de commande : allez dans
Vue d'ensemble des 4 morceaux principaux de YunoHost
Moulinette
C'est un petit framework "fait maison". Son rôle principal est de permettre de construire une API Web et une API en ligne de commande à partir d'un même code Python et d'un schéma YAML que nous appelons [l'actionmap] (https://github.com/YunoHost/yunohost/blob/stretch-unstable/data/actionsmap/yunohost.yml).
Il prend en charge d'autres mécanismes tels que l'authentification, l'internationalisation et des petites fonctions utilitaires techniques (par ex. lecture/écriture de fichiers json).
Moulinette dispose de sa propre documentation ici.
YunoHost
C'est le coeur même de YunoHost. Il contient :
- le code Python qui gère les utilisateurs, domaines, applications, services et autres
- des helpers bash principalement utilisés par les packageurs d'applications dans les scripts de ces applications
- des hooks et templates qui sont utilisés pour configurer les différents éléments de l'écosystème tels que nginx, postfix, ....
- des chaînes internationalisées
- des tests
SSOwat
C'est le système de connexion unique (single sign-on) de YunoHost. Il contient principalement:
- du code LUA interfacé directement avec nginx et qui gère tous les aspects "techniques" de l'authentification et de la gestion des accès aux ressources.
- le portail web utilisateur qui est l'interface finale visible par les utilisateurs de YunoHost
SSOwat est configuré via /etc/ssowat/conf.json
qui est généré par YunoHost.
YunoHost-admin
C'est une dépendance optionnelle de YunoHost et correspond à une interface pour l'API web créée par YunoHost et Moulinette (service yunohost-api
).
Il contient essentiellement :
- des templates pour les vues
- les contrôleurs JavaScript correspondants, qui interagissent avec l'API YunoHost
- et ses chaînes internationalisées
Travailler sur le cœur Python / ligne de commande
-
Allez dans
/ynh-dev/yunohost/
. -
Exécutez
cd /ynh-dev && ./ynh-dev use-git yunohost
. -
Le fichier actionsmap (
data/actionsmap/yunohost.yml
) définit les différentes catégories, actions et arguments de la ligne de commande YunoHost. Choisissez comment vous voulez que les utilisateurs utilisent votre fonctionnalité, et ajoutez/éditez les catégories, actions et arguments correspondants. Par exemple, dansyunohost domain add some.domain.tld
, la catégorie estdomain
, l'action estadd
etsome.domain.tld
est un argument. -
Moulinette va automatiquement faire le lien entre les commandes de l'actionsmap et les fonctions Python (ainsi que leurs arguments) dans
src/yunohost/
. Par exemple,yunohost domain add some.domain.tld
déclenchera un appel dedomain_add(domainName)
dansdomain.py
, avec l'argumentdomainName
qui vaudra"some.domain.tld"
.
Helpers / style de code
-
Pour gérer les exceptions, il existe un type
YunohostError()
-
Pour aider avec l'internationalisation des messages, utilisez
m18n.n('some-message-id')
et mettez le message correspondant danslocales/en.json
. Vous pouvez aussi utiliser des arguments pour construire les messages, avec{{some-argument:s}}
. Ne modifiez pas de fichiers de locales autres que en.json, la traduction sera faite avec Weblate ! -
YunoHost essaye de suivre le style de code PEP8. Des outils existent pour vérifier automatiquement la conformité du code.
-
Mettre un
_
devant les noms des fonctions "privées".
Travailler sur l'interface d'administration web
-
Allez dans
/ynh-dev/yunohost-admin/src/
. -
Exécutez
cd /ynh-dev && ./ynh-dev use-git yunohost-admin
. Ceci lance gulp, de sorte qu'à chaque fois que vous modifiez les sources, il recompilera le code (JS) et vous pourrez voir les changements dans le navigateur web (Ctrl+F5). Pour stopper la commande, faites simplement Ctrl+C. -
L'interface web utilise une API pour communiquer avec YunoHost. Les commandes/requêtes de l'API sont également définies dans l'actionsmap. Par exemple, accéder à la page
https://domain.tld/yunohost/api/users
correspond à une requêteGET /users
vers l'API YunoHost. Cette requête est mappée suruser_list()
. Accéder à cette URL devrait afficher le json retourné par cette fonction. Les requêtes 'GET' sont typiquement destinées à demander de l'information au serveur, tandis que les requêtes 'POST' sont destinées à demander au serveur de modifier/changer des informations ou de réaliser des actions. -
js/yunohost/controllers
contient les parties JavaScript, et définit quelles requêtes faire à l'API pendant le chargement d'une page donnée de l'interface, et comment traiter les données récupérées pour générer la page, en utilisant des templates. -
views
contient les templates des pages de l'interface. Dans le template, les données venant du JavaScript peuvent être utilisées avec la syntaxe{{some-variable}}
, qui sera remplacée pendant la construction de la page. Il est également possible d'avoir des conditions avec la syntaxe d'handlebars.js :{{#if some-variable}}<p>du HTML conditionnel ici !</p>{{/if}}
-
Pour l'internationalisation des messages, utilisez
y18n.t('some-string-code')
dans le JavaScript, ou{{t 'some-string-code'}}
dans le template HTML, et mettez votre message danslocales/en.json
. Ne modifiez pas de fichiers de locales autres que en.json, la traduction sera faite avec Weblate !
N'oubliez pas
-
À chaque modification de l'actionsmap, il faut redémarrer l'API yunohost :
service yunohost-api restart
(Il faudra retaper le mot de passe administrateur dans l'interface web) -
Il faudra peut-être régulièrement forcer le rafraîchissement du cache navigateur pour propager correctement le JavaScript et/ou HTML (à chaque fois que l'on change quelque chose dans
js
ouviews
, donc).
Votre fonctionnalité est prête et vous souhaitez qu'elle soit intégrée dans YunoHost
-
Forkez le dépòt correspondant sur Github, et commitez vos changements dans une nouvelle branche, Il est recommandé de nommer la branche avec la convention suivante :
- Pour une nouvelle fonctionnalité ou amélioration :
enh-ISSUENUMBER-description-fonctionnalité
- Pour une correction de bug :
fix-REDMINETICKET-description-correctif
ISSUENUMBER
est optionnel et correspond au numéro du ticket sur le bug tracker
- Pour une nouvelle fonctionnalité ou amélioration :
-
Une fois prêt, ouvrez une Pull Request (PR) sur Github. De préférence, inclure
[fix]
ou[enh]
au début du titre de la PR. -
Après relecture, test et validation par les autres contributeurs, votre branche sera mergée dans
unstable
!