Koa.js - API RESTful

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.