Le code de vos projets n'est jamais parfait et se détériore avec le temps. C'est un problème important car, s'il est mal maîtrisé, vos temps de développement seront plus longs et vos bugs plus fréquents.
La définition de Wikipedia est éclairante:
Il s'inspire du concept existant de dette dans le contexte du financement des entreprises et l'applique au domaine du développement logiciel. En résumé, quand on code au plus vite et de manière non optimale, on contracte une dette technique que l'on rembourse tout au long de la vie du projet sous forme de temps de développement de plus en plus long et de bugs de plus en plus fréquents.
Sauf exception, si vous avez du code, vous avez de la dette. Ce n'est pas parce vous n'en parlez pas ou que vous ne la traitez pas qu'il n'y en a pas, c'est même plutôt le contraire.
Pour savoir où elle se cache, il y a des expressions et des comportements qui ne trompent pas :
Les endroits endettés sont souvent connus par les développeurs. On peut les garder longtemps sous le tapis si les membres du projet ne sont pas sensibilisés à ce problème ou si on leur refuse les moyens de s’en occuper. Pour les identifier une réunion courte sur le thème « Parlez-nous des parties du code qui vous posent des problèmes » suffit.
J'ai décidé de m'occuper de ma dette technique
Développer, c'est faire de la dette.
La dette volontaire :
La dette involontaire :
Ces différents facteurs peuvent se combiner et les « intérêts » augmenter proportionnellement : il sera ainsi très compliqué de faire intervenir de nouveaux développeurs sur du code non testé, surtout s'il utilise des outils qui ne sont plus supportés et devenus difficiles à installer.
Certaines parties d'un projet sont toujours de moins bonne qualité que les autres car le développement logiciel est une activité artisanale qui produit des résultats imparfait.
Comme l'indique la définition, du code endetté va être plus difficile à faire évoluer car, pour cela, il faudra « rembourser la dette ». Investir pour garder la dette technique sous contrôle permet de conserver un système adaptable et d'éviter d'être bloqué.
Coincés par la dette technique
L'enjeu est que cette vision soit partagée par l'équipe de développement et le product owner ou chef de projet.
Les non-développeurs ont parfois une vision erronée du refactoring: pour eux refactorer du code qui fonctionne est vu comme de la sur-qualité alors que le sujet est bel est bien la capacité à délivrer.
Lorsqu'il y a un désaccord sur ce point, il faut essayer de factualiser la dette au maximum. Mettre en regard le coût prévu de désendettement avec les surcoûts qu'elle a entrainé et les gains escomptés – diminution de risque et gain de vélocité – permet d'échanger de manière concrète. Mais ces prévisions ne sont jamais certaines, et accepter de développer moins de fonctionnalités pour traiter la dette demande de faire confiance aux développeurs. Si vous vous fiez à leur capacité de réaliser vos demandes, il faut aussi le faire dans ce cas là, ou vous prenez un risque réel sur le futur.
La première étape est d'admettre qu'elle existe et de faire un état des lieux.
On peut la matérialiser dans un backlog : ça permet d'avoir une bonne visibilité et de rester vigilant, attention cependant au backlog kilométrique qui étouffe les bonnes volontés et au backlog bouc émissaire où noter les problèmes permet d'éviter de les traiter.
Un investissement raisonnable mais régulier permet de traiter les cas simples en partant d'une base de code saine. Pour les cas moins favorables vous trouverez des suggestions plus bas.
Pas de remboursement de dette technique sans tests automatisés. S'il n'y en a pas ou pas assez, le premier pas doit être d'en ajouter. Si un temps limité impose de choisir entre test et refactoring mieux vaut mieux privilégier les tests quitte à reprendre le code plus tard : le risque est trop grand de ne pas aboutir ou d'aboutir à une situation plus mauvaise tout en ayant l'impression d'avoir progressé.
Sur un système qui n'a pas été conçu pour être testable ajouter des tests demande de modifier l'application. L'exercice est donc délicat. La meilleure approche consiste à démarrer avec de tests en boite noire puis de petit à petit rendre votre système testable en sécurisant chaque étape. Pour une description plus détaillée voir Working Effectively with Legacy Code de Michael Feathers.
Le refactoring sans test
Lors d'une longue action de refactoring, les livraisons de nouvelles fonctionnalités ne doivent jamais complètement s'arrêter au risque de briser la dynamique du projet. Il vaut mieux dédier un développeur seul pendant une ou plusieurs itérations en compartimentant les éléments à traiter plutôt que de mobiliser toute l'équipe.
Votre chef de projet si vous lui proposez que l'équipe consacre six mois à temps plein à reprendre le code existant
Attention à bien organiser ces tâches qui peuvent plaire à certains mais qui seront une punition pour d'autres. Comme elles peuvent être l'occasion de reprendre de la connaissance sur des composants oubliés, il faut faire attention à maintenir un équilibre entre les membres de l'équipe, pour éviter de basculer d'un « personne ne sait plus » à « X fait tous les refactoring donc maintenant c'est lui qui sait ».
Sur le long terme, l'important est d'avoir un système qui ne s'endette pas trop et de garder le contrôle. Les apports de l'agile et de software craftmanship sont ici précieux : il faut être vigilant, échanger et traiter à temps. Les feedback réguliers et les revues de code sous une forme ou une autre sont essentiels. Même s'ils sont imparfaits et ne remplacent pas un œil humain, les outils automatisés (détection de duplication… ) sont également utiles.
La première et la plus importante des choses est que l'équipe au sens large ait conscience des enjeux et soit impliquée sur le sujet.
Quand le code est sous contrôle, rien ne vous arrête
À l'origine, la métaphore est due à Ward Cunningham qui l'expliquait ainsi : parfois, on peut faire un écart temporaire à nos règles de qualités du design (simplicité, DRY, etc.) en vue de livrer rapidement (comprendre d'ici la fin de l'itération) une solution démontrable.
Pour une dette financière qui passe par un emprunt, on élabore un budget soutenable et on établit un contrat signé avec une banque. La dette technique n'a pas du tout ce fonctionnement là car elle se créé en partie involontairement et est bien plus difficile à maitriser.
Parler d'une base de code « lourdement endettée », c'est se voiler la face, il vaut mieux parler de code de mauvaise qualité ou de code insalubre : cela fait grincer des dents mais au moins les choses sont claires et tout le monde comprend qu'il faut s'en occuper.
La dette technique vous en avez. Si jusqu'à présent ce n'est pas un sujet que vous traitez, il est vital de faire un état des lieux (et si vous voulez, on peut vous aider).
La dette ce n'est pas que dans le code, c'est aussi vos outils (usine de développement, infrastructure) et vote méthode (process et organisation).
Cet article doit beaucoup aux échanges internes à Octo, merci aux participants.