1. Objectif
Créer une Web API Rest permettant de créer, modifier, supprimer et consulter des données en JSON. Ce type de service s’inscrit totalement dans les architectures SOA qui ont la vertu d’être évolutive (scalable). En effet, imaginez une application composé d’une multitude de micro service, il est alors possible de les héberger sur différentes machines, voir même des fermes de machines.
Les Web API sont également par définition simple. Le fait qu’elle soit en REST (Representational State Transfer), contrairement à SOAP, signifie qu’elles reposent sur une couche protocole simple (rien d’autre qu’HTTP), qu’elles ne maintiennent pas d’états, son facilement mises en cache, et communique des informations généralement via Json qui est devenu un standard dans la communication Machine – Machine, ayant supplanté XML qui lui était plus verbeux (et donc d’avantage destiné à une compréhension humaine).
Les Web API sont par excellence les composants simples d’une application légère type smartphone ou site web mono page sollicitant différents composants pour établir son contenu.
Au final, nous ferons en sorte que notre Web Api soit hébergé de façon pérenne sur notre Raspberry Pi.
2. Pré requis
- Avoir configuré correctement son environnement de travail que ce soit sur son ordinateur ainsi que sur le raspberry (Accès SSH, VNC, …).
- Connaitre les bases du C# et de la programmation ASP.net MVC.
- Avoir réalisé le tutoriel Créer un site Web .Net Core Razor MVC hébergé sur Raspbian. Cela permettra de connaitre les commandes pour créer la base, … Nous n’y reviendrons pas ici.
3. Création de la Web Api
Quitte à prendre un exemple, autant que ce soit un exemple qui pourra servir. J’ai pour projet de réaliser une solution domotique. Cette solution comprendra un serveur Raspberry Pi ainsi que des composants qui seront connectés via différents protocoles de communications. Réalisons donc une web Api permettant de gérer les éléments liés au composants et à leur connexion au réseau.
A. Données manipulées
Voici une représentation synthétique des données manipulées :
Component: Composant caractérisé par
- Id = Un identifiant
- Name = Nom du composant
- NetworkId = Son identifiant sur le réseau qui sera attribué par le serveur.
- IsConnected = Booléen permettant de savoir s’il est connecté au serveur.
- Type = Type de composant
ComponentType:
- Code = Un identifiant
- Name = Nom du type
- Description = Description du type
Nous y retrouverons les Actionneurs, les capteurs ou encore les transducteurs.
NetworkComType:
- Id = Identifiant du type de communication
- Name = Nom du type de communication
Nous y retrouverons WiredI2c, WiredSerial, Wifi, Bluetooth, WaveXbee, WaveZwave, WaveNrf24 et Wave433Mhz
ComponentNetwork:
- ComponentId : Identifiant du composant
- NetworkComTypeId : Identifiant du moyen de communication
- IsLeaf : S’agit il d’une feuille ou d’un noeud du moyen de communication ?
B. Création de la WebApi et la partie Modèle
Lancez visual studio et créez un nouveau projet. Il s’agit qu’une application Web ASP.NET Core que nous nommerons WebApiComponent.
Dans le second écran, choisissez Api Web. L’application ainsi créé est rudimentaire.
1. Petite explication sur le contenu de cette WebApi
Vous pouvez y voir dans startup.cs qu’il s’agit d’une application MVC. Il existe par défaut un répertoire controllers contenant un contrôleur de base.
Par convention un contrôleur possède un nom ayant pour suffix Controller. Le nom par défaut correspond au chemin dans l’url.
Vous noterez la présence dans le contrôleur ValuesController d’attributs (Ce sont des ‘directives’ en entête de class, de méthodes, de propriétés au même titre que celle que nous avons positionné pour le modèle).
Ces attributs permettent de définir ici les liens avec les requêtes HTTP : Chemin et méthodes de requêtes type GET, PUT, … Vous voyez devant le nom de class du contrôleur la route [Route(« api/[controller] »)] qui permet de définir le chemin d’accès HTTP à notre contrôleur ainsi que les différentes méthodes avec des HttpGet, … Nous y reviendrons plus tard.
Il nous faut maintenant créer nos entités dans une partie modèle. Par convention et définition, une entité est une classe ne possédant que des propriétés avec des Getteur et Setteur. Elles sont une représentation objet de nos données en base. C’est Entity Framework qui fera le travail d’alimenter un context (qui définie plusieurs collections d’entités (voir sa définition plus loin dans le tuto) avec les éléments en base. Il se chargera de toutes les requêtes SQL).
Nous allons donc maintenant rajouter notre modèle.
2. Création du modèle
Nous souhaitons pour nos différentes entités ajouter une date de création et de modification. Ces champs seront implémenté dans les class entité sur la base d’une interface. Il est possible avec Entity Framework de réaliser un héritage de class, mais cela nécessite que la clé soit présente dans la class mère. Dans notre cas, nous avons une entité ayant pour clé un code sur trois lettre. Nous n’utiliserons donc qu’une Interface.
a. Création des entités
Faite un clic droit sur le projet>Ajouter>répertoire>nommé le Models (qui par convention représente le répertoire contenant les données).
Y créer l’interface suivante :
[pastacode lang= »cpp » manual= »using%20System%3B%0A%0Anamespace%20WebApiComponent.Models%0A%7B%0A%20%20%20%20public%20interface%20IEntity%0A%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20DateTime%20Created%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20DateTime%20LastModified%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%7D%0A%7D » message= »IEntity.cs » highlight= » » provider= »manual »/]
Créez maintenant les class suivantes :
[pastacode lang= »cpp » manual= »using%20System%3B%0Ausing%20System.Collections.Generic%3B%0Ausing%20System.ComponentModel.DataAnnotations%3B%0Ausing%20System.ComponentModel.DataAnnotations.Schema%3B%0A%0Anamespace%20WebApiComponent.Models%0A%7B%0A%20%20%20%20public%20class%20Component%20%3A%20IEntity%0A%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20public%20uint%20Id%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20string%20Name%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20ushort%3F%20NetworkId%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20bool%20IsConnected%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20string%20ComponentTypeCode%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20%5BForeignKey(%22ComponentTypeCode%22)%5D%0A%20%20%20%20%20%20%20%20public%20ComponentType%20Type%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20ICollection%3CComponentNetwork%3E%20ComponentNetworkList%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20public%20DateTime%20Created%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20public%20DateTime%20LastModified%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%7D%0A%7D%0A » message= »Component.cs » highlight= » » provider= »manual »/]
Vous pouvez voir que dans cette entité, on retrouve des clés étrangères mais également d’autres entités, soit simplement une entité comme ComponentType ou un collection pour ComponentNetworkList.
Les clés permettent de définir les clés étrangères en base. Les Entités permettent de charger dans l’entité des informations connexes. Nous verrons comment plus tard.
Vous voyez également la présence d’attributs, ici des annotations (documentation). On souligne ici qu’un champ est requis mais surtout que pour une entité la clé étrangère et l’attribut défini juste avant.
[pastacode lang= »cpp » manual= »using%20System%3B%0Ausing%20System.ComponentModel.DataAnnotations%3B%0Ausing%20System.ComponentModel.DataAnnotations.Schema%3B%0A%0Anamespace%20WebApiComponent.Models%0A%7B%0A%20%20%20%20public%20class%20ComponentNetwork%20%3A%20IEntity%0A%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20uint%20ComponentId%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20%5BForeignKey(%22ComponentId%22)%5D%0A%20%20%20%20%20%20%20%20public%20Component%20Component%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20int%20NetworkComTypeId%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20%5BForeignKey(%22NetworkComTypeId%22)%5D%0A%20%20%20%20%20%20%20%20public%20NetworkComType%20CommunicationType%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20bool%20isLeaf%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20DateTime%20Created%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20public%20DateTime%20LastModified%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%7D%0A%7D%0A » message= »ComponentNetwork.cs » highlight= » » provider= »manual »/]
[pastacode lang= »cpp » manual= »using%20System%3B%0Ausing%20System.ComponentModel.DataAnnotations%3B%0A%0Anamespace%20WebApiComponent.Models%0A%7B%0A%20%20%20%20public%20class%20ComponentType%20%3A%20IEntity%0A%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%5BKey%2C%20MinLength(3)%2C%20MaxLength(3)%5D%0A%20%20%20%20%20%20%20%20public%20string%20Code%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20string%20Name%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20string%20Description%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20DateTime%20Created%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20public%20DateTime%20LastModified%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%7D%0A%7D%0A » message= »ComponentType.cs » highlight= » » provider= »manual »/]
[pastacode lang= »cpp » manual= »using%20System%3B%0Ausing%20System.Collections.Generic%3B%0Ausing%20System.ComponentModel.DataAnnotations%3B%0A%0Anamespace%20WebApiComponent.Models%0A%7B%0A%20%20%20%20public%20class%20NetworkComType%20%3A%20IEntity%0A%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20public%20int%20Id%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20%5BRequired%5D%0A%20%20%20%20%20%20%20%20public%20string%20Name%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20ICollection%3CComponentNetwork%3E%20ComponentNetworkList%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20DateTime%20Created%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%20public%20DateTime%20LastModified%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%7D%0A%7D%0A » message= »NetworkComType.cs » highlight= » » provider= »manual »/]
Voilà pour les entités représentant les données manipulées. Passons maintenant au context Entity Framework
b. Création du context Entity Framework
Le context Entity Framework va permettre de définir la liste des collections d’entités mappées avec la base de données. Il s’agit d’un objet qui hérite de DbContext. On y retrouvera également dans une méthode OnModelCreating, l’appel à des méthodes permettant de compléter les annotations précédemment mises sur nos entités. En effet, les attributs on l’avantage de donner de la visibilité mais ne permettent pas de tout définir. Il est alors possible de définir les directive de création de base de données via ce que l’on appel Code Fluent (Attention, on retrouve les mêmes concepts que pour les attributs, il faut donc veiller à ne pas être contradictoire car il s’agit en fait de la même chose : Paramétrer la création de la base).
Ici nous définirons la création de valeurs par défaut mais également les liens entre entités ou encore le fait qu’un identifiant est en auto incrément. Pour donner la lisibilité au code, je regroupe mes traitements par entité dans des régions.
[pastacode lang= »cpp » manual= »using%20Microsoft.EntityFrameworkCore%3B%0A%0Anamespace%20WebApiComponent.Models%0A%7B%0A%20%20%20%20public%20class%20ApiDataContext%20%3A%20DbContext%0A%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20public%20ApiDataContext(DbContextOptions%3CApiDataContext%3E%20options)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%3A%20base(options)%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20DbSet%3CComponent%3E%20Components%20%7B%20get%3B%20set%3B%20%7D%0A%20%20%20%20%20%20%20%0A%20%20%20%20%20%20%20%20public%20DbSet%3CComponentType%3E%20ComponentTypes%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20DbSet%3CNetworkComType%3E%20NetworkCommunicationTypes%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20DbSet%3CComponentNetwork%3E%20ComponentsInNetwork%20%7B%20get%3B%20set%3B%20%7D%0A%0A%20%20%20%20%20%20%20%20protected%20override%20void%20OnModelCreating(ModelBuilder%20p_modelBuilder)%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%23region%20Component%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2F%20Autoincr%C3%A9ment%20pour%20la%20cl%C3%A9%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponent%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(f%20%3D%3E%20f.Id)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.ValueGeneratedOnAdd()%3B%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2F%20Valeurs%20par%20d%C3%A9faut%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponent%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(e%20%3D%3E%20e.Created)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasDefaultValueSql(%22now()%22)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponent%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(e%20%3D%3E%20e.LastModified)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasDefaultValueSql(%22now()%22)%3B%20%20%20%20%20%20%20%20%20%20%20%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20%23endregion%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20%23region%20ComponentType%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2F%20Valeurs%20par%20d%C3%A9faut%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponentType%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(e%20%3D%3E%20e.Created)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasDefaultValueSql(%22now()%22)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponentType%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(e%20%3D%3E%20e.LastModified)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasDefaultValueSql(%22now()%22)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%23endregion%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20%23region%20NetworkComType%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2F%20Autoincr%C3%A9ment%20pour%20la%20cl%C3%A9%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CNetworkComType%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(f%20%3D%3E%20f.Id)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.ValueGeneratedOnAdd()%3B%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2F%20Valeurs%20par%20d%C3%A9faut%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CNetworkComType%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(e%20%3D%3E%20e.Created)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasDefaultValueSql(%22now()%22)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CNetworkComType%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(e%20%3D%3E%20e.LastModified)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasDefaultValueSql(%22now()%22)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%23endregion%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20%23region%20ComponentNetwork%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponentNetwork%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasKey(keys%20%3D%3E%20new%20%7B%20keys.ComponentId%2C%20keys.NetworkComTypeId%20%7D)%3B%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponentNetwork%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasOne(cn%20%3D%3E%20cn.Component)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.WithMany(c%20%3D%3E%20c.ComponentNetworkList)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasForeignKey(cn%20%3D%3E%20cn.ComponentId)%3B%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponentNetwork%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasOne(cn%20%3D%3E%20cn.CommunicationType)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.WithMany(n%3D%3E%20n.ComponentNetworkList)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasForeignKey(cn%20%3D%3E%20cn.NetworkComTypeId)%3B%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2F%20Valeurs%20par%20d%C3%A9faut%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponentNetwork%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(e%20%3D%3E%20e.Created)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasDefaultValueSql(%22now()%22)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20p_modelBuilder.Entity%3CComponentNetwork%3E()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Property(e%20%3D%3E%20e.LastModified)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.HasDefaultValueSql(%22now()%22)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%23endregion%0A%0A%0A%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%7D%0A%7D%0A » message= »ApiDataContext.cs » highlight= » » provider= »manual »/]
Je vais ici expliquer une chose qui n’est peut être pas évidente à comprendre à la lecture du code. Notre table ComponentNetwork sera une table d’association entre Component et NetworkComType. Cela veut donc dire qu’il y a deux clés étrangères vers chacune de ces tables. On y retrouve également les objets Component et NetworkComType. Dans ces deux entités, on retrouve par conséquent une collection de ComponentNetwork puisque nous sommes dans une association n-ère.
C’est ce que décrit le code suivant :
p_modelBuilder.Entity<ComponentNetwork>() .HasKey(keys => new { keys.ComponentId, keys.NetworkComTypeId }); p_modelBuilder.Entity<ComponentNetwork>() .HasOne(cn => cn.Component) .WithMany(c => c.ComponentNetworkList) .HasForeignKey(cn => cn.ComponentId); p_modelBuilder.Entity<ComponentNetwork>() .HasOne(cn => cn.CommunicationType) .WithMany(n=> n.ComponentNetworkList) .HasForeignKey(cn => cn.NetworkComTypeId);
La clé de la table est double. Ensuite prenons l’exemple pour l’association avec le Component. On explique que pour 1 Component ici présent (HasOne dans la table ComponentNetwork), il y a une collection (WithMany) de ComponentNetwork liés par la clé ComponentId.
3. Création de la base de données
Nous utilisons une base de données MariaDb sur le Raspberry. Nous devons donc créer la base, configurer les droits et rajouter à notre projet le package Nuget permettant de s’interfacer avec les bases de données MariaDb.
Créez la base de données via Workbench ou via S.S.H. (voir tuto cité au besoin).
Ensuite installez le package Pomelo Entity Framework MySql et Design.
Dans le projet, ouvrer le fichier appsettings.json et rajoutez la chaine de connection tout en remplaçant les valeurs.
[pastacode lang= »javascript » manual= »%7B%0A%20%20%22Logging%22%3A%20%7B%0A%20%20%20%20%22IncludeScopes%22%3A%20false%2C%0A%20%20%20%20%22Debug%22%3A%20%7B%0A%20%20%20%20%20%20%22LogLevel%22%3A%20%7B%0A%20%20%20%20%20%20%20%20%22Default%22%3A%20%22Warning%22%0A%20%20%20%20%20%20%7D%0A%20%20%20%20%7D%2C%0A%20%20%20%20%22Console%22%3A%20%7B%0A%20%20%20%20%20%20%22LogLevel%22%3A%20%7B%0A%20%20%20%20%20%20%20%20%22Default%22%3A%20%22Warning%22%0A%20%20%20%20%20%20%7D%0A%20%20%20%20%7D%0A%20%20%7D%2C%0A%20%20%22ConnectionStrings%22%3A%20%7B%0A%20%20%20%20%22DefaultConnection%22%3A%20%22Server%3DIpRasberry%3BDatabase%3DNomMaBase%3BUid%3DNomUser%3BPwd%3DMotDePasse%3B%22%0A%20%20%7D%0A%7D%0A » message= »appsettings.json » highlight= » » provider= »manual »/]
Dans le fichier Startup.cs, nous devons maintenant remplacer l’appel au DbContext comme ceci :
[pastacode lang= »cpp » manual= »public%20void%20ConfigureServices(IServiceCollection%20services)%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20services.AddMvc()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20services.AddDbContext%3CApiDataContext%3E(options%20%3D%3E%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20options.UseMySql(Configuration.GetConnectionString(%22DefaultConnection%22)))%3B%0A%20%20%20%20%20%20%20%20%7D » message= » » highlight= » » provider= »manual »/]
Voilà, maintenant notre projet est prêt. Reste à créer le modèle de données. Rien de plus simple avec Entity Framework code first !
Allez dans la console Nuget. On va y créer un point de migration. Comme c’est le premier, par convention, nous le nommons Initial.
Add-Migration Initial
on modifie maintenant la base de données
update-database
Et voilà, vous pouvez aller voir sur Workbench, le modèle a été créé.
C. Définition des méthodes du contrôleur
Nous allons maintenant générer les contrôleurs permettant de réaliser un C.R.U.D. (Create, Read, Update, Delete) sur chacune de nos entités.
Faite un clic droit sur le répertoire Controllers>Ajouter>Contrôleur.
On va choisir un contrôleur d’API avec Actions utilisant Entity Framework
Une seconde fenêtre apparaît ou il faut choisir le modèle que l’on souhaite mapper ainsi que le contexte Entity Framework. Ainsi, le lien sera fait entre le contrôleur qui implémente un CRUD et la base de données via le contexte Entity Framework pour alimenter notre Entité dans le modèle.
Allez voir le contrôleur ainsi créé. Vous notez la présence de la définition de notre context Entity Framework en tant que membre de la class ainsi que les attributs permettant de définir les routes Http.
Il y a également [Produces(« application/json »)] qui stipule que les données échangés seront de type Json (content-type dans les pages web). De base, nous possédons les éléments suivants pour réaliser notre CRUD :
- Un Post permettant d’ajouter un nouvel élément (CREATE)
- Deux Get, un sans valeur pour obtenir la liste des éléments, ainsi qu’un permettant de stipuler l’identifiant (READ)
- Un Put, permettant de réaliser des modifications (UPDATE)
- Un Delete permettant de réaliser des suppressions (DELETE)
De la même façon, rajoutez maintenant les contrôleurs pour les autres modèles implémentés. C’est presque magique, notre WebApi est ainsi prête et nous pouvons maintenant la lancer.
Mais comment la tester ! Et bien nous allons utiliser un utilitaire bien pratique.
Lorsque le navigateur sera chargé. Copiez l’url.
D. Test et utilisation de Postman
Notre Web Api utilise des méthodes de requêtes pour interagir avec la base de données via Entity Framework. Vous pouvez voir ces méthodes dans les contrôleurs comme nous l’avons déjà évoqué. On y retrouve GET, POST, PUT et DELETE.
Postman est un outil développé par Google qui permet d’interagir facilement avec notre API. Il possède de nombreuses fonctionnalités, le tout avec une interface graphique sympathique. Il existe sous forme d’application ou encore de plugin sous Chrome.
Installez Postman qui vous trouverez ici, puis lancez le.
Fermez la première fenêtre puis collez dans l’url le chemin de votre WebApi.
Rajoutez /api/ComponentTypes. A gauche, choisissez POST.
Cliquez sur Body, raw et choisissez JSON (et non le text par défaut) comme type de données.
Dans la fenêtre au dessous tapez :
[pastacode lang= »javascript » manual= »%7B%0A%22Code%22%3A%22CPT%22%2C%0A%22Name%22%3A%22Capteurs%22%2C%0A%22Description%22%3A%22Dispositif%20transformant%20l’%C3%A9tat%20d’une%20grandeur%20physique%20observ%C3%A9e%20en%20une%20grandeur%20utilisable%2C%20telle%20qu’une%20tension%20%C3%A9lectrique%2C%20une%20hauteur%20de%20mercure%2C%20une%20intensit%C3%A9%20ou%20la%20d%C3%A9viation%20d’une%20aiguille.%20Exemple%20%3A%20Sonde%20de%20temp%C3%A9rature%2C%20de%20pression%2C%20capteur%20de%20pr%C3%A9sence.%22%0A%7D » message= » » highlight= » » provider= »manual »/]
Appuyez sur send. Vous pouvez observer la réponse qui comporte en plus nos champs par défaut de création et de modification.
Rajoutez de même pour les autres contrôleurs des élements.
api/NetworkComTypes
[pastacode lang= »javascript » manual= »%7B%0A%22Name%22%3A%22WiredI2c%22%0A%7D » message= » » highlight= » » provider= »manual »/]
Vous pouvez ici en rajouter un second, il nous servira.
Pour api/Components
[pastacode lang= »javascript » manual= »%7B%0A%22Name%22%3A%22Thermom%C3%A8tre%20ext%C3%A9rieur%22%2C%0A%22IsConnected%22%3Afalse%2C%0A%22ComponentTypeCode%22%3A%22CPT%22%0A%7D » message= » » highlight= » » provider= »manual »/]
et enfin pour api/ComponentNetworks les deux post suivant (a séparer lors de l’appel)
[pastacode lang= »javascript » manual= »%7B%0A%20%20%20%20%20%20%20%20%22componentId%22%3A%201%2C%0A%20%20%20%20%20%20%20%20%22networkComTypeId%22%3A%201%2C%0A%20%20%20%20%20%20%20%20%22isLeaf%22%3A%20false%0A%20%20%20%20%7D%0A%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%22componentId%22%3A%201%2C%0A%20%20%20%20%20%20%20%20%22networkComTypeId%22%3A%207%2C%0A%20%20%20%20%20%20%20%20%22isLeaf%22%3A%20true%0A%20%20%20%20%7D » message= » » highlight= » » provider= »manual »/]
Maintenant, passez sur la commande GET des Components. Appuyez sur Send.
On remarque que le type et le ComponentNetworkList sont null. Il s’agit d’entités plus complexes et notamment lié à des clés étrangères. Pour pouvoir avoir les Json correspondant, il faut indiquer au context Entity Framework de ramener des données connexe (infos ici). Nous allons voir comment faire ceci.
Vous pouvez arrêter votre application.
E. Chargement des données connexes
Nous allons donc devoir indiquer au context Entity Framework qu’il faut charger les données connexes. Il faut noter plusieurs choses importantes. Certes, il est confortable de pouvoir charger des entités, dans des entités mais ceci s’avère coûteux en charge. Il faut le garder à l’esprit.
Il faut également limiter le niveau de chargement car dans notre cas, nous pouvons observer que si nous chargeons la liste des ComponentNetworks, ils peuvent contenir des Component qui eux mêmes comportent des ComponentNetworksList et ainsi suite. Ce qui reviendrait à charger en boucle les données.
Nous allons donc dans les contrôleurs inclure 1 voir 2 niveaux de lecture des données connexes.
Avant cela, nous devons préciser au chargement MVC une option Json concernant la sérialisation des données.
Chargez startup.cs, dans le chargement du service MVC, rajoutez l’option suivante.
[pastacode lang= »cpp » manual= »services.AddMvc()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.AddJsonOptions(options%20%3D%3E%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20options.SerializerSettings.ReferenceLoopHandling%20%3D%20Newtonsoft.Json.ReferenceLoopHandling.Ignore)%3B » message= »startup.cs » highlight= » » provider= »manual »/]
Nous allons maintenant à titre d’exemple, charger les données connexes dans le contrôleur Component.
Ouvrez ComponentsController.cs.
Nous allons modifier les deux Get.
Pour le GetComponents, nous allons remplacer le return _context.Components; par ce qui suit :
[pastacode lang= »cpp » manual= »%2F%2F%20GET%3A%20api%2FComponents%0A%20%20%20%20%20%20%20%20%5BHttpGet%5D%0A%20%20%20%20%20%20%20%20public%20IEnumerable%3CComponent%3E%20GetComponents()%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20DbSet%3CComponent%3E%20v_set%20%3D%20_context.Components%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2FChargement%20des%20donn%C3%A9es%20connexes%0A%20%20%20%20%20%20%20%20%20%20%20%20v_set.Include(c%20%3D%3E%20c.Type).Load()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20v_set.Include(c%20%3D%3E%20c.ComponentNetworkList).ToList()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20return%20v_set%3B%0A%20%20%20%20%20%20%20%20%7D » message= » » highlight= » » provider= »manual »/]
Vous observez que l’on peut sur une même commande d’ensemble enchaîner les appels Include. Vous noterez que sur la première, j’ai réalisé un load et sur la seconde un ToList. Cela revient au même. Mais il est important de noter que pour les méthodes asynchrones, il exsite un ToListAsync ce qui n’est pas le cas pour le load.
Relancez votre application et voyons le résultat d’un Get dans Postman :
C’est pas mal du tout, nous avons maintenant le type renseigné et le tableau de ComponentNetwork. Nous observons néanmoins qu’il manque le communicationType à ce niveau.
On pourrait essayer à ce niveau d’utiliser un ThenInclude fourni par Entity Framework mais nous obtiendrions alors une erreur.
L’astuce consiste alors à passer par une requête Linq qui elle est plus permissive. Modifiez donc la méthode comme ceci :
[pastacode lang= »cpp » manual= »%5BHttpGet%5D%0A%20%20%20%20%20%20%20%20public%20IEnumerable%3CComponent%3E%20GetComponents()%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20DbSet%3CComponent%3E%20v_set%20%3D%20_context.Components%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2FChargement%20des%20donn%C3%A9es%20connexes%0A%20%20%20%20%20%20%20%20%20%20%20%20v_set.Include(c%20%3D%3E%20c.Type).Load()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20var%20v_query%20%3D%20v_set.Include(c%20%3D%3E%20c.ComponentNetworkList).AsQueryable()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20v_query.SelectMany(c%20%3D%3E%20c.ComponentNetworkList)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Include(w%20%3D%3E%20w.CommunicationType)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.ToList()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20return%20v_query%3B%0A%20%20%20%20%20%20%20%20%7D » message= » » highlight= » » provider= »manual »/]
Vous pouvez maintenant relancer l’application et constater le résultat dans Postman :
C’est parfait. Nous avons maintenant le niveau souhaité d’information.
Passons maintenant à la méthode Get d’un élément. Malheureusement, nous allons y perdre son côté asynchrone. Il nous faut la transformer ainsi :
[pastacode lang= »cpp » manual= »%2F%2F%20GET%3A%20api%2FComponents%2F5%0A%20%20%20%20%20%20%20%20%5BHttpGet(%22%7Bid%7D%22)%5D%0A%20%20%20%20%20%20%20%20public%20IActionResult%20GetComponent(%5BFromRoute%5D%20uint%20id)%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20if%20(!ModelState.IsValid)%0A%20%20%20%20%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20return%20BadRequest(ModelState)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%7D%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20DbSet%3CComponent%3E%20v_set%20%3D%20_context.Components%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%2F%2FChargement%20des%20donn%C3%A9es%20connexes%0A%20%20%20%20%20%20%20%20%20%20%20%20v_set.Include(c%20%3D%3E%20c.Type).Load()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20var%20v_query%20%3D%20v_set.Include(c%20%3D%3E%20c.ComponentNetworkList).AsQueryable()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20v_query.SelectMany(c%20%3D%3E%20c.ComponentNetworkList)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Include(w%20%3D%3E%20w.CommunicationType)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.ToList()%3B%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20var%20v_component%20%3D%20v_query.SingleOrDefaultAsync(m%20%3D%3E%20m.Id%20%3D%3D%20id).Result%3B%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20if%20(v_component%20%3D%3D%20null)%0A%20%20%20%20%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20return%20NotFound()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%7D%0A%0A%20%20%20%20%20%20%20%20%20%20%20%20return%20Ok(v_component)%3B%0A%20%20%20%20%20%20%20%20%7D » message= » » highlight= » » provider= »manual »/]
Vous obtiendrez un résultat équivalent que pour la list précédente. Je vous laisse faire de même pour les autres contrôleurs.
Nous avons maintenant une WebApi fonctionnelle et complète. Reste à l’héberger sur notre raspberry de façon définitive.
F. Compilation pour le raspberry
Déchargez le projet et faite un clic droit sur le projet pour modifier le csproj.
Dans la partie PropertyGroup du XML rajoutez la balise RuntimeIdentifiers
[pastacode lang= »markup » manual= »%20%3CPropertyGroup%3E%0A%20%20%20%20%3CTargetFramework%3Enetcoreapp2.0%3C%2FTargetFramework%3E%0A%20%20%20%20%3CRuntimeIdentifiers%3Ewin-x64%3Bwin10-x64%3Blinux-arm%3C%2FRuntimeIdentifiers%3E%0A%20%20%3C%2FPropertyGroup%3E » message= » » highlight= » » provider= »manual »/]
Rechargez le projet. Allez dans Générer > Publier
Choisissez un répertoire sur votre Raspberry et créez un profil.
Dans les paramètres choisissez le mode release ainsi que linux-arm. (Pour plus de détail, voir le tutoriel cité en pré requis).
Aller hop, on publie notre WebApi sur le Raspberry Pi.
4. Installation sous un serveur Web
A. Nginx
Nginx est un serveur web sans interface. Il nous servira de reverse proxy apportant son lot de fonctionnalités utiles au bon fonctionnement de nos applications web (notamment du cache). Pour faire simple, il sert de point d’entrée aux requêtes Http qui seront faite sur notre Raspberry Pi.
Pourquoi Nginx alors qu’Apache est très connu sous linux ? Et bien parce que Nginx est un serveur dit asynchrone capable d’exécuter plusieurs tâches simultanément. Cela fait de lui un serveur web particulièrement performant.
Il répond plus vite, il permet d’avantage d’utilisateurs simultanés, et il consomme moins de mémoire vive. Bien évidemment sur un Raspberry, on cherche la performance. Il n’en faut donc pas plus pour justifier ce choix.
1. Installation
Sous une invite de commande tapez :
sudo apt-get install nginx
Nginx s’installe (validez la question). Une fois l’installation terminée, Il faut maintenant lancer le service :
sudo service nginx start
Configurons maintenant la bête :).
2. Configuration
Il nous faut maintenant configurer le serveur Nginx en tant que proxy inverse. Nous avons notamment remarqué que les sites avec Kestrel se lancent sur le port 5000. Nous allons router via Nginx les demandes du port 80 sur le port 5000.
sudo nano /etc/nginx/sites-available/default
Dans la partie server mettez les lignes suivantes, puis enregistrez et quittez nano.
[pastacode lang= »javascript » manual= »server%20%7B%0A%20%20%20%20listen%2080%3B%0A%20%20%20%20location%20%2F%20%7B%0A%20%20%20%20%20%20%20%20proxy_pass%20http%3A%2F%2Flocalhost%3A5000%3B%0A%20%20%20%20%20%20%20%20proxy_http_version%201.1%3B%0A%20%20%20%20%20%20%20%20proxy_set_header%20Upgrade%20%24http_upgrade%3B%0A%20%20%20%20%20%20%20%20proxy_set_header%20Connection%20keep-alive%3B%0A%20%20%20%20%20%20%20%20proxy_set_header%20Host%20%24http_host%3B%0A%20%20%20%20%20%20%20%20proxy_cache_bypass%20%24http_upgrade%3B%0A%20%20%20%20%7D%0A%7D » message= » » highlight= » » provider= »manual »/]
Ici, nous voyons directement les effets de Nginx qui sert de reverse proxy : D’un côté, il écoute sur le port 80 les requêtes entrantes puis établi la jonction avec en local avec les trames http du port 5000 émise par Kestrel de dotnet Core.
lancez la commande suivante pour vérifier que le fichier de configuration est ok:
sudo nginx -t
Et comme tout est ok, on peut recharger la configuration
sudo nginx -s reload
Nginx est désormais configuré pour transférer les requêtes faites à http://yourhost:80
à l’application ASP.NET Core s’exécutant sur Kestrel à l’adresse http://127.0.0.1:5000
a. Adaptation de l’application pour utiliser kestrel
Retournez sous Visual Studio et chargez le fichier program.cs. Nous allons rajouter l’appel à Kestrel :
[pastacode lang= »cpp » manual= »public%20class%20Program%0A%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20public%20static%20void%20Main(string%5B%5D%20args)%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20BuildWebHost(args).Run()%3B%0A%20%20%20%20%20%20%20%20%7D%0A%0A%20%20%20%20%20%20%20%20public%20static%20IWebHost%20BuildWebHost(string%5B%5D%20args)%20%3D%3E%0A%20%20%20%20%20%20%20%20%20%20%20%20WebHost.CreateDefaultBuilder(args)%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.UseKestrel()%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.UseStartup%3CStartup%3E()%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20.Build()%3B%0A%20%20%20%20%7D » message= » » highlight= » » provider= »manual »/]
Vous pouvez maintenant republier sur le raspberry.
Ensuite, allez dans le répertoire de votre application et lancez la :
sudo dotnet WebApiComponent.dll
Et maintenant, vous pouvez retourner sur Postman et remplacer localhost par l’IP de votre Raspberry. Et voilà, le tour est joué, Nginx nous sert maintenant de reverse proxy.
B. Création d’un service pour notre WebApi
Maintenant, il faut faire en sorte que notre application soit lancée automatique tel un service au démarrage de Raspbian afin d’avoir une WebApi perpétuellement accessible.
On va créer un répertoire d’hébergement dans /var/AspNetCore/WebApiComponent
puis on va recopier les fichiers compilés ici
cp -r /chemin/Partage/WebApi/* /var/AspNetCore/WebApiComponent
maintenant créons le fichier de service
sudo nano /etc/systemd/system/kestrel-WebApiComponent.service
[pastacode lang= »bash » manual= »%5BUnit%5D%0ADescription%3DWebApiComponent%0A%0A%5BService%5D%0AWorkingDirectory%3D%2Fvar%2Faspnetcore%2FWebApiComponent%0AExecStart%3D%2Fusr%2Flocal%2Fbin%2Fdotnet%20%2Fvar%2FAspNetCore%2FWebApiComponent%2FWebApiComponent.dll%0ARestart%3Dalways%0ARestartSec%3D10%20%20%23%20Restart%20service%20after%2010%20seconds%20if%20dotnet%20service%20crashes%0ASyslogIdentifier%3Ddotnet-example%0AUser%3Dwww-data%0AEnvironment%3DASPNETCORE_ENVIRONMENT%3DProduction%0AEnvironment%3DDOTNET_PRINT_TELEMETRY_MESSAGE%3Dfalse%0A%0A%5BInstall%5D%0AWantedBy%3Dmulti-user.target » message= » » highlight= » » provider= »manual »/]
Enregistrez et quittez.
Nous voyons que l’utilisateur par défaut qui lance le service est www-data. Assez commun pour les applications web.
Nous allons nous assurer de sa présence :
cat /etc/passwd | awk -F: '{print $ 1}'
Si ce n’est pas le cas, il faudra le créer mais normalement il devrait être présent. Après la copie de vos fichiers, ils doivent certainement posséder les droits suivant:
Dans le répertoire /var/AspNetCore/WebApiComponent faite un
ls -l
vos fichiers ont certainement les droits avec le propriétaire root
-rwxr--r--
Nous allons définir comme propriétaire www-data
sudo chown -R www-data *
On va rajouter le droit d’exécution sur le service :
sudo chmod +x kestrel-WebApiComponent.service
Vous pouvez par la suite affiner les droits à votre guise. Des informations complémentaires sont disponibles ici, notamment pour sécuriser d’avantage votre serveur.
On démarre le service :
sudo systemctl start kestrel-WebApiComponent.service
Et enfin on peut vérifier son statut :
sudo systemctl status kestrel-WebApiComponent.service
On active le service lors du démarrage du raspberry :
sudo systemctl enable kestrel-WebApiComponent.service
Et voilà le service est fonctionnel et maintenant notre Web Api fonctionne définitivement.
5. Conclusion
Nous avons vu qu’il était très simple à l’aide d’Entity Framework Core et du modèle MVC de générer des Api Rest permettant d’échanger des données qui seront consommé par un site web par exemple. Dans un prochain tutoriel, nous verrons comment héberger plusieurs Api et un site les consommant. Si vous avez des commentaires, n’hésitez pas.
Merci pour ce post , très intéressant pour bien comprendre les web api .