Développer un connecteur personnalisé pour Power Automate et Azure Logic App à base de déclencheurs (Triggers)
Cet article assume que vous connaissiez déjà les basiques de la création d'un connecteur personnalisé pour Logic App ou Power Automate.
Je ne vais donc pas y revenir, mais si tel n'est pas le cas, je vous laisse le soin d’aller voir la documentation en ligne.
Néanmoins, pour résumé, un connecteur
C'est un proxy ou un wrapper autour d’une API qui permet au service sous-jacent de communiquer avec Microsoft Power Automate, Microsoft Power Apps et Azure Logic Apps. Il offre aux utilisateurs un moyen de se connecter à leurs comptes et de tirer parti d’un ensemble d’actions et de déclencheurs prédéfinis pour créer leurs applications et leurs workflows.
Chaque connecteur offre un ensemble d’opérations classées comme « Actions » et « Déclencheurs ». Une fois que vous vous êtes connecté au service sous-jacent, ces opérations peuvent être facilement exploitées dans vos applications et vos workflows.
Type d'opération | Commentaires |
---|---|
Action | Par exemple, vous utiliseriez une action pour rechercher, écrire, mettre à jour ou supprimer des données dans une base de données. |
Déclencheurs Polling | Ces déclencheurs appellent votre service selon une fréquence spécifiée pour vérifier l’existence de nouvelles données. Lorsque de nouvelles données sont disponibles, cela entraîne une nouvelle exécution de votre instance de workflow avec les données en entrée |
Déclencheurs Webhook | Ces déclencheurs écoutent les données sur un point de terminaison, c'est-à-dire qu'ils attendent qu'un événement se produise. L'occurrence de cet événement provoque une nouvelle exécution de votre instance de workflow. |
Dans cet article, nous allons aborder la création d'un connecteur de type déclencheur de type Webhook
Note Un didacticiel, est disponible également dans la documentation Utiliser un webhook en tant que déclencheur pour Azure Logic Apps et Power Automate
Imaginons le scénario très simple suivant. Je suis directeur d’une grande épicerie en ligne, j'utilise votre plate-forme d'ecommerce, qui permet à mes fournisseurs d'ajouter de nouveaux produits dans mon catalogue.
J’aimerai pouvoir être notifié
-
Lorsqu'un fournisseur demande l'ajout d'un produit dans le catalogue, à des fins d'approbation de la demande
-
Lorsqu'un magasin réceptionne de nouvelles marchandises.
J'imagine sans aucun doute que vous avez déjà prévu le cas, et que votre plate-forme le gère correctement. Mais en tant que directeur, je suis souvent sur les routes, pas toujours connecté à mon PC et j'aimerai pouvoir être notifié par un autre moyen, par exemple sur mon mobile.
Si vous n'avez pas l'équivalent d'une application Mobile, vous pouvez vous appuyer sur la Power Platform de Microsoft.
En créant un connecteur personnalisé, vous allez ainsi ajouter de la souplesse à votre système, en permettant au directeur (ou à son IT) de créer un Workflow très simple, dés réception d’une marchandise, de notifier l’application Mobile Power Automate, comme illustré sur l’image suivante.
Lorsqu’un évènement se passe sur votre plate-forme, le déclencheur en informera le connecteur qui pourra alors initier un workflow Logic App ou Power Automate.
Les avantages sont nombreux du fait même qu’ils existent de nombreux autres connecteurs Logic App et Power Automate
-
Ils vous permettront de vous intégrer à moindre frais et facilement à la plate-forme Office 365, Azure, Dynamics 365, et autres SalesForce, Twitter, etc.. Liste des connecteurs, et ceci que les systèmes d'informations de vos clients soient dans le cloud ou à demeure.
-
Pas besoin de développer une solution d’intégration pour chaque système, les connecteurs disponibles sont fait pour ça.
-
Vous permettez à vos clients "non-développeur" de créer facilement leurs propres workflows d’intégration à l’aide de Power Automate ou Logic App.
-
Lors de l'enregistrement d'un Workflow, Logic App/Power Automate s'abonne au déclencheur, en lui passant une Url de rappel, un point de terminaison, dans le corps du message
Url https://prod-00.francecentral.logic.azure.com/workflows/0182d35837c1417d859da07f8752bc9d/triggers/Lorsque_qu'un_nouveau_produit_arrive_dans_le_magasin.... -
Le déclencheur sauvegarde dans son système cette Url pour une utilisation future.
-
Le déclencheur doit impérativement retourner à Logic App/Power Automate, une Url de suppression de l'abonnement.
-
Lorsqu'un évènement est déclenché, par exemple l'arrivée d'un produit en magasin, votre système initie le workflow à l'aide de l'Url de rappel.
L'intégration d'un connecteur personnalisé se fait par l'intermédiaire d'un fichier au format Json, qui suit la spécification OpenAPI version 2.0, plus connue sous le nom de spécification Swagger , standard qui permet de définir les interfaces RestFul.
Note: Logic App et Power Automate, ne supporte pas encore la version 3.0
Dans les lignes qui suivent nous allons donc voir comment :
- Créer le code du Webhook qui va gérer les abonnements
- Créer le fichier de définition OpenAPI
- Sécuriser le connecteur
Microsoft a étendu la définition OpenAPI pour ses propres besoins, afin de pouvoir l'intégrer à Logic App et à Power Automate.
Pour définir ce type de déclencheur, il faut rajouter la propriété "x-ms-trigger": "single" dans la définition OpenAPI, afin d'afficher l'opération en tant que déclencheur dans l'éditeur de connecteur personnalisé, comme illustré sur la figure suivante :
Note : Ne pas mettre la propriété "x-ms-trigger" défini l'opération comme étant une Action.
Extrait d'une définition au format OpenAPI d'une opération de type déclencheur
"/event/instore": {
"x-ms-notification-content": {
"description": "Arrivée de nouveaux produits",
"schema": {
"$ref": "#/definitions/InStore"
}
},
"post": {
"description": "Lorsque qu'un nouveau produit arrive dans le magasin (version d'évaluation)",
"summary": "Lorsque qu'un nouveau produit arrive dans le magasin (version d'évaluation)",
"operationId": "NewInstoreProduct",
"x-ms-trigger": "single",
"parameters": [
{
"in": "body",
"name": "Callback",
"schema": {
"$ref": "#/definitions/Callback"
}
}
],
"responses": {
"201": {
"description": "Created"
}
}
}
},
...
"definitions": {
"Callback": {
"type": "object",
"required": [ "Url" ],
"properties": {
"Url": {
"x-ms-notification-url": true,
"x-ms-visibility": "internal",
"required": [ "Url" ],
"description": "URL de rappel",
"title": "URL de rappel",
"type": "string"
}
}
},
"InStore": {
"type": "object",
"properties": {
"storeName": { "type": "string" },
"productName": {"type": "string"},
"quantity": {"format": "int32","type": "integer"}
}
},
...
Propriété | Définition |
---|---|
x-ms-notification-content | Schéma de la charge utile qui sera envoyée à Logic App/Power Automate, lors du déclenchement de l'événement. En d'autres termes lorsqu'un nouveau produit arrive en magasin, c'est le schéma contenant les informations sur le produit qui sera envoyé à l'Url de notification, afin de déclencher le Workflow. Par exemple si votre système reçois {storeName": "Magasin 1,"productName": "RTX 3090","quantity": 500}, vous le sauvegarderez, dans votre système, et vous le fournirez également dans le corps du message, lors de l'appel à l'Url de rappel du workflow |
description & summary | Chaines de caractères affichées dans l'interface de l'éditeur de Workflow Conseils sur les chaînes de connecteur |
operationId | Identification de l'opération |
x-ms-trigger | Définie une opération déclencheur de type Webhook |
parameters | Propriété importante, car elle définie en entrée "in":"body", c'est à dire dans le corps du message, le paramètre nommé arbitrairement Callback qui inclura l'Url de rappel fournie par Logic App/Power Automate |
Note : Tous le champs définis dans la définition du Callback sont important et à ne pas omettre
Propriété | Définition |
---|---|
"required":["Url"] | Indique que le champ Url est requis. |
"x-ms-notification-url":"true" | Indique à Logic App/Power Automate de placer l'Url de rappel dans le champ Url. |
"x-ms-visibility":"internal" | Indique que le champ doit être masqué aux utilisateurs. |
Voici sa représentation en C#, ce n'est ni plus ni moins qu'une classe avec une propriété de type string
public class CallBack
{
public string Url { get; set; }
}
Et la représentation C# de l'opération NewInstoreProduct
[Consumes(MediaTypeNames.Application.Json)]
[ProducesResponseType(StatusCodes.Status201Created)]
[HttpPost, Route("/event/instore")]
public IActionResult NewInstoreProduct([FromBody] CallBack callback)
{
_logger.LogInformation("Abonnement à l'évènement 'Arrivée d'un nouveau produit' ");
Subscription subscription = AddSubscription(callback, TypeEvent.InStore, this.Request.Headers);
string location = $"https://{this.Request.Host.Host}/event/remove/{subscription.Oid}/{subscription.Id}/";
return new CreatedResult(location, null);
}
Note: C'est cette méthode qui sera appelée par Logic App/Power Automate avec l'Url de rappel contenu dans la propriété CallBack.Url.
Lors de l'inscription de l'abonnement du Workflow, la méthode NewInstoreProduct est invoquée, c'est à ce moment-là qu'elle doit retourner dans le champ d'entête Location, l'Url de suppression d'un abonnement. Cette action se déclenche lorsque le connecteur personnalisé n'est plus utilisé ou que le workflow l'utilisant est supprimé.
Note: Le champ d'entête Location n'a de sens que si la méthode retourne une réponse d'état 201 (crée).
Afin de retrouver l'abonnement à supprimer, l'Url de suppression sera constituée d'un identificateur oid représentant l'utilisateur authentifié et de l'identificateur Id représentant le numéro de la souscription renvoyé par Logic App/Power Automate dans le champ d'entête "x-ms-workflow-subscription-id".
Le format de cette Url sera donc "/event/remove/{oid}/{Id}/". Bien évidement ce n'est pas figé et dépendra sans doute de votre propre logique.
La définition OpenAPI pour la suppression de l'abonnement, doit donc reprendre le format de cette Url, comme illustré dans l'extrait suivant.
"/event/remove/{oid}/{id}": {
"delete": {
"description": "Supprimer l'abonnement",
"summary": "Supprimer l'abonnement",
"operationId": "RemoveSubscription",
"x-ms-visibility": "internal",
"parameters": [
{
"in": "path",
"name": "oid",
"required": true,
"type": "string"
},
{
"name": "id",
"in": "path",
"description": "Identification de l'abonnement",
"required": true,
"type": "string"
}
],
"responses": {
"200": {
"description": "Success"
}
}
}
}
...
Les deux paramètres oid et id sont requis et doivent être placés dans l'Url elle même "in":"path". Comme c'est une opération de type Action, nous ne souhaitons pas qu'elle soit visible dans l'éditeur de workflow "x-ms-visibility":"internal"
Sa représentation C# est la suivante :
[HttpDelete, Route("/event/remove/{oid}/{id}")]
public IActionResult RemoveSubscription(string oid, string id)
{
if (_subscriptions.Count == 0)
{
_logger.LogInformation("Aucun abonnement en cours...");
return Ok();
}
var itemToRemove = _subscriptions.Single(r => r.Id == id && r.Oid == oid);
if (itemToRemove !=null)
{
_logger.LogInformation($"Suppression de l'abonnement : {id}, pour l'utilisateur '{itemToRemove.Name}'");
_subscriptions.Remove(itemToRemove);
}
return Ok();
}
Note : C'est un code très naïf, car les abonnements sont placés en mémoire dans une simple liste. Il faudra sans doute penser à un système plus robuste et autonome, avec vrai référentiel. Mais cela suffit ici pour nos besoins de démonstration.
Pour déclencher un workflow, il suffira d'exécuter une requête Post sur l'Url de rappel, en n'oubliant pas delui passer le contenu du corps du message, comme illustré dans le code c# suivant.
[HttpPost, Route("/fire/instore")]
[AllowAnonymous]
public async Task<IActionResult> FireInstore([FromBody] InStore inStore)
{
_inStores.Add(inStore);
//Retrouve tous les abonnements de type InStore
var inStoreSubscriptions = _subscriptions
.Where(s => s.Event == TypeEvent.InStore)
.Select(s => s).ToList();
var client = _clientFactory.CreateClient();
//Parcourir les abonnements
foreach (var instoreSub in inStoreSubscriptions)
{
//Construire la charge de travail au format Json
string jsonData = JsonConvert.SerializeObject(inStore);
StringContent stringContent = new StringContent(jsonData, System.Text.Encoding.UTF8, "application/json");
try
{
// Invoque l'Url de rappel avec la charge de travail
await client.PostAsync(instoreSub.CallBackUrl, stringContent);
instoreSub.LastStartTime= DateTime.UtcNow.ToLongTimeString();
}
catch (Exception ex)
{
_logger.LogError(ex.Message);
}
}
return Accepted($"Il y a {inStoreSubscriptions.Count} abonnement(s) au connecteur");
}
En l'état, il est possible de commencer à tester la création du connecteur personnalisé.
Nous allons le tester sur Power Automate, si vous n'avez pas d'abonnement vous pouvez obtenir un essai gratuit en suivant la procédure ici
- Récupérez le fichier définition.
Note : Ce fichier est également compatible avec un connecteur personnalisé Azure Logic App
-
Connectez-vous au portail https://make.powerapps.com/.
-
Dans le panneau gauche, sélectionnez "Données" | "Connecteurs Personnalisés".
-
En haut à droite sélectionnez "+ Nouveau connecteur personnalisé" | "Importer un fichier OpenAPI"
-
Dans la boîte de dialogue qui apparait, donnez un nom au connecteur puis, importez le fichier que vous venez de récupérer.
-
Vous noterez dans l'onglet "1.Général" que le champ Hôte n'est pas renseigné. Pour cela il vous faut publier le code de cet article. Néanmoins, vous pouvez continuer les étapes suivantes, sans vous en préoccuper outre mesure, en indiquant dans le champ hôte un FQDN factice, style toto.contoso.com, par contre à l'enregistrement du flux, rien ne se passera bien évidement. Sinon suivez les sous-étapes suivantes :
-
Cloner le code. git clone https://github.com/EricVernie/CustomConnector.git
-
Ajoutez le champ Hôte du style [NOM DE L'APPLICATION].azurewebsites.com. Ou [NOM DE L'APPLICATION] est le nom de l'application que vous avez donné lors de la publication à l'étape IV.
-
Allez ensuite dans l'onglet "3. Définition" afin de vérifier qu'aucune erreur n'est survenue. Vous noterez qu'aucun déclencheur n'est disponible. Ceci peut être déroutant, mais ils sont bien présent. Vous pourrez le vérifier en éditant le swagger dans l'interface.
-
Sélectionnez "Créer le connecteur". Si tout se passe bien, le connecteur est créé.
-
Nous allons maintenant créer un Flux en sélectionnant dans le panneau Gauche "Flux".
-
Puis en haut de l'écran "+ Nouveau Flux" | "Flux de cloud automatisé".
-
Donnez un nom au flux, puis activez le bouton "Ignorer", afin de créer un flux vierge.
-
Sélectionnez l'onglet "Personnalisé" | puis sélectionnez le connecteur personnalisé que vous venez de créer.
-
Deux déclencheurs devrait apparaitre comme illustré sur la figure suivante :
-
Sélectionnez "lorsqu'un nouveau produit arrive dans le magasin (version d'évaluation)". A ce stade comme aucune information de sécurité n'a été ajoutée, la connexion se fait automatiquement.
-
Ajoutez une nouvelle étape de type "Notifications" | par exemple "Send me a mobile notification" ou "Send me an email Notification" si vous n'avez pas l'application Power Automate Mobile
-
Vous pouvez alors remplir, la zone de texte rapidement, en choisissant du contenu dynamique. Vous noterez la correspondance entre les champs du contenu dynamique et la définition/inStore du fichier de définition OpenAPI vu plus haut.
En tant que développeur d'une API, je n'imagine pas une seule seconde, que votre API ne soit pas protégée d'une manière ou d'une autre. Soit par une clé d'API, ou soit via l'intermédiaire d'un fournisseur d'identité.
Dans notre exemple, notre API Webhook est protégée via le fournisseur d'identité Azure Active Directory, car c'est la manière la plus naturelle avec un connecteur Power Automate d'identifier un utilisateur.
En d'autres termes, la méthode d'abonnement ne peut être appelée que si un utilisateur s'authentifie.
Tout d'abord il faut inscrire l'application dans Azure Active Directory
Voici les différentes étapes à suivre :
-
A l'aide du portail https://aad.portal.azure.com/, sélectionnez "Azure Active Directory" | Inscription d'applications.
-
"+ Nouvelle inscription" | Donnez un Nom | Cochez "Comptes dans un annuaire d'organisation (tout annuaire Azure AD - Multilocataire).
-
Cliquez sur le bouton "S'inscrire".
-
Une fois l'application inscrite, sélectionnez "Vue d'ensemble" et copiez le GUID "ID d'application (client)".
-
Ensuite sélectionnez "Certificats & Secrets" | "+ Nouveau secret client" (copiez le secret pour une utilisation future).
-
Puis sélectionnez "API autorisées" | "+ Ajouter une autorisation" | "Microsoft Graph" | "Autorisations déléguées" | cochez "openid et profile".
-
Sélectionnez "Exposer une API" | "+ Ajouter une étendue" | Nom de l’étendue : "impersonate" | Qui peut accepter :" Administrateurs et Utilisateurs" | puis remplissez les autres champs obligatoires.
-
Copiez l'étendue qui doit être de la forme api://[ID de l'application]/impersonate
A ce stade vous devez avoir copié trois paramètres
-
ID d'application (client)
-
Le secret de l'application
-
l'étendue de l'application
-
-
Il nous reste encore un élément essentiel que nous n'avons pas encore renseigné, mais qui ne peut être que fourni que par l'éditeur de connecteur personnalisé, lorsqu'on renseigne les différents champs dans l'onglet sécurité c'est l'Url de redirection. Retournez sur le portail Power Automate et renseignez les champs dans l'onglet "2.Securité" comme illustré sur la figure suivante :
Champs Valeur Type Authentification Oauth 2.0 Fournisseur d'identité Azure Active Directory Client id copiez ID d'application (client) Client Secret copiez le secret de l'application Resource Url copiez ID d'application (client)* Etendue copiez l'étendue de l'application Une fois le connecteur enregistré, copiez l'URL de redirection, car nous allons finir l'enregistrement de notre application Azure Active Directory.
Note : Logic App retourne une URL du style : https://logic-apis-francecentral.consent.azure-apim.net/redirect, Power Automate retourne une URL du style : https://global.consent.azure-apim.net/redirect
Note:*Il est important dans ce champ Resource Url de bien y mettre l'application ID.
-
Retournez sur le portail Azure Active Directory https://aad.portal.azure.com, sélectionnez l'application que vous venez d'inscrire
-
Sélectionnez "Authentification" | "+ Ajoutez une plateforme" | Application Web | Web | Copiez l'URI de redirection
- Editez le fichier appsettings.json et copiez vos informations obtenues lors de l'enregistrement de l'application Azure Active Directory dans la section AzureAd.
"AzureAd": {
"Instance": "https://login.microsoftonline.com/",
"Domain": "[VOTRE NOM DE DOMAINE ex demozonex.com]",
"Audience": "[copiez ID d'application (client)]",
"TenantId": "[copiez ID du tenant Azure Active Directory]",
"ClientId": "[copiez ID d'application (client)]"
},
-
Une fois l'application publiée, vous devez avoir un FQDN du style [NOM DE L'APPLICATION].azurewebsites.net qu'il faudra renseigner dans la propriété host du fichier de définition ou dans le champ Hôte de l'onglet 1.Général, lors de la création du connecteur
{
"swagger": "2.0",
"info": {
"title": "Exemple de connecteur personnalisé",
"version": "v1"
},
"host": "[NOM DE L'APPLICATION].azurewebsites.net",
"basePath": "/",
"schemes": [ "https" ],
-
Créez un nouveau flux sur le portail Power Automate en prenant soin de supprimer toutes références aux connexions du connecteur personnalisé. Si tout fonctionne correctement vous devriez voir apparaître le bouton "Se connecter" comme illustrer sur la figure suivante.
-
Une fois le flux crée, sélectionnez en haut à droite "Test" | "Manuellement" | "Enregistrer et tester".
-
Ensuite allez dans un navigateur, puis entrez l'url https://[NOM DE L'APPLICATION].azurewebsites.net/swagger.
-
Sélectionnez la méthode POST /fire/instore comme illustré sur la figure suivante :
-
Et si tout a fonctionné, comme illustré sur la figure suivante.
-
et si vous avez l'application Power Automate sur votre mobile, vous devriez recevoir la notification.
-
Ou alors si vous avez choisi plus haut l'action "Send me an email Notification".
Les connecteurs personnalisés à base de déclencheurs, vont vous permettre de déverrouiller des scénarios, sans que vous soyez obligé d'investir lourdement dans le développement de solutions personnalisées.
Il y a quand même un peu de travail. Il faut écrire le WebHook pour la gestion des abonnements, et ajouter à vos évènements le déclenchement des workflows. Mais rien en comparaison si vous aviez à développer des solutions personnalisées pour tous types de fournisseurs.
Appuyez-vous sur l'écosystème des connecteurs personnalisés, afin de fournir à vos clients des solutions clés en main adaptées à leurs besoins, voir leur laisser le soin de créer leurs propres workflows à leur guise.
Ce n'est pas super évident de déboguer un Webhook dans ce contexte (C'est pour cela que dans le code j'ai ajouté des logs), alors si vous avez installé l'application sur Azure, il est possible de s'abonner aux flux de logs du WebHook afin de pister les erreurs.
Pour ce faire vous pouvez utiliser l'utilitaire az cli
Ensuite la commande az webapp log tail --name [NOM APPLICATION] -g [NOM DU RESSOURCE GROUPE] va vous permettre de vous connecter au flux de logs à partir de votre poste de travail, comme illustré sur la figure suivante.