Koa.js - Guide rapide

Un framework d'application Web vous fournit une API simple pour créer des sites Web, des applications Web et des backends. Vous n'avez pas à vous soucier des protocoles de bas niveau, des processus, etc.

Qu'est-ce que Koa?

Koa fournit une interface minimale pour créer des applications. Il s'agit d'un très petit framework (600 LoC) qui fournit les outils nécessaires pour créer des applications et qui est assez flexible. Il existe de nombreux modules disponibles sur npm pour Koa, qui peuvent y être directement branchés. Koa peut être considéré comme le noyau d'express.js sans toutes les cloches et sifflets.

Pourquoi Koa?

Koa a un faible encombrement (600 LoC) et est une très fine couche d'abstraction sur le nœud pour créer des applications côté serveur. Il est complètement enfichable et possède une énorme communauté. Cela nous permet également d'étendre facilement Koa et de l'utiliser selon nos besoins. Il est construit en utilisant la technologie de pointe (ES6) qui lui donne un avantage sur les cadres plus anciens tels que express.

Carlin

Pug (anciennement connu sous le nom de Jade) est un langage laconique pour l'écriture de modèles HTML.

  • Produit du HTML
  • Prend en charge le code dynamique
  • Prend en charge la réutilisabilité (DRY)

C'est l'un des langages de création de modèles les plus populaires utilisés avec Koa.

MongoDB et Mongoose

MongoDB est une base de données de documents open source conçue pour faciliter le développement et la mise à l'échelle. Nous utiliserons cette base de données pour stocker des données.

Mongoose est une API client pour node.js qui facilite l'accès à notre base de données depuis notre application Koa.

Pour commencer à développer à l'aide du framework Koa, vous devez avoir installé Node et npm (gestionnaire de packages de nœuds). Si vous ne les avez pas déjà, accédez à la configuration du nœud pour installer le nœud sur votre système local. Vérifiez que le nœud et npm sont installés en exécutant les commandes suivantes dans votre terminal.

$ node --version
$ npm --version

Vous devriez recevoir une sortie similaire à -

v5.0.0
3.5.2

Veuillez vous assurer que la version de votre nœud est supérieure à 6.5.0. Maintenant que nous avons configuré Node et npm, comprenons ce qu'est npm et comment l'utiliser.

Gestionnaire de packages de nœuds (npm)

npm est le gestionnaire de packages pour node. Le registre npm est une collection publique de packages de code open source pour Node.js, les applications Web frontales, les applications mobiles, les robots, les routeurs et d'innombrables autres besoins de la communauté JavaScript. npm nous permet d'accéder à tous ces packages et de les installer localement. Vous pouvez parcourir la liste des packages disponibles sur npm à npmJS .

Comment utiliser npm?

Il existe deux façons d'installer un package à l'aide de npm: globalement et localement.

Globally- Cette méthode est généralement utilisée pour installer des outils de développement et des packages basés sur CLI. Pour installer un package globalement, utilisez la commande suivante.

$ npm install -g <package-name>

Locally- Cette méthode est généralement utilisée pour installer des frameworks et des bibliothèques. Un package installé localement ne peut être utilisé que dans le répertoire dans lequel il est installé. Pour installer un package localement, utilisez la même commande que ci-dessus sans le -g drapeau.

$ npm install <package-name>

Chaque fois que nous créons un projet à l'aide de npm, nous devons fournir un fichier package.json, qui contient tous les détails sur notre projet. npm nous facilite la configuration de ce fichier. Mettons en place notre projet de développement.

Step 1 - Lancez votre terminal / cmd, créez un nouveau dossier nommé hello-world et cd dedans -

Step 2 - Maintenant, pour créer le fichier package.json à l'aide de npm, utilisez ce qui suit.

npm init

Il vous demandera les informations suivantes -

Continuez simplement à appuyer sur Entrée et entrez votre nom dans le champ «nom de l'auteur».

Step 3- Maintenant que nous avons notre fichier package.json configuré, nous allons installer Koa. Pour installer Koa et l'ajouter dans notre fichier package.json, utilisez la commande suivante.

$ npm install --save koa

Pour confirmer que Koa est correctement installé, exécutez la commande suivante.

$ ls node_modules #(dir node_modules for windows)

Tip - Le --save l'indicateur peut être remplacé par -Sdrapeau. Cet indicateur garantit que Koa est ajouté en tant que dépendance à notre fichier package.json. Cela présente un avantage, la prochaine fois que nous aurons besoin d'installer toutes les dépendances de notre projet, nous devons simplement exécuter la commande npm install et il trouvera les dépendances dans ce fichier et les installera pour nous.

C'est tout ce dont nous avons besoin pour commencer le développement à l'aide du framework Koa. Pour rendre notre processus de développement beaucoup plus facile, nous installerons un outil de npm, nodemon. Ce que fait cet outil, c'est qu'il redémarre notre serveur dès que nous modifions l'un de nos fichiers, sinon nous devons redémarrer le serveur manuellement après chaque modification de fichier. Pour installer nodemon, utilisez la commande suivante.

$ npm install -g nodemon

Maintenant, nous sommes tous prêts à plonger dans Koa!

Une fois que nous avons configuré le développement, il est temps de commencer à développer notre première application en utilisant Koa. Créez un nouveau fichier appeléapp.js et tapez ce qui suit.

var koa = require('koa');
var app = new koa();

app.use(function* (){
   this.body = 'Hello world!';
});

app.listen(3000, function(){
   console.log('Server running on https://localhost:3000')
});

Enregistrez le fichier, accédez à votre terminal et tapez.

$ nodemon app.js

Cela démarrera le serveur. Pour tester cette application, ouvrez votre navigateur et accédez àhttps://localhost:3000 et vous devriez recevoir le message suivant.

Comment fonctionne cette application?

La première ligne importe Koa dans notre fichier. Nous avons accès à son API via la variable Koa. Nous l'utilisons pour créer une application et l'assigner à var app.

app.use(function)- Cette fonction est un middleware, qui est appelé chaque fois que notre serveur reçoit une requête. Nous en apprendrons plus sur le middleware dans les chapitres suivants. La fonction de rappel est un générateur, que nous verrons dans le chapitre suivant. Le contexte de ce générateur est appelé contexte en Koa. Ce contexte est utilisé pour accéder et modifier les objets de requête et de réponse. Nous faisons en sorte que le corps de cette réponse soitHello world!.

app.listen(port, function)- Cette fonction lie et écoute les connexions sur le port spécifié. Le port est le seul paramètre requis ici. La fonction de rappel est exécutée si l'application s'exécute avec succès.

L'une des nouvelles fonctionnalités les plus intéressantes de JavaScript ES6 est une nouvelle génération de fonctions, appelée générateur. Avant les générateurs, l'ensemble du script était généralement utilisé pour s'exécuter de haut en bas, sans moyen simple d'arrêter l'exécution du code et de reprendre plus tard avec la même pile. Les générateurs sont des fonctions qui peuvent être quittées et réintroduites ultérieurement. Leur contexte (liaisons variables) sera sauvegardé à travers les rentrées.

Les générateurs nous permettent d'arrêter l'exécution du code entre les deux. Jetons donc un coup d'œil à un générateur simple.

var generator_func = function* (){
   yield 1;
   yield 2;
};

var itr = generator_func();
console.log(itr.next());
console.log(itr.next());
console.log(itr.next());

Lors de l'exécution du code ci-dessus, le résultat sera le suivant.

{ value: 1, done: false }
{ value: 2, done: false }
{ value: undefined, done: true }

Regardons à l'intérieur du code ci-dessus. Nous créons d'abord un générateur appelégenerator_func(). Nous avons créé une instance de cette fonction étrange et l'avons assignée àitr. Puis nous avons commencé à appelernext() sur cette variable itr.

L'appel de next () démarre le générateur et il s'exécute jusqu'à ce qu'il atteigne un rendement. Ensuite, il renvoie l'objet avec valeur et terminé, où la valeur a la valeur d'expression. Cette expression peut être n'importe quoi. À ce stade, il interrompt l'exécution. Encore une fois, lorsque nous appelons cette fonction (next), le générateur reprend l'exécution à partir du dernier seuil d'élasticité, l'état de la fonction étant le même au moment de la pause, jusqu'au prochain seuil d'élimination. Ceci est fait jusqu'à ce qu'il n'y ait plus de points de rendement dans le code.

Générateurs à Koa

Alors, pourquoi parlons-nous des générateurs dans ce didacticiel. Comme vous vous en souvenez peut-être dans le programme hello world, nous avons utilisé unfunction* ()notation pour passer un rappel à app.use (). Koa est un objet, qui contient un tableau de fonctions génératrices de middleware, qui sont toutes composées et exécutées à la manière d'une pile à chaque requête. Koa met également en œuvre l'aval suivi de l'amont du flux de contrôle.

Jetez un œil à l'exemple suivant pour mieux comprendre cela.

var koa = require('koa');
var app = koa();
 
app.use(function* (next) {
   //do something before yielding to next generator function 
   
   //in line which will be 1st event in downstream
   console.log("1");
   yield next;
 
   //do something when the execution returns upstream, 
   //this will be last event in upstream
   console.log("2");
});
app.use(function* (next) {
   // This shall be 2nd event downstream
   console.log("3");
   yield next;
 
   // This would be 2nd event upstream
   console.log("4");
});
app.use(function* () { 
   // Here it would be last function downstream
   console.log("5");
   
   // Set response body
   this.body = "Hello Generators";

   // First event of upstream (from the last to first)
   console.log("6");
});

app.listen(3000);

Lors de l'exécution du code ci-dessus et de la navigation vers https://localhost:3000/ nous obtenons la sortie suivante sur notre console.

1
3
5
6
4
2

C'est essentiellement ainsi que Koa utilise les générateurs. Cela nous permet de créer un middleware compact en utilisant cette propriété et d'écrire du code pour les fonctionnalités en amont et en aval, nous évitant ainsi les rappels.

Les frameworks Web fournissent des ressources telles que des pages HTML, des scripts, des images, etc. sur différentes routes. Koa ne prend pas en charge les routes dans le module principal. Nous devons utiliser le module Koa-router pour créer facilement des routes dans Koa. Installez ce module à l'aide de la commande suivante.

npm install --save koa-router

Maintenant que Koa-router est installé, regardons un exemple simple de route GET.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var _ = router();              //Instantiate the router
_.get('/hello', getMessage);   // Define routes

function *getMessage() {
   this.body = "Hello world!";
};

app.use(_.routes());           //Use the routes defined using the router
app.listen(3000);

Si nous exécutons notre application et allons sur localhost: 3000 / hello, le serveur reçoit une requête get sur la route "/ hello". Notre application Koa exécute la fonction de rappel associée à cette route et envoie "Hello World!" comme réponse.

Nous pouvons également avoir plusieurs méthodes différentes sur le même itinéraire. Par exemple,

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var _ = router(); //Instantiate the router

_.get('/hello', getMessage);
_.post('/hello', postMessage);

function *getMessage() {
	this.body = "Hello world!";
};
function *postMessage() {
   this.body = "You just called the post method at '/hello'!\n";
};
app.use(_.routes()); //Use the routes defined using the router
app.listen(3000);

Pour tester cette requête, ouvrez votre terminal et utilisez cURL pour exécuter la requête suivante

curl -X POST "https://localhost:3000/hello"

Une méthode spéciale, all, est fourni par express pour gérer tous les types de méthodes http sur une route particulière en utilisant la même fonction. Pour utiliser cette méthode, essayez ce qui suit -

_.all('/test', allMessage);

function *allMessage(){
   this.body = "All HTTP calls regardless of the verb will get this response";
};

Nous pouvons maintenant définir des itinéraires; ils sont statiques ou fixes. Pour utiliser des routes dynamiques, nous devons fournir différents types de routes. L'utilisation de routes dynamiques nous permet de passer des paramètres et de traiter en fonction de ceux-ci. Voici un exemple d'itinéraire dynamique.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var _ = router();

_.get('/:id', sendID);

function *sendID() {
   this.body = 'The id you specified is ' + this.params.id;
}

app.use(_.routes());
app.listen(3000);

Pour tester cela, allez à https://localhost:3000/123. Vous obtiendrez la réponse suivante.

Vous pouvez remplacer «123» dans l'URL par autre chose et cela sera reflété dans la réponse. Voici un exemple complexe de ce qui précède.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var _ = router();

_.get('/things/:name/:id', sendIdAndName);

function *sendIdAndName(){
   this.body = 'id: ' + this.params.id + ' and name: ' + this.params.name;
};

app.use(_.routes());

app.listen(3000);

Pour tester cela, allez à https://localhost:3000/things/tutorialspoint/12345.

Vous pouvez utiliser le this.paramsobject pour accéder à tous les paramètres que vous passez dans l'URL. Notez que les deux ci-dessus ont des chemins différents. Ils ne se chevaucheront jamais. De plus, si vous souhaitez exécuter le code lorsque vous obtenez '/ things', vous devez le définir séparément.

Itinéraires avec correspondance de modèle

Vous pouvez également utiliser regex pour restreindre la correspondance des paramètres d'URL. Supposons que vous ayez besoin que l'identifiant soit composé de cinq chiffres. Vous pouvez utiliser la définition d'itinéraire suivante.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var _ = router();

_.get('/things/:id([0-9]{5})', sendID);

function *sendID(){
   this.body = 'id: ' + this.params.id;
}

app.use(_.routes());
app.listen(3000);

Notez que cela va onlycorrespondent aux demandes qui ont un identifiant long à 5 chiffres. Vous pouvez utiliser des expressions régulières plus complexes pour faire correspondre / valider vos itinéraires. Si aucun de vos itinéraires ne correspond à la demande, vous recevrez un message Not found en réponse.

Par exemple, si nous définissons les mêmes routes que ci-dessus, sur demande avec une URL valide, nous obtenons -

La méthode HTTP est fournie dans la requête et spécifie l'opération que le client a demandée. Le tableau suivant résume les méthodes HTTP couramment utilisées.

Sr.No. Méthode et description
1

GET

La méthode GET demande une représentation de la ressource spécifiée. Les requêtes utilisant GET ne doivent récupérer que des données et ne doivent avoir aucun autre effet.

2

POST

La méthode POST demande que le serveur accepte les données incluses dans la demande en tant que nouvel objet / entité de la ressource identifiée par l'URI.

3

PUT

La méthode PUT demande au serveur d'accepter les données incluses dans la demande en tant que modification de l'objet existant identifié par l'URI. S'il n'existe pas, la méthode PUT doit en créer un.

4

DELETE

La méthode DELETE demande au serveur de supprimer la ressource spécifiée.

Ce sont les méthodes HTTP les plus courantes. Pour en savoir plus à leur sujet, rendez-vous surhttps://www.tutorialspoint.com/http/http_methods.htm.

Un objet Koa Request est une abstraction au-dessus de l'objet de requête vanilla du nœud, fournissant des fonctionnalités supplémentaires utiles pour le développement de serveurs HTTP au quotidien. L'objet de requête Koa est intégré dans l'objet de contexte,this. Déconnectons l'objet de requête chaque fois que nous recevons une requête.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var _ = router();

_.get('/hello', getMessage);

function *getMessage(){
   console.log(this.request);
   this.body = 'Your request has been logged.';
}
app.use(_.routes());
app.listen(3000);

Lorsque vous exécutez ce code et accédez à https://localhost:3000/hello, vous recevrez alors la réponse suivante.

Sur votre console, vous obtiendrez l'objet de requête déconnecté.

{ 
   method: 'GET',
   url: '/hello/',
   header: 
   { 
      host: 'localhost:3000',
      connection: 'keep-alive',
      'upgrade-insecure-requests': '1',
      'user-agent': 'Mozilla/5.0 (X11; Linux x86_64) 
         AppleWebKit/537.36 (KHTML, like Gecko) Chrome/52.0.2743.116 Safari/537.36',
      accept: 'text/html,application/xhtml+xml,
         application/xml;q = 0.9,image/webp,*/*;q = 0.8',
      dnt: '1',
      'accept-encoding': 'gzip, deflate, sdch',
      'accept-language': 'en-US,en;q = 0.8' 
   }
}

Nous avons accès à de nombreuses propriétés utiles de la requête en utilisant cet objet. Regardons quelques exemples.

request.header

Fournit tous les en-têtes de demande.

request.method

Fournit la méthode de requête (GET, POST, etc.)

request.href

Fournit l'URL de la requête complète.

request.path

Fournit le chemin de la demande. Sans chaîne de requête ni URL de base.

request.query

Donne la chaîne de requête analysée. Par exemple, si nous enregistrons cela sur une demande telle quehttps://localhost:3000/hello/?name=Ayush&age=20&country=India, alors nous obtiendrons l'objet suivant.

{
   name: 'Ayush',
   age: '20',
   country: 'India'
}

request.accepts (type)

Cette fonction renvoie vrai ou faux selon que les ressources demandées acceptent le type de demande donné.

Vous pouvez en savoir plus sur l'objet de requête dans la documentation sur Request .

Un objet Koa Response est une abstraction au-dessus de l'objet de réponse vanille du nœud, fournissant des fonctionnalités supplémentaires utiles pour le développement quotidien de serveurs HTTP. L'objet de réponse Koa est intégré dans l'objet de contexte,this. Déconnectons l'objet de réponse chaque fois que nous recevons une demande.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var _ = router();

_.get('/hello', getMessage);

function *getMessage(){
   this.body = 'Your request has been logged.';
   console.log(this.response);
}

app.use(_.routes());
app.listen(3000);

Lorsque vous exécutez ce code et accédez à https://localhost:3000/hello alors vous recevrez la réponse suivante.

Sur votre console, vous obtiendrez l'objet de requête déconnecté.

{ 
   status: 200,
   message: 'OK',
   header: 
   {
      'content-type': 'text/plain; charset=utf-8',
      'content-length': '12' 
   },
   body: 'Your request has been logged.' 
}

Le statut et le message sont automatiquement définis par Koa mais peuvent être modifiés par nos soins. Si nous ne définissons pas le corps de la réponse, le code d'état est défini sur 404. Une fois que nous avons défini le corps de la réponse, l'état est défini sur 200 par défaut. Nous pouvons explicitement remplacer ce comportement.

Nous avons accès à de nombreuses propriétés utiles de la réponse en utilisant cet objet. Regardons quelques exemples -

response.header

Fournit tous les en-têtes de réponse.

response.status

Fournit l'état de la réponse (200, 404, 500, etc.). Cette propriété est également utilisée pour définir l'état de la réponse.

response.message

Fournit le message de réponse. Cette propriété est également utilisée pour définir des messages personnalisés avec des réponses. Il est associé à response.status.

response.body

Obtenez ou définissez le corps de la réponse. Habituellement, nous y accédons à l'aide de l'objet context. C'est juste une autre façon d'y accéder. Le corps peut être du type: String, Buffer, Stream, Object ou Null.

response.type

Obtenez ou définissez le type de contenu de la réponse actuelle.

response.get (champ)

Cette fonction est utilisée pour obtenir les valeurs des en-têtes avec un champ de valeur insensible à la casse.

response.set (champ, valeur)

Cette fonction est utilisée pour définir un en-tête sur la réponse à l'aide d'une paire champ et valeur.

response.remove (champ)

Cette fonction est utilisée pour annuler un en-tête sur la réponse à l'aide d'un nom de champ.

Vous pouvez en savoir plus sur l'objet de réponse dans la documentation sur Response .

La redirection est très importante lors de la création de sites Web. Si une URL malformée est demandée ou s'il y a des erreurs sur votre serveur, vous devez les rediriger vers les pages d'erreur respectives. Les redirections peuvent également être utilisées pour empêcher les gens d'accéder aux zones restreintes de votre site Web.

Créons une page d'erreur et redirigeons vers cette page chaque fois que quelqu'un demande une URL mal formée.

var koa = require('koa');
var router = require('koa-router');
var app = koa();
var _ = router();

_.get('/not_found', printErrorMessage);
_.get('/hello', printHelloMessage);

app.use(_.routes());
app.use(handle404Errors);

function *printErrorMessage() {
   this.status = 404;
   this.body = "Sorry we do not have this resource.";
}
function *printHelloMessage() {
   this.status = 200;
   this.body = "Hey there!";
}
function *handle404Errors(next) {
   if (404 != this.status) return;
   this.redirect('/not_found');
}
app.listen(3000);

Lorsque nous exécutons ce code et naviguons vers une route autre que / hello, nous serons redirigés vers / not_found. Nous avons placé le middleware à la fin (appel de la fonction app.use vers ce middleware). Cela garantit que nous atteignons enfin le middleware et envoyons la réponse correspondante. Voici les résultats que nous voyons lorsque nous exécutons le code ci-dessus.

Lorsque nous naviguons vers https://localhost:3000/hello, nous obtenons -

Si nous naviguons vers une autre route, nous obtenons -

La gestion des erreurs joue un rôle important dans la création d'applications Web. Koa utilise également un middleware à cette fin.

Dans Koa, vous ajoutez un middleware qui fait try { yield next }comme l'un des premiers middleware. Si nous rencontrons une erreur en aval, nous retournons à la clause catch associée et traitons l'erreur ici. Par exemple -

var koa = require('koa');
var app = koa();

//Error handling middleware
app.use(function *(next) {
   try {
      yield next;
   } catch (err) {
      this.status = err.status || 500;
      this.body = err.message;
      this.app.emit('error', err, this);
   }
});

//Create an error in the next middleware
//Set the error message and status code and throw it using context object

app.use(function *(next) {
   //This will set status and message
   this.throw('Error Message', 500);
});

app.listen(3000);

Nous avons délibérément créé une erreur dans le code ci-dessus et gérons l'erreur dans le bloc catch de notre premier middleware. Ceci est ensuite émis sur notre console et envoyé comme réponse à notre client. Voici le message d'erreur que nous recevons lorsque nous déclenchons cette erreur.

InternalServerError: Error Message
   at Object.module.exports.throw 
      (/home/ayushgp/learning/koa.js/node_modules/koa/lib/context.js:91:23)
   at Object.<anonymous> (/home/ayushgp/learning/koa.js/error.js:18:13)
   at next (native)
   at onFulfilled (/home/ayushgp/learning/koa.js/node_modules/co/index.js:65:19)
   at /home/ayushgp/learning/koa.js/node_modules/co/index.js:54:5
   at Object.co (/home/ayushgp/learning/koa.js/node_modules/co/index.js:50:10)
   at Object.toPromise (/home/ayushgp/learning/koa.js/node_modules/co/index.js:118:63)
   at next (/home/ayushgp/learning/koa.js/node_modules/co/index.js:99:29)
   at onFulfilled (/home/ayushgp/learning/koa.js/node_modules/co/index.js:69:7)
   at /home/ayushgp/learning/koa.js/node_modules/co/index.js:54:5

À l'heure actuelle, toute demande envoyée au serveur entraînera cette erreur.

Les fonctions middleware sont des fonctions qui ont accès au context objectet la fonction middleware suivante dans le cycle demande-réponse de l'application. Ces fonctions sont utilisées pour modifier les objets de requête et de réponse pour des tâches telles que l'analyse des corps de requête, l'ajout d'en-têtes de réponse, etc. Koa va encore plus loin en donnant'downstream', puis refouler le contrôle 'upstream'. Cet effet s'appellecascading.

Voici un exemple simple d'une fonction middleware en action.

var koa = require('koa');
var app = koa();
var _ = router();

//Simple request time logger
app.use(function* (next) {
   console.log("A new request received at " + Date.now());
   
   //This function call is very important. It tells that more processing is 
   //required for the current request and is in the next middleware function/route handler.
   yield next;
});

app.listen(3000);

Le middleware ci-dessus est appelé pour chaque requête sur le serveur. Ainsi, après chaque demande, nous obtiendrons le message suivant dans la console.

A new request received at 1467267512545

Pour le restreindre à une route spécifique (et à tous ses sous-itinéraires), il suffit de créer les routes comme nous l'avons fait pour le routage. En fait, ce sont ces middlewares uniquement qui gèrent notre demande.

Par exemple,

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var _ = router();

//Simple request time logger
_.get('/request/*', function* (next) {
   console.log("A new request received at " + Date.now());
   yield next;
});

app.use(_.routes());
app.listen(3000);

Maintenant, chaque fois que vous demandez un sous-itinéraire de '/ request', alors seulement il enregistrera l'heure.

Ordre des appels middleware

L'une des choses les plus importantes à propos des middlewares dans Koa est que l'ordre dans lequel ils sont écrits / inclus dans votre fichier, est l'ordre dans lequel ils sont exécutés en aval. Dès que nous atteignons une instruction yield dans un middleware, il passe au middleware suivant en ligne, jusqu'à ce que nous atteignions le dernier. Ensuite, nous recommençons à remonter et à reprendre les fonctions à partir des instructions yield.

Par exemple, dans l'extrait de code suivant, la première fonction s'exécute d'abord jusqu'à yield, puis le deuxième middleware jusqu'à yield, puis la troisième. Comme nous n'avons plus de middleware ici, nous commençons à remonter, à exécuter dans l'ordre inverse, c'est-à-dire troisième, deuxième, premier. Cet exemple résume comment utiliser le middleware à la manière Koa.

var koa = require('koa');
var app = koa();

//Order of middlewares
app.use(first);
app.use(second);
app.use(third);

function *first(next) {
   console.log("I'll be logged first. ");
   
   //Now we yield to the next middleware
   yield next;
   
   //We'll come back here at the end after all other middlewares have ended
   console.log("I'll be logged last. ");
};

function *second(next) {
   console.log("I'll be logged second. ");
   yield next;
   console.log("I'll be logged fifth. ");
};

function *third(next) {
   console.log("I'll be logged third. ");
   yield next;
   console.log("I'll be logged fourth. ");
};

app.listen(3000);

Lorsque nous visitons '/' après avoir exécuté ce code, sur notre console, nous obtiendrons -

I'll be logged first. 
I'll be logged second. 
I'll be logged third. 
I'll be logged fourth. 
I'll be logged fifth. 
I'll be logged last.

Le diagramme suivant résume ce qui se passe réellement dans l'exemple ci-dessus.

Maintenant que nous savons comment créer notre propre middleware, parlons de certains des middlewares créés par la communauté les plus couramment utilisés.

Middleware tiers

Une liste des middlewares tiers pour express est disponible ici. Voici quelques-uns des intergiciels les plus couramment utilisés -

  • koa-bodyparser
  • koa-router
  • koa-static
  • koa-compress

Nous discuterons de plusieurs intergiciels dans les chapitres suivants.

Pug est un moteur de création de modèles. Les moteurs de création de modèles sont utilisés pour supprimer l'encombrement du code de notre serveur avec du HTML, concaténant les chaînes de manière extravagante avec des modèles HTML existants. Pug est un moteur de création de modèles très puissant, doté de nombreuses fonctionnalités telles quefilters, includes, inheritance, interpolation, etc. Il y a beaucoup de chemin à parcourir à ce sujet.

Pour utiliser Pug avec Koa, nous devons l'installer à l'aide de la commande suivante.

$ npm install --save pug koa-pug

Une fois pug installé, définissez-le comme moteur de création de modèles pour votre application. Ajoutez le code suivant à votre fichier app.js.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var Pug = require('koa-pug');
var pug = new Pug({
   viewPath: './views',
   basedir: './views',
   app: app //Equivalent to app.use(pug)
});

var _ = router(); //Instantiate the router

app.use(_.routes()); //Use the routes defined using the router
app.listen(3000);

Maintenant, créez un nouveau répertoire appelé vues. Dans le répertoire, créez un fichier nommé first_view.pug et saisissez-y les données suivantes.

doctype html
html
   head
      title = "Hello Pug"
   body
      p.greetings#people Hello Views!

Pour exécuter cette page, ajoutez l'itinéraire suivant à votre application.

_.get('/hello', getMessage); // Define routes

function *getMessage(){
   this.render('first_view');
};

Vous recevrez la sortie comme -

Ce que fait Pug, c'est qu'il convertit ce balisage très simple en html. Nous n'avons pas besoin de garder une trace de la fermeture de nos balises, pas besoin d'utiliser des mots-clés de classe et d'identifiant, utilisez plutôt "." et '#' pour les définir. Le code ci-dessus est d'abord converti en

<!DOCTYPE html>
<html>
   <head>
      <title>Hello Pug</title>
   </head>
    
   <body>
      <p class = "greetings" id = "people">Hello Views!</p>
   </body>
</html>

Pug est capable de faire bien plus que de simplifier le balisage HTML. Explorons certaines de ces fonctionnalités de Pug.

Balises simples

Les balises sont imbriquées en fonction de leur indentation. Comme dans l'exemple ci-dessus,<title> était en retrait dans le <head>tag, donc c'était à l'intérieur. Cependant, le<body> tag était sur la même indentation, donc c'était un frère de <head> marque.

Nous n'avons pas besoin de fermer les balises. Dès que Pug rencontre la balise suivante au même niveau ou au niveau d'indentation externe, il ferme la balise pour nous.

Il existe trois méthodes pour mettre du texte à l'intérieur d'une balise -

  • Espace séparé -
h1 Welcome to Pug
  • Texte diffusé -
div
   | To insert multiline text, 
   | You can use the pipe operator.
  • Bloc de texte -
div.
   But that gets tedious if you have a lot of text. 
   You can use "." at the end of tag to denote block of text. 
   To put tags inside this block, simply enter tag in a new line and 
   indent it accordingly.

commentaires

Pug utilise la même syntaxe que JavaScript (//) pour créer des commentaires. Ces commentaires sont convertis en commentaires html (<! - comment ->). Par exemple,

//This is a Pug comment

Ce commentaire est converti en -

<!--This is a Pug comment-->

Les attributs

Pour définir les attributs, nous utilisons une liste d'attributs séparés par des virgules, entre parenthèses. Les attributs de classe et d'ID ont des représentations spéciales. La ligne de code suivante couvre la définition des attributs, des classes et de l'ID pour une balise html donnée.

div.container.column.main#division(width = "100",height = "100")

Cette ligne de code est convertie en -

<div class = "container column main" id = "division" width = "100" height = "100"></div>

Passer des valeurs aux modèles

Lorsque nous rendons un modèle Pug, nous pouvons en fait lui transmettre une valeur de notre gestionnaire de route, que nous pouvons ensuite utiliser dans notre modèle. Créez un nouveau gestionnaire d'itinéraire avec le code suivant.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var Pug = require('koa-pug');
var pug = new Pug({
   viewPath: './views',
   basedir: './views',
   app: app // equals to pug.use(app) and app.use(pug.middleware)
});

var _ = router(); //Instantiate the router

_.get('//dynamic_view', dynamicMessage); // Define routes

function *dynamicMessage(){
   this.render('dynamic', {
      name: "TutorialsPoint", 
      url:"https://www.tutorialspoint.com"
   });
};

app.use(_.routes()); //Use the routes defined using the router
app.listen(3000);

Ensuite, créez un nouveau fichier de vue dans le répertoire de vues, nommé dynamic.pug, à l'aide du code suivant.

html
   head
      title = name
   body
      h1 = name
      a(href = url) URL

Ouvert localhost:3000/dynamicdans votre navigateur et les suivants devraient être la sortie. -

Nous pouvons également utiliser ces variables passées dans le texte. Pour insérer des variables passées entre le texte d'une balise, nous utilisons la syntaxe # {variableName}. Par exemple, dans l'exemple ci-dessus, si nous voulons insérer des salutations à partir de TutorialsPoint, nous devons utiliser le code suivant.

html
   head
      title = name
   body
      h1 Greetings from #{name}
      a(href = url) URL

Cette méthode d'utilisation des valeurs est appelée interpolation.

Conditionnels

Nous pouvons également utiliser des instructions conditionnelles et des constructions en boucle. Considérez cet exemple pratique, si un utilisateur est connecté, nous voudrions afficher "Salut, Utilisateur" et sinon, nous voudrions lui montrer un lien "Connexion / Inscription". Pour y parvenir, nous pouvons définir un modèle simple tel que -

html
   head
      title Simple template
   body
      if(user)
         h1 Hi, #{user.name}
      else
         a(href = "/sign_up") Sign Up

Lorsque nous rendons cela en utilisant nos routes, et si nous passons un objet comme -

this.render('/dynamic',{user: 
   {name: "Ayush", age: "20"}
});

Il donnera un message affichant Salut, Ayush. Cependant, si nous ne transmettons aucun objet ou n'en transmettons aucun sans clé utilisateur, nous obtiendrons un lien d'inscription.

Inclure et composants

Pug fournit un moyen très intuitif de créer des composants pour une page Web. Par exemple, si vous voyez un site Web d'actualités, l'en-tête avec le logo et les catégories est toujours fixe. Au lieu de copier cela dans chaque vue, nous pouvons utiliser un include. L'exemple suivant montre comment nous pouvons utiliser un include -

Créez trois vues avec le code suivant -

header.pug

div.header.
   I'm the header for this website.

content.pug

html
   head
      title Simple template
   body
      include ./header.pug
      h3 I'm the main content
      include ./footer.pug

footer.pug

div.footer.
   I'm the footer for this website.

Créez un itinéraire pour cela comme suit.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var Pug = require('koa-pug');
var pug = new Pug({
   viewPath: './views',
   basedir: './views',
   app: app //Equivalent to app.use(pug)
});

var _ = router(); //Instantiate the router

_.get('/components', getComponents);

function *getComponents(){
   this.render('content.pug');
}

app.use(_.routes()); //Use the routes defined using the router
app.listen(3000);

Aller à localhost:3000/components, vous devriez obtenir la sortie suivante.

include peut également être utilisé pour inclure du texte brut, CSS et JavaScript.

Il existe de nombreuses autres fonctionnalités de Pug. Cependant, ceux-ci sont hors de portée de ce didacticiel. Vous pouvez explorer davantage Pug à Pug .

Les formulaires font partie intégrante du Web. Presque tous les sites Web que nous visitons nous proposent des formulaires qui soumettent ou récupèrent des informations pour nous. Pour commencer avec les formulaires, nous allons d'abord installer le koa-body. Pour l'installer, allez sur votre terminal et utilisez -

$ npm install --save koa-body

Remplacez le contenu de votre fichier app.js par le code suivant.

var koa = require('koa');
var router = require('koa-router');
var bodyParser = require('koa-body');
var app = koa();

//Set up Pug
var Pug = require('koa-pug');
var pug = new Pug({
   viewPath: './views',
   basedir: './views',
   app: app //Equivalent to app.use(pug)
});

//Set up body parsing middleware
app.use(bodyParser({
   formidable:{uploadDir: './uploads'},
   multipart: true,
   urlencoded: true
}));

_.get('/', renderForm);
_.post('/', handleForm);

function * renderForm(){
   this.render('form');
}
function *handleForm(){
   console.log(this.request.body);
   console.log(this.req.body);
   this.body = this.request.body; //This is where the parsed request is stored
}

app.use(_.routes()); 
app.listen(3000);

Les nouvelles choses que nous faisons ici sont l'importation de l'analyseur corporel et du multer. Nous utilisons l'analyseur de corps pour analyser les demandes d'en-tête json et x-www-form-urlencoded, tandis que nous utilisons multer pour analyser multipart / form-data.

Créons un formulaire html pour tester cela! Créez une nouvelle vue nommée form.pug avec le code suivant.

html
   head
      title Form Tester
   body
      form(action = "/", method = "POST")
         div
            label(for = "say") Say: 
            input(name = "say" value = "Hi")
         br
         div
            label(for = "to") To: 
            input(name = "to" value = "Koa form")
         br
         button(type = "submit") Send my greetings

Exécutez votre serveur en utilisant -

nodemon index.js

Maintenant, allez sur localhost: 3000 / et remplissez le formulaire comme vous le souhaitez, puis soumettez-le. Vous recevrez la réponse comme -

Jetez un œil à votre console, elle vous montrera le corps de votre requête en tant qu'objet JavaScript. Par exemple -

le this.request.bodyobject contient votre corps de requête analysé. Pour utiliser les champs de cet objet, utilisez-les simplement comme des objets JS normaux.

Ce n'est qu'une façon d'envoyer une demande. Il existe de nombreuses autres façons, mais celles-ci ne sont pas pertinentes à couvrir ici, car notre application Koa traitera toutes ces demandes de la même manière. Pour en savoir plus sur les différentes façons de faire une demande, consultez cette page.

Les applications Web doivent fournir la fonctionnalité permettant les téléchargements de fichiers. Voyons comment nous pouvons recevoir des fichiers des clients et les stocker sur notre serveur.

Nous avons déjà utilisé le middleware koa-body pour analyser les requêtes. Ce middleware est également utilisé pour gérer les téléchargements de fichiers. Créons un formulaire qui nous permet de télécharger des fichiers puis de sauvegarder ces fichiers en utilisant Koa. Créez d'abord un modèle nomméfile_upload.pug avec le contenu suivant.

html
   head
      title File uploads
   body
      form(action = "/upload" method = "POST" enctype = "multipart/form-data")
         div
            input(type = "text" name = "name" placeholder = "Name")
         
         div
            input(type = "file" name = "image")
         
         div
            input(type = "submit")

Notez que vous devez donner le même type d'encodage que ci-dessus dans votre formulaire. Maintenant, traitons ces données sur notre serveur.

var koa = require('koa');
var router = require('koa-router');
var bodyParser = require('koa-body');
var app = koa();

//Set up Pug
var Pug = require('koa-pug');
var pug = new Pug({
   viewPath: './views',
   basedir: './views',
   app: app 
});

//Set up body parsing middleware
app.use(bodyParser({
   formidable:{uploadDir: './uploads'},    //This is where the files would come
   multipart: true,
   urlencoded: true
}));

var _ = router(); //Instantiate the router

_.get('/files', renderForm);
_.post('/upload', handleForm);

function * renderForm(){
   this.render('file_upload');
}

function *handleForm(){
   console.log("Files: ", this.request.body.files);
   console.log("Fields: ", this.request.body.fields);
   this.body = "Received your data!"; //This is where the parsed request is stored
}

app.use(_.routes()); 
app.listen(3000);

Lorsque vous exécutez cela, vous obtenez le formulaire suivant.

Lorsque vous soumettez ceci, votre console produira la sortie suivante.

Les fichiers qui ont été téléchargés sont stockés dans le chemin dans la sortie ci-dessus. Vous pouvez accéder aux fichiers de la demande en utilisantthis.request.body.files et les champs de cette demande par this.request.body.fields.

Les fichiers statiques sont des fichiers que les clients téléchargent lorsqu'ils sont sur le serveur. Créez un nouveau répertoire,public. Express, par défaut, ne vous permet pas de servir des fichiers statiques.

Nous avons besoin d'un middleware pour atteindre cet objectif. Allez-y et installezkoa-serve -

$ npm install --save koa-static

Maintenant nous devons usece middleware. Avant cela, créez un répertoire appelé public. Nous stockerons tous nos fichiers statiques ici. Cela nous permet de garder notre code serveur sécurisé car rien au-dessus de ce dossier public ne serait accessible aux clients. Après avoir créé un répertoire public, créez un fichier nomméhello.txtdedans avec n'importe quel contenu que vous aimez. Ajoutez maintenant ce qui suit à votre app.js.

var serve = require('koa-static');
var koa = require('koa');
var app = koa();

app.use(serve('./public'));

app.listen(3000);

Note- Koa recherche les fichiers relatifs au répertoire statique, donc le nom du répertoire statique ne fait pas partie de l'URL. La route racine est maintenant définie sur votre répertoire public, donc tous les fichiers statiques que vous chargez seront considérés comme publics comme racine. Pour vérifier que cela fonctionne correctement, exécutez votre application et visitezhttps://localhost:3000/hello.txt

Vous devriez obtenir la sortie suivante. Notez que ce n'est pas un document HTML ou une vue Pug, mais plutôt un simple fichier txt.

Dirs statiques multiples

Nous pouvons également définir plusieurs répertoires d'actifs statiques en utilisant -

var serve = require('koa-static');
var koa = require('koa');
var app = koa();

app.use(serve('./public'));
app.use(serve('./images'));

app.listen(3000);

Désormais, lorsque nous demandons un fichier, Koa recherchera ces répertoires et nous enverra le fichier correspondant.

Les cookies sont de petits fichiers / données simples qui sont envoyés au client avec une demande du serveur et stockés côté client. Chaque fois que l'utilisateur charge le site Web, ce cookie est envoyé avec la demande. Cela permet de suivre les actions des utilisateurs. Il existe de nombreuses utilisations des cookies HTTP.

  • Gestion de session
  • Personnalisation (systèmes de recommandation)
  • Suivi des utilisateurs

Pour utiliser des cookies avec Koa, nous avons les fonctions: ctx.cookies.set() et ctx.cookies.get(). Pour définir un nouveau cookie, définissons une nouvelle route dans notre application Koa.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

_.get('/', setACookie);

function *setACookie() {
   this.cookies.set('foo', 'bar', {httpOnly: false});
}

var _ = router();

app.use(_.routes());
app.listen(3000);

Pour vérifier si le cookie est défini ou non, accédez simplement à votre navigateur, lancez la console et entrez -

console.log(document.cookie);

Cela produira le résultat suivant (vous pouvez avoir plus de cookies définis peut-être en raison d'extensions dans votre navigateur).

"foo = bar"

Voici un exemple de ce qui précède.

Le navigateur renvoie également des cookies à chaque fois qu'il interroge le serveur. Pour afficher un cookie sur votre serveur, sur la console du serveur dans une route, ajoutez le code suivant à cette route.

console.log('Cookies: foo = ', this.cookies.get('foo'));

La prochaine fois que vous enverrez une demande à cet itinéraire, vous obtiendrez le résultat suivant.

Cookies: foo = bar

Ajout de cookies avec heure d'expiration

Vous pouvez ajouter des cookies qui expirent. Pour ajouter un cookie qui expire, transmettez simplement un objet avec la propriété «expires» définie sur l'heure à laquelle vous souhaitez qu'il expire. Par exemple,

var koa = require('koa');
var router = require('koa-router');
var app = koa();

_.get('/', setACookie);

function *setACookie(){
   //Expires after 360000 ms from the time it is set.
	this.cookies.set('name', 'value', { 
      httpOnly: false, expires: 360000 + Date.now() });
}

var _ = router();

app.use(_.routes());
app.listen(3000);

Supprimer les cookies existants

Pour supprimer un cookie, définissez simplement le cookie sur une chaîne vide. Par exemple, si vous devez effacer un cookie nomméfoo, utilisez le code suivant.

var koa = require('koa');
var router = require('koa-router');
var app = koa();

_.get('/', setACookie);

function *setACookie(){
   //Expires after 360000 ms from the time it is set.
   this.cookies.set('name', '');
}

var _ = router();

app.use(_.routes());
app.listen(3000);

Cela annulera ledit cookie. Notez que vous devez laisser leHttpOnly option à true lorsque vous n'utilisez pas le cookie dans le code côté client.

HTTP est sans état, par conséquent, pour associer une requête à toute autre requête, vous avez besoin d'un moyen de stocker les données utilisateur entre les requêtes HTTP. Les cookies et les paramètres d'URL sont tous deux des moyens appropriés pour transporter des données entre le client et le serveur. Cependant, ils sont tous les deux lisibles côté client. Les sessions résolvent exactement ce problème. Vous attribuez un ID au client et il effectue toutes les autres demandes en utilisant cet ID. Les informations associées au client sont stockées sur le serveur lié à cet ID.

Nous aurons besoin de la koa-session, donc installez-la en utilisant -

npm install --save koa-session

Nous mettrons le koa-sessionmiddleware en place. Dans cet exemple, nous utiliserons la RAM pour stocker les sessions. Ne l'utilisez jamais dans des environnements de production. L'intergiciel de session gère tout, c'est-à-dire la création de la session, la configuration du cookie de session et la création de l'objet de session dans l'objet de contexte.

Chaque fois que nous faisons à nouveau une demande du même client, nous aurons leurs informations de session stockées avec nous (étant donné que le serveur n'a pas été redémarré). Nous pouvons ajouter plus de propriétés à cet objet de session. Dans l'exemple suivant, nous allons créer un compteur de vues pour un client.

var session = require('koa-session');
var koa = require('koa');
var app = koa();

app.keys = ['Shh, its a secret!'];
app.use(session(app));  // Include the session middleware

app.use(function *(){
   var n = this.session.views || 0;
   this.session.views = ++n;
   
   if(n === 1)
      this.body = 'Welcome here for the first time!';
   else
      this.body = "You've visited this page " + n + " times!";
})

app.listen(3000);

Ce que fait le code ci-dessus, c'est que lorsqu'un utilisateur visite le site, il crée une nouvelle session pour l'utilisateur et attribue un cookie. Lors de la prochaine visite de l'utilisateur, le cookie est vérifié et la variable de session page_view est mise à jour en conséquence.

Maintenant, si vous exécutez l'application et accédez à localhost:3000, vous obtiendrez la réponse suivante.

Si vous revisitez la page, le compteur de pages augmentera. Dans ce cas, la page a été actualisée 12 fois.

L'authentification est un processus dans lequel les informations d'identification fournies sont comparées à celles enregistrées dans la base de données des informations des utilisateurs autorisés sur un système d'exploitation local ou dans un serveur d'authentification. Si les informations d'identification correspondent, le processus est terminé et l'utilisateur reçoit une autorisation d'accès.

Nous allons créer un système d'authentification très basique qui utilisera Basic HTTP Authentication. C'est le moyen le plus simple d'appliquer le contrôle d'accès car il ne nécessite pas de cookies, de sessions ou quoi que ce soit d'autre. Pour utiliser cela, le client doit envoyer l'en-tête d'autorisation avec chaque demande qu'il fait. Le nom d'utilisateur et le mot de passe ne sont pas chiffrés, mais sont concaténés en une seule chaîne comme la suivante.

username:password

Cette chaîne est codée avec Base64 et le mot Basic est placé avant cette valeur. Par exemple, si votre nom d'utilisateur est Ayush et votre mot de passe India, la chaîne"Ayush:India" serait envoyé tel que codé dans l'en-tête d'autorisation.

Authorization: Basic QXl1c2g6SW5kaWE=

Pour implémenter cela dans votre application koa, vous aurez besoin du middleware koa-basic-auth. Installez-le en utilisant -

$ npm install --save koa-basic-auth

Ouvrez maintenant votre fichier app.js et entrez le code suivant.

//This is what the authentication would be checked against
var credentials = { name: 'Ayush', pass: 'India' }

var koa = require('koa');
var auth = require('koa-basic-auth');
var _ = require('koa-router')();

var app = koa();

//Error handling middleware
app.use(function *(next){
   try {
      yield next;
   } catch (err) {
      if (401 == err.status) {
         this.status = 401;
         this.set('WWW-Authenticate', 'Basic');
         this.body = 'You have no access here';
      } else {
         throw err;
      }
   }
});

// Set up authentication here as first middleware. 
// This returns an error if user is not authenticated.
_.get('/protected', auth(credentials), function *(){
   this.body = 'You have access to the protected area.';
   yield next;
});

// No authentication middleware present here.
_.get('/unprotected', function*(next){
   this.body = "Anyone can access this area";
   yield next;
});

app.use(_.routes());
app.listen(3000);

Nous avons créé un middleware de gestion des erreurs pour gérer toutes les erreurs liées à l'authentification. Ensuite, nous avons créé 2 itinéraires -

  • /protected- Cette route n'est accessible que si l'utilisateur envoie l'en-tête d'authentification correct. Pour tous les autres, cela donnera une erreur.

  • /unprotected - Cette route est accessible à tous, avec ou sans authentification.

Maintenant, si vous envoyez une demande à / protected sans en-tête d'authentification ou avec les mauvaises informations d'identification, vous recevrez une erreur. Par exemple,

$ curl https://localhost:3000/protected

Vous recevrez la réponse comme -

HTTP/1.1 401 Unauthorized
WWW-Authenticate: Basic
Content-Type: text/plain; charset=utf-8
Content-Length: 28
Date: Sat, 17 Sep 2016 19:05:56 GMT
Connection: keep-alive

Please authenticate yourself

Cependant, avec les bonnes informations d'identification, vous obtiendrez la réponse attendue. Par exemple,

$ curl -H "Authorization: basic QXl1c2g6SW5kaWE=" https://localhost:3000/protected -i

Vous obtiendrez la réponse comme -

HTTP/1.1 200 OK
Content-Type: text/plain; charset=utf-8
Content-Length: 38
Date: Sat, 17 Sep 2016 19:07:33 GMT
Connection: keep-alive

You have access to the protected area.

L'itinéraire / non protégé est toujours accessible à tous.

La compression est un moyen simple et efficace d'économiser de la bande passante et d'accélérer votre site. Il n'est compatible qu'avec les navigateurs modernes et doit être utilisé avec prudence si vos utilisateurs utilisent également des navigateurs hérités.

Lors de l'envoi de réponses depuis le serveur, si la compression est utilisée, cela peut considérablement améliorer le temps de chargement. Nous utiliserons un middleware appelékoa-compress pour prendre en charge la compression des fichiers ainsi que la définition des en-têtes appropriés.

Allez-y et installez le middleware en utilisant -

$ npm install --save koa-compress

Maintenant, dans votre fichier app.js, ajoutez le code suivant -

var koa = require('koa');
var router = require('koa-router');
var app = koa();

var Pug = require('koa-pug');
var pug = new Pug({
   viewPath: './views',
   basedir: './views',
   app: app //Equivalent to app.use(pug)
});

app.use(compress({
   filter: function (content_type) {
      return /text/i.test(content_type)
   },
   threshold: 2048,
   flush: require('zlib').Z_SYNC_FLUSH
}));

var _ = router(); //Instantiate the router

_.get('/', getRoot);

function *getRoot(next){
   this.render('index');
}

app.use(_.routes()); //Use the routes defined using the router
app.listen(3000);

Cela met en place notre middleware de compression. L'option de filtre est une fonction qui vérifie le type de contenu de la réponse pour décider s'il faut compresser. L'option de seuil est la taille de réponse minimale en octets à compresser. Cela garantit que nous ne compressons pas chaque petite réponse.

Voici une réponse sans compression.

Voici la réponse similaire avec la compression.

Si vous regardez l'onglet de taille en bas, vous pouvez très bien voir la différence entre les deux. Il y a plus de 150% d'amélioration, lorsque nous compressons les fichiers.

La mise en cache est le terme pour stocker des réponses réutilisables afin de rendre les demandes ultérieures plus rapides. Chaque navigateur est livré avec une implémentation d'un cache HTTP. Tout ce que nous avons à faire est de nous assurer que chaque réponse du serveur fournit des directives d'en-tête HTTP correctes pour indiquer au navigateur quand et pendant combien de temps la réponse peut être mise en cache par le navigateur.

Voici quelques avantages de l'inclusion de la mise en cache dans vos applications Web -

  • Vos coûts de réseau diminuent. Si votre contenu est mis en cache, vous devrez en envoyer moins pour chaque demande ultérieure.

  • La vitesse et les performances de votre site Web augmentent.

  • Votre contenu peut être rendu disponible même si votre client est hors ligne.

Nous utiliserons le middleware koa-static-cache pour implémenter la mise en cache dans notre application. Installez ces middleware en utilisant -

$ npm install --save koa-static-cache

Accédez à votre fichier app.js et ajoutez-y le code suivant.

var koa = require('koa');
var app = koa();

var path = require('path');
var staticCache = require('koa-static-cache');

app.use(staticCache(path.join(__dirname, 'public'), {
   maxAge: 365 * 24 * 60 * 60  //Add these files to caches for a year
}))

app.listen(3000);

le koa-static-cachele middleware est utilisé pour mettre en cache les réponses du serveur côté client. lecache-controll'en-tête est défini en fonction des options que nous fournissons lors de l'initialisation de l'objet cache. Nous avons fixé le délai d'expiration de cette réponse mise en cache à 1 an. Voici les comparaisons de demandes que nous avons envoyées avant et après la mise en cache du fichier.

Avant la mise en cache de ce fichier, le code d'état renvoyé était 200, ce qui est OK. Les en-têtes de réponse contenaient plusieurs informations concernant le contenu à mettre en cache et avaient également donné unETag pour le contenu.

La prochaine fois que la demande a été envoyée, elle a été envoyée avec l'ETtag. Puisque notre contenu n'avait pas changé sur le serveur, son ETag correspondant est également resté le même et le client a été informé que la copie qu'il a localement est à jour avec ce que le serveur fournirait et devrait utiliser le local au lieu de demander encore.

Note- Pour invalider un fichier mis en cache, il vous suffit de changer son nom de fichier et de mettre à jour sa référence. Cela garantira que vous avez un nouveau fichier à envoyer au client et que le client ne peut pas le recharger à partir du cache.

Nous recevons les demandes, mais ne les stockons nulle part. Nous avons besoin d'une base de données pour stocker les données. Nous utiliserons une célèbre base de données NoSQL appeléeMongoDB. Pour installer et lire sur Mongo, rendez-vous sur ce lien.

Afin d'utiliser Mongo avec Koa, nous avons besoin d'une API client pour le nœud. Il existe plusieurs options pour nous, mais pour ce tutoriel, nous nous en tiendrons à la mangouste . La mangouste est utilisée pourdocument modelingdans Node pour MongoDB. La modélisation de documents signifie que nous allons créer unModel (un peu comme un class en programmation orientée document), puis nous produirons documents en utilisant ce modèle (comme nous créons documents of a classen POO). Tous nos traitements se feront sur ces "documents", puis enfin, nous écrirons ces documents dans notre base de données.

Configurer Mongoose

Maintenant que Mongo est installé, installons mongoose, de la même manière que nous avons installé nos autres packages de nœuds.

$ npm install --save mongoose

Avant de commencer à utiliser mongoose, nous devons créer une base de données à l'aide du shell Mongo. Pour créer une nouvelle base de données, ouvrez votre terminal et entrez "mongo". Un shell Mongo démarre, entrez ce qui suit.

use my_db

Une nouvelle base de données sera créée pour vous. Chaque fois que vous ouvrez le shell Mongo, il sera par défaut "test" db et vous devrez passer à votre base de données en utilisant la même commande que ci-dessus.

Pour utiliser mongoose, nous en aurons besoin dans notre fichier app.js, puis nous nous connecterons au service mongod fonctionnant sur mongodb: // localhost

var koa = require('koa');
var _ = require('koa-router')();
var app = koa();

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/my_db');

app.use(_.routes());
app.listen(3000);

Maintenant que notre application est connectée à notre base de données, créons un nouveau modèle. Ce modèle agira comme une collection dans notre base de données. Pour créer un nouveau modèle, utilisez le code suivant avant de définir des itinéraires.

var koa = require('koa');
var _ = require('koa-router')();
var app = koa();

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/my_db');

var personSchema = mongoose.Schema({
   name: String,
   age: Number,
   nationality: String
});

var Person = mongoose.model("Person", personSchema);

app.use(_.routes());
app.listen(3000);

Le code ci-dessus définit le schéma d'une personne et est utilisé pour créer un modèle mangouste Person.

Enregistrer des documents

Nous allons maintenant créer un nouveau formulaire html, qui obtiendra les détails d'une personne et l'enregistrera dans notre base de données. Pour créer le formulaire, créez un nouveau fichier de vue appelé person.pug dans le répertoire de vues avec le contenu suivant.

html
   head
      title Person
   body
      form(action = "/person", method = "POST")
         div
            label(for = "name") Name: 
            input(name = "name")
         br
         div
            label(for = "age") Age: 
            input(name = "age")
         br
         div
            label(for = "nationality") Nationality: 
            input(name = "nationality")
         br
         button(type = "submit") Create new person

Ajoutez également une nouvelle route get dans index.js pour rendre ce document.

var koa = require('koa');
var _ = require('koa-router')();
var app = koa();

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/my_db');

var personSchema = mongoose.Schema({
   name: String,
   age: Number,
   nationality: String
});

var Person = mongoose.model("Person", personSchema);

_.get('/person', getPerson);

function *getPerson(next){
   this.render('person');
   yield next;
}

app.use(_.routes());
app.listen(3000);

Allez sur localhost: 3000 / personne pour vérifier si notre formulaire s'affiche correctement. Notez que ce n'est que l'interface utilisateur, cela ne fonctionne pas encore. Voici à quoi ressemble notre formulaire.

Nous allons maintenant définir un gestionnaire de route de publication à '/ person' qui gérera cette requête.

var koa = require('koa');
var _ = require('koa-router')();
var app = koa();

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/my_db');

var personSchema = mongoose.Schema({
   name: String,
   age: Number,
   nationality: String
});

var Person = mongoose.model("Person", personSchema);

_.post('/person', createPerson);

function *createPerson(next){
   var self = this;
   var personInfo = self.request.body; //Get the parsed information
   
   if(!personInfo.name || !personInfo.age || !personInfo.nationality){
      self.render(
         'show_message', {message: "Sorry, you provided wrong info", type: "error"});
   } else {
      var newPerson = new Person({
         name: personInfo.name,
         age: personInfo.age,
         nationality: personInfo.nationality
      });
      yield newPerson.save(function(err, res) {
         if(err)
            self.render('show_message', 
               {message: "Database error", type: "error"});
         else
            self.render('show_message', 
               {message: "New person added", type: "success", person: personInfo});
      });
   }
}

app.use(_.routes());
app.listen(3000);

Dans le code ci-dessus, si nous recevons un champ vide ou ne recevons aucun champ, nous enverrons une réponse d'erreur. Cependant, si nous recevons un document bien formé, nous créons un document newPerson à partir du modèle Person et l'enregistrons dans notre base de données en utilisantnewPerson.save()fonction. Ceci est défini dans mangouste et accepte un rappel comme argument. Ce rappel a deux arguments,error et response. Cela rendra la vue show_message, nous devons donc la créer également.

Pour afficher la réponse de cette route, nous devrons également créer un show_messagevue. Créez une nouvelle vue avec le code suivant.

html
   head
      title Person
   body
      if(type = "error")
         h3(style = "color:red") #{message}
      else
         h3 New person, name: 
            #{person.name}, age: 
            #{person.age} and nationality: 
            #{person.nationality} added!

Voici la réponse que nous recevons en soumettant avec succès le formulaire (show_message.pug).

Nous avons maintenant une interface pour créer des personnes!

Récupération de documents

Mongoose fournit de nombreuses fonctions pour récupérer des documents, nous nous concentrerons sur trois d'entre elles. Toutes ces fonctions prennent également un rappel comme dernier paramètre, et tout comme la fonction de sauvegarde, leurs arguments sont l'erreur et la réponse.

Les trois fonctions sont -

Model.find (conditions, rappel)

Cette fonction trouve tous les documents correspondant aux champs de l'objet conditions. Les mêmes opérateurs utilisés dans Mongo fonctionnent également dans la mangouste. Par exemple, cela récupérera tous les documents de la collection des personnes.

Person.find(function(err, response){
   console.log(response);
});

Cela récupérera tous les documents dont le nom de champ est "Ayush" et l'âge est de 20 ans.

Person.find({name: "Ayush", age: 20}, 
   function(err, response){
      console.log(response);
   });

Nous pouvons également fournir la projection dont nous avons besoin, c'est-à-dire les champs dont nous avons besoin. Par exemple, si nous voulons seulement lenames des gens dont nationalityest "indien" , nous utilisons -

Person.find({nationality: "Indian"}, 
   "name", function(err, response) {
      console.log(response);
   });

Model.findOne (conditions, rappel)

Cette fonction récupère toujours un seul document le plus pertinent. Il a les mêmes arguments exacts que Model.find ().

Model.findById (id, rappel)

Cette fonction prend dans le _id(défini par mongo) comme premier argument, une chaîne de projection facultative et un rappel pour gérer la réponse. Par exemple,

Person.findById("507f1f77bcf86cd799439011", 
   function(err, response){
      console.log(response);
   });

Créons un itinéraire pour afficher tous les enregistrements de personnes.

var koa = require('koa');
var _ = require('koa-router')();
var app = koa();

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/my_db');

var personSchema = mongoose.Schema({
   name: String,
   age: Number,
   nationality: String
});

var Person = mongoose.model("Person", personSchema);

_.get('/people', getPeople);
function *getPeople(next){
   var self = this;
   
   yield Person.find(function(err, response){
      self.body = response;
   });
}
app.use(_.routes());
app.listen(3000);

Mise à jour des documents

Mongoose fournit trois fonctions pour mettre à jour les documents.

Model.update (condition, mises à jour, rappel)

Cette fonction prend une condition et met à jour l'objet en entrée et applique les modifications à tous les documents correspondant aux conditions de la collection. Par exemple, le code suivant mettra à jour tous les documents Personne pour avoir une nationalité «américaine».

Person.update({age: 25},
   {nationality: "American"}, 
   function(err, response){
      console.log(response);
   });

Model.findOneAndUpdate (condition, mises à jour, rappel)

Il fait exactement ce qui est dit. Recherche un document en fonction de la requête et le met à jour en fonction du deuxième argument. Il prend également un rappel comme dernier argument. Par exemple,

Person.findOneAndUpdate({name: "Ayush"}, 
   {age: 40}, 
   function(err, response){
      console.log(response);
   });

Model.findByIdAndUpdate (id, mises à jour, rappel)

Cette fonction met à jour un seul document identifié par son identifiant. Par exemple,

Person.findByIdAndUpdate("507f1f77bcf86cd799439011", 
   {name: "James"}, 
   function(err, response){
      console.log(response);
   });

Créons une route pour mettre à jour les gens. Ce sera une route PUT avec l'identifiant comme paramètre et les détails dans la charge utile.

var koa = require('koa');
var _ = require('koa-router')();
var app = koa();
var mongoose = require('mongoose');

mongoose.connect('mongodb://localhost/my_db');

var personSchema = mongoose.Schema({
   name: String,
   age: Number,
   nationality: String
});

var Person = mongoose.model("Person", personSchema);

_.put('/people/:id', updatePerson);

function *updatePerson() {
   var self = this;
   yield Person.findByIdAndUpdate(self.params.id, 
      {$set: {self.request.body}}, function(err, response){
      
      if(err) {
         self.body = {
            message: "Error in updating person with id " + self.params.id};
      } else {
         self.body = response;
      }
   });
}

app.use(_.routes());
app.listen(3000);

Pour tester cet itinéraire, saisissez ce qui suit dans votre terminal (remplacez l'identifiant par un identifiant des personnes que vous avez créées).

curl -X PUT --data "name = James&age = 20&nationality = American" https://localhost:3000/people/507f1f77bcf86cd799439011

Cela mettra à jour le document associé à l'identifiant fourni dans l'itinéraire avec les détails ci-dessus.

Supprimer des documents

Nous avons couvert Ccréer, Read et Update, nous allons maintenant voir comment la mangouste peut être utilisée pour supprimer des documents. Il y a trois fonctions ici, exactement comme la mise à jour.

Model.remove (condition, [rappel])

Cette fonction prend un objet de condition en entrée et supprime tous les documents correspondant aux conditions. Par exemple, si nous devons supprimer toutes les personnes âgées de 20 ans,

Person.remove({age:20});

Model.findOneAndRemove (condition, [rappel])

Cette fonction supprime un single, document le plus pertinent selon les conditions objet. Par exemple,

Person.findOneAndRemove({name: "Ayush"});

Model.findByIdAndRemove (id, [rappel])

Cette fonction supprime un seul document identifié par son identifiant. Par exemple,

Person.findByIdAndRemove("507f1f77bcf86cd799439011");

Créons maintenant une route pour supprimer des personnes de notre base de données.

var koa = require('koa');
var _ = require('koa-router')();
var app = koa();

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/my_db');

var personSchema = mongoose.Schema({
   name: String,
   age: Number,
   nationality: String
});

var Person = mongoose.model("Person", personSchema);

_.delete('/people/:id', deletePerson);
function *deletePerson(next){
   var self = this;
   yield Person.findByIdAndRemove(self.params.id, function(err, response){
      if(err) {
         self.body = {message: "Error in deleting record id " + self.params.id};
      } else {
         self.body = {message: "Person with id " + self.params.id + " removed."};
      }
   });
}

app.use(_.routes());
app.listen(3000);

Pour tester cela, utilisez la commande curl suivante -

curl -X DELETE https://localhost:3000/people/507f1f77bcf86cd799439011

Cela supprimera la personne avec l'identifiant donné produisant le message suivant. -

{message: "Person with id 507f1f77bcf86cd799439011 removed."}

Cela conclut comment nous pouvons créer des applications CRUD simples en utilisant MongoDB, mongoose et Koa. Pour explorer davantage la mangouste, lisez la documentation de l' API.

Pour créer des applications mobiles, des applications à page unique, utiliser des appels AJAX et fournir des données aux clients, vous aurez besoin d'une API. Un style architectural populaire sur la façon de structurer et de nommer ces API et les points de terminaison est appeléREST(Representational Transfer State). HTTP 1.1 a été conçu en gardant à l'esprit les principes REST. REST a été introduit parRoy Fielding en 2000 dans son article Fielding Dissertations.

Les URI et méthodes RESTful nous fournissent presque toutes les informations dont nous avons besoin pour traiter une demande. Le tableau suivant résume comment les différents verbes doivent être utilisés et comment les URI doivent être nommés. Nous allons créer une API de films vers la fin, alors discutons de la manière dont elle sera structurée.

Méthode URI Détails Fonction
AVOIR /films Sûr, cachable Obtient la liste de tous les films et leurs détails
AVOIR / films / 1234 Sûr, cachable Obtient les détails de l'ID de film 1234
PUBLIER /films N / A Crée un nouveau film avec les détails fournis. La réponse contient l'URI de cette ressource nouvellement créée.
METTRE / films / 1234 Idempotent Modifie l'ID de film 1234 (en crée un s'il n'existe pas déjà). La réponse contient l'URI de cette ressource nouvellement créée.
EFFACER / films / 1234 Idempotent L'ID de film 1234 doit être supprimé, s'il existe. La réponse doit contenir le statut de la demande.
SUPPRIMER ou METTRE /films Invalide Doit être invalide. DELETE et PUT doivent spécifier la ressource sur laquelle ils travaillent.

Créons maintenant cette API dans Koa. Nous utiliserons JSON comme format de données de transport car il est facile à utiliser en JavaScript et présente de nombreux autres avantages. Remplacez votre fichier index.js par ce qui suit -

INDEX.JS

var koa = require('koa');
var router = require('koa-router');
var bodyParser = require('koa-body');

var app = koa();

//Set up body parsing middleware
app.use(bodyParser({
   formidable:{uploadDir: './uploads'},
   multipart: true,
   urlencoded: true
}));

//Require the Router we defined in movies.js
var movies = require('./movies.js');

//Use the Router on the sub route /movies
app.use(movies.routes());

app.listen(3000);

Maintenant que notre application est configurée, concentrons-nous sur la création de l'API. Commencez par configurer le fichier movies.js. Nous n'utilisons pas de base de données pour stocker les films, mais les stockons en mémoire, de sorte que chaque fois que le serveur redémarre, les films que nous avons ajoutés disparaissent. Cela peut facilement être imité en utilisant une base de données ou un fichier (en utilisant le module node fs).

Importez koa-router, créez un routeur et exportez-le en utilisant module.exports.

var Router = require('koa-router');
var router = Router({
  prefix: '/movies'
});  //Prefixed all routes with /movies

var movies = [
   {id: 101, name: "Fight Club", year: 1999, rating: 8.1},
   {id: 102, name: "Inception", year: 2010, rating: 8.7},
   {id: 103, name: "The Dark Knight", year: 2008, rating: 9},
   {id: 104, name: "12 Angry Men", year: 1957, rating: 8.9}
];

//Routes will go here

module.exports = router;

OBTENIR des itinéraires

Définissez l'itinéraire GET pour obtenir tous les films.

router.get('/', sendMovies);
function *sendMovies(next){
   this.body = movies;
   yield next;
}

C'est ça. Pour tester si cela fonctionne correctement, exécutez votre application, puis ouvrez votre terminal et entrez -

curl -i -H "Accept: application/json" -H "Content-Type: application/json" -X GET localhost:3000/movies

Vous obtiendrez la réponse suivante -

[{"id":101,"name":"Fight 
Club","year":1999,"rating":8.1},{"id":102,"name":"Inception","year":2010,"rating":8.7},
{"id":103,"name":"The Dark Knight","year":2008,"rating":9},{"id":104,"name":"12 Angry 
Men","year":1957,"rating":8.9}]

Nous avons un itinéraire pour obtenir tous les films. Créons maintenant un itinéraire pour obtenir un film spécifique par son identifiant.

router.get('/:id([0-9]{3,})', sendMovieWithId);

function *sendMovieWithId(next){
   var ctx = this;
   var currMovie = movies.filter(function(movie){
      if(movie.id == ctx.params.id){
         return true;
      }
   });
   if(currMovie.length == 1){
      this.body = currMovie[0];
   } else {
      this.response.status = 404;//Set status to 404 as movie was not found
      this.body = {message: "Not Found"};
   }
   yield next;
}

Cela nous permettra d'obtenir les films en fonction de l'identifiant que nous fournissons. Pour tester cela, utilisez la commande suivante dans votre terminal.

curl -i -H "Accept: application/json" -H "Content-Type: application/json" -X GET localhost:3000/movies/101

Vous obtiendrez la réponse comme -

{"id":101,"name":"Fight Club","year":1999,"rating":8.1}

Si vous visitez un itinéraire non valide, cela produira une erreur ne peut pas GET, tandis que si vous visitez un itinéraire valide avec un identifiant qui n'existe pas, cela produira une erreur 404.

Nous en avons terminé avec les routes GET. Maintenant, passons à la route POST.

Route POST

Utilisez l'itinéraire suivant pour gérer les données POSTées.

router.post('/', addNewMovie);

function *addNewMovie(next){
   //Check if all fields are provided and are valid:
   if(!this.request.body.name || 
      !this.request.body.year.toString().match(/^[0-9]{4}$/g) || 
      !this.request.body.rating.toString().match(/^[0-9]\.[0-9]$/g)){
      
      this.response.status = 400;
      this.body = {message: "Bad Request"};
   } else {
      var newId = movies[movies.length-1].id+1;
      
      movies.push({
         id: newId,
         name: this.request.body.name,
         year: this.request.body.year,
         rating: this.request.body.rating
      });
      this.body = {message: "New movie created.", location: "/movies/" + newId};
   }
   yield next;
}

Cela créera un nouveau film et le stockera dans la variable films. Pour tester cet itinéraire, saisissez ce qui suit dans votre terminal -

curl -X POST --data "name = Toy%20story&year = 1995&rating = 8.5" 
https://localhost:3000/movies

Vous obtiendrez la réponse suivante -

{"message":"New movie created.","location":"/movies/105"}

Pour tester si cela a été ajouté à l'objet movies, exécutez à nouveau la demande d'obtention pour / movies / 105. Vous obtiendrez la réponse suivante -

{"id":105,"name":"Toy story","year":"1995","rating":"8.5"}

Passons à la création des routes PUT et DELETE.

Itinéraire PUT

La route PUT est presque exactement la même que la route POST. Nous spécifierons l'identifiant de l'objet qui sera mis à jour / créé. Créez l'itinéraire de la manière suivante -

router.put('/:id', updateMovieWithId);

function *updateMovieWithId(next){
   //Check if all fields are provided and are valid:
   if(!this.request.body.name || 
      !this.request.body.year.toString().match(/^[0-9]{4}$/g) || 
      !this.request.body.rating.toString().match(/^[0-9]\.[0-9]$/g) ||
      !this.params.id.toString().match(/^[0-9]{3,}$/g)){
      
      this.response.status = 400;
      this.body = {message: "Bad Request"};
   } else {
      //Gets us the index of movie with given id.
      var updateIndex = movies.map(function(movie){
         return movie.id;
      }).indexOf(parseInt(this.params.id));
      
      if(updateIndex === -1){
         //Movie not found, create new movies.push({
            id: this.params.id,
            name: this.request.body.name,
            year: this.request.body.year,
            rating: this.request.body.rating
         });
         this.body = {message: "New movie created.", location: "/movies/" + this.params.id};    
      } else {
         //Update existing movie
         movies[updateIndex] = {
            id: this.params.id,
            name: this.request.body.name,
            year: this.request.body.year,
            rating: this.request.body.rating
         };
         this.body = {message: "Movie id " + this.params.id + " updated.", location: "/movies/" + this.params.id};
      }
   }
}

Cette route fera la fonction que nous avons spécifiée dans le tableau ci-dessus. Il mettra à jour l'objet avec de nouveaux détails s'il existe. S'il n'existe pas, il créera un nouvel objet. Pour tester cette route, utilisez la commande curl suivante. Cela mettra à jour un film existant. Pour créer un nouveau film, changez simplement l'identifiant en un identifiant non existant.

curl -X PUT --data "name = Toy%20story&year = 1995&rating = 8.5" 
https://localhost:3000/movies/101

Réponse

{"message":"Movie id 101 updated.","location":"/movies/101"}

SUPPRIMER l'itinéraire

Utilisez le code suivant pour créer une route de suppression.

router.delete('/:id', deleteMovieWithId);

function *deleteMovieWithId(next){
   var removeIndex = movies.map(function(movie){
      return movie.id;
   }).indexOf(this.params.id); //Gets us the index of movie with given id.
   
   if(removeIndex === -1){
      this.body = {message: "Not found"};
   } else {
      movies.splice(removeIndex, 1);
      this.body = {message: "Movie id " + this.params.id + " removed."};
   }
}

Testez l'itinéraire de la même manière que nous l'avons fait pour les autres. En cas de suppression réussie (par exemple, l'ID 105), vous obtiendrez -

{message: "Movie id 105 removed."}

Enfin, notre fichier movies.js ressemble à -

var Router = require('koa-router');
var router = Router({
   prefix: '/movies'
});  //Prefixed all routes with /movies
var movies = [
   {id: 101, name: "Fight Club", year: 1999, rating: 8.1},
   {id: 102, name: "Inception", year: 2010, rating: 8.7},
   {id: 103, name: "The Dark Knight", year: 2008, rating: 9},
   {id: 104, name: "12 Angry Men", year: 1957, rating: 8.9}
];

//Routes will go here
router.get('/', sendMovies);
router.get('/:id([0-9]{3,})', sendMovieWithId);
router.post('/', addNewMovie);
router.put('/:id', updateMovieWithId);
router.delete('/:id', deleteMovieWithId);

function *deleteMovieWithId(next){
   var removeIndex = movies.map(function(movie){
      return movie.id;
   }).indexOf(this.params.id); //Gets us the index of movie with given id.
   
   if(removeIndex === -1){
      this.body = {message: "Not found"};
   } else {
      movies.splice(removeIndex, 1);
      this.body = {message: "Movie id " + this.params.id + " removed."};
   }
}

function *updateMovieWithId(next) {
   //Check if all fields are provided and are valid:
   if(!this.request.body.name ||
      !this.request.body.year.toString().match(/^[0-9]{4}$/g) ||
      !this.request.body.rating.toString().match(/^[0-9]\.[0-9]$/g) ||
      !this.params.id.toString().match(/^[0-9]{3,}$/g)){
      
      this.response.status = 400;
      this.body = {message: "Bad Request"};
   } else {
      //Gets us the index of movie with given id.
      var updateIndex = movies.map(function(movie){
         return movie.id;
      }).indexOf(parseInt(this.params.id));
      
      if(updateIndex === -1){
         //Movie not found, create new
         movies.push({
            id: this.params.id,
            name: this.request.body.name,
            year: this.request.body.year,
            rating: this.request.body.rating
         });
         this.body = {message: "New movie created.", location: "/movies/" + this.params.id};
      } else {
         //Update existing movie
            movies[updateIndex] = {
            id: this.params.id,
            name: this.request.body.name,
            year: this.request.body.year,
            rating: this.request.body.rating
         };
         this.body = {message: "Movie id " + this.params.id + " updated.", 
            location: "/movies/" + this.params.id};
      }
   }
}

function *addNewMovie(next){
   //Check if all fields are provided and are valid:
   if(!this.request.body.name ||
      !this.request.body.year.toString().match(/^[0-9]{4}$/g) ||
      !this.request.body.rating.toString().match(/^[0-9]\.[0-9]$/g)){
      
      this.response.status = 400;
      this.body = {message: "Bad Request"};
   } else {
      var newId = movies[movies.length-1].id+1;
      
      movies.push({
         id: newId,
         name: this.request.body.name,
         year: this.request.body.year,
         rating: this.request.body.rating
      });
      this.body = {message: "New movie created.", location: "/movies/" + newId};
   }
   yield next;
}
function *sendMovies(next){
   this.body = movies;
   yield next;
}
function *sendMovieWithId(next){
   var ctx = this
   
   var currMovie = movies.filter(function(movie){
      if(movie.id == ctx.params.id){
         return true;
      }
   });
   if(currMovie.length == 1){
      this.body = currMovie[0];
   } else {
      this.response.status = 404;//Set status to 404 as movie was not found
      this.body = {message: "Not Found"};
   }
   yield next;
}
module.exports = router;

Ceci complète notre API REST. Vous pouvez désormais créer des applications beaucoup plus complexes en utilisant ce style architectural simple et Koa.

La journalisation est très utile lors de la création d'applications Web car elles nous indiquent exactement où les choses se sont mal passées. Nous obtenons également le contexte des choses qui ont mal tourné et pouvons trouver des solutions possibles pour les mêmes.

Pour activer la journalisation dans Koa, nous avons besoin du middleware, koa-logger. Installez-le à l'aide de la commande suivante.

$ npm install --save-dev koa-logger

Maintenant, dans votre application, ajoutez le code suivant pour activer la journalisation.

var logger = require('koa-logger')
var koa = require('koa')

var app = koa()
app.use(logger())

app.use(function*(){
   this.body = "Hello Logger";
})

app.listen(3000)

Exécutez ce serveur et visitez n'importe quelle route sur le serveur. Vous devriez voir les journaux comme -

Maintenant, si vous obtenez une erreur sur une route ou une demande spécifique, ces journaux devraient vous aider à comprendre ce qui n'a pas fonctionné dans chacun d'eux.

L'échafaudage nous permet de créer facilement un skeleton for a web application. Nous avons créé manuellement notre répertoire public, ajouté un middleware, créé des fichiers de routage séparés, etc. Un outil d'échafaudage configure toutes ces choses pour nous afin que nous puissions directement commencer à construire notre application.

L'échafaudeur que nous utiliserons s'appelle Yeoman. C'est un outil d'échafaudage construit pour Node.js mais qui possède également des générateurs pour plusieurs autres frameworks (tels que flask, rails, django, etc.). Pour installer yeoman, entrez la commande suivante dans votre terminal.

$ npm install -g yeoman

Yeoman utilise des générateurs pour échafauder les applications. Pour découvrir les générateurs disponibles sur npm à utiliser avec yeoman, rendez-vous ici . Pour les besoins de ce didacticiel, nous utiliserons le 'générateur-koa'. Pour installer ce générateur, entrez la commande suivante dans votre terminal.

$ npm install -g generator-koa

Pour utiliser ce générateur, entrez -

yo koa

Ensuite, il créera une structure de répertoires et créera les fichiers suivants pour vous. Il installera également les modules npm et les composants bower nécessaires pour vous.

create package.json
create test/routeSpec.js
create views/layout.html
create views/list.html
create public/styles/main.css
create public/scripts/.gitkeep
create controllers/messages.js
create app.js
create .editorconfig
create .jshintrc

I'm all done. Running npm install & bower install for you to install 
the required dependencies. 
If this fails, try running the command yourself.

Ce générateur crée une structure très simple pour nous.

.
├── controllers
│   └── messages.js
├── public
|   ├── scripts
|   └── styles
|       └── main.css    
├── test
|   └── routeSpec.js
├── views
|   ├── layout.html
|   └── list.html
├── .editorconfig
├── .jshintrc
├── app.js
└── package.json

Explorez les nombreux générateurs disponibles pour Koa et choisissez celui qui vous convient le mieux. Les étapes pour travailler avec tous les générateurs sont les mêmes. Vous devrez installer un générateur, l'exécuter à l'aide de yeoman, il vous posera quelques questions, puis créera un squelette pour votre application en fonction de vos réponses.

Voici une liste des ressources que nous avons utilisées lors du développement de ce tutoriel -

  • Koajs.com

  • Koajs - Exemples Une liste d'exemples créés par la communauté

  • Liste des officiels et 3 ème intergiciels du parti.

  • Une API CRUD utilisant koa.js - Un court screencast qui passe par la création d'une API CRUD dans Koa.js

  • Screencast de démarrage rapide Koa.js

  • Introduction à Koa.js et générateurs