Vous avez décidé de créer une API révolutionnaire avec Node.js. Bravo ! Mais avez-vous choisi Node pour les bonnes raisons ? Quelles sont les conséquences de cette décision ? Comment votre appli va-t-elle évoluer et comment éviter des downtimes ? C’est pour répondre à ces questions que nous sommes ici.
Node.js vous permet de n’avoir qu’une seule technologie que ce soit pour le front ou le back : Javascript. Finie la recherche d’un profil idéal qui maîtrise 3 technos : quelqu’un maîtrisant Javascript peut très vite monter en compétence et être opérationnel en quelques jours.
C’est certes un avantage en termes de recrutement, mais qu’en est-il des avantages techniques ?
Nous avons parlé des avantages, mais qu’en est-il des inconvénients ? Il y en a un principal : Node.js est mono-thread. Nous en reparlerons plus tard, mais il faudra à tout prix éviter les gros calculs pour ne pas bloquer le thread d’exécution de notre app.
Node.js a un cycle de release bien défini : il y a une release majeure tous les 6 mois, avec potentiellement des breaking changes. Les releases paires sont maintenues pendant une longue période (30 mois), ce qui permet de les utiliser librement. La dernière release paire est appelée la version “active”.
Les versions impaires sont utiles pour préparer le passage à la version paire suivante pour les développeurs de librairies.
Rythme des releases Node.js
Pour éviter d’avoir des problèmes de compatibilité, il faut choisir une version active ou en maintenance paire de Node.js pour tous les environnements (même de dév).
Les erreurs qui ne sont pas catchées avec un bloc try/catch termineront le processus. Ceci est fait pour éviter que l’application ne se trouve dans un état inconnu et qu’elle soit corrompue. Pour cela, il y a plusieurs choses à mettre en place :
Il existe un moyen de réagir aux erreurs non catchées : process.on(‘uncaughtException’). Il ne faut pas l’utiliser pour éviter que l’application s’arrête car elle se retrouverait dans un état corrompu. On peut en revanche l’utiliser pour nettoyer des ressources utilisées par l’application avant qu’elle ne se ferme et donc toujours le terminer avec process.exit(errorCode) s’il n’y a aucune opération asynchrone en cours ou avec process.exitCode = errorCode qui permet de ne pas forcer la fin du processus immédiatement.
Coder avec Node, c’est faire du Javascript, et Javascript est mono-thread. Cela signifie que toute opération synchrone bloque le processus qui correspond à votre application. Notez bien que j’ai dit Javascript : Node.js propose des fonctions synchrone ou asynchrone (ex: fs.readFile et fs.readFileSync), mais les opérations faites en Javascript, comme une boucle for, un map, des additions et autres sont elles 100% synchrones.
Si vous voulez savoir si l’opération que vous faites est bloquante ou non, référez vous à la documentation Node officielle.
Si vous voulez mieux comprendre le côté bloquant ou non du Javascript, je vous conseille la vidéo de Philip Roberts sur l’event-loop, faite lors d’une JsConf.
Si vous bloquez le processus de votre application, elle ne réagira plus. Vous avez peut-être déjà vu des sites ou des applis qui ne réagissaient plus au clic. Bloquer votre processus principal aura le même effet.
Documentation Node : https://nodejs.org/api/
Vidéo event loop : https://www.youtube.com/watch?v=8aGhZQkoFbQ
Il faut donc éviter au maximum de bloquer le processus. Mais qu’est-ce qui peut bloquer le processus ? Par exemple :
Bref, tout ce qui est une opération complexe est à éviter. Vous pouvez utiliser performance.now() avant et après l’opération et faire la différence pour avoir le temps qu’elle prend en millisecondes.
Mais comment faire si on doit passer par une opération complexe même après avoir optimisé son code au maximum ?
Il y a quelques solutions :
Disons, par exemple, que nous devons calculer un nombre de la suite de Fibonacci. Pour éviter de bloquer le processus principal, nous allons utiliser les Worker Threads :
index.js
<pre>
const {Worker} = require(“worker_threads”)
let num = 40
// Create new worker
const worker = new Worker(“./worker.js”, {workerData: {num: num}})
//Listen for a new message from worker
worker.once(“message”, result => {
console.log(`${num}th Fibonacci Number: ${result}`)
})
worker.on(”error”, result => {
console.log(error)
})
worker.on(“exit”, exitCode => {
console.log(exitCode)
})
console.log(“Executed in the parent thread”)
</pre>
worker.js
<pre>
const {parentPort, workerData} = require(“worker_threads”)
parentPort.postMessage(getFib(workerData.num))
function getFib(num) {
if (num === 0) {
return 0
}
else if (num === 1) {
return 1
}
else {
return getFib(num - 1) + getFib(num - 2)
}
}
</pre>
Ici, nous calculons le 40ème numéro de la suite de Fibonacci sans bloquer le processus principal et celui-ci sera affiché grâce au console.log() dans worker.once(“message”, …).
Pour le cas des APIs, vous pouvez aussi utiliser un Cluster qui crée des threads partageant les ports du serveur. Le processeur principal servira alors juste à rediriger les connexions utilisateurs vers le bon thread.
Cluster : https://nodejs.org/api/cluster.html#cluster
Worker threads : https://nodejs.org/api/worker_threads.html#worker-threads
Code : https://github.com/amatthieu/worker-thread-demo
Liste de bonnes pratiques :https://github.com/goldbergyoni/nodebestpractices
Notre Manifeste est le garant des droits et devoirs de chaque CodeWorker et des engagements que CodeWorks a vis-à-vis de chaque membre.
Il se veut réaliste, implémenté, partagé et inscrit dans une démarche d'amélioration continue.
Tu veux partager tes connaissances et ton temps avec des pairs empathiques, incarner une vision commune de l'excellence logicielle et participer activement à un modèle d'entreprise alternatif, rejoins-nous.