Écrire un plug-in personnalisé pour le Portail d’appareil Windows
Découvrez comment écrire une application UWP qui utilise Windows Device Portal (WDP) pour héberger une page web et fournir des informations de diagnostic.
À compter de Windows 10 Creators Update (version 1703, build 15063), vous pouvez utiliser le Portail d’appareil pour héberger les interfaces de diagnostic de votre application. Cet article décrit les trois tâches nécessaires pour créer un DevicePortalProvider pour votre application : les changements du manifeste du package d’application, la configuration de la connexion de votre application au service Portail d’appareil et la gestion d’une demande entrante.
Dans Microsoft Visual Studio, créez un projet d’application UWP. Accédez à Fichier > Nouveau > Projet et sélectionnez Application vide (Windows universelle) pour C#, puis cliquez sur Suivant. Dans la boîte de dialogue Configurer votre nouveau projet, nommez le projet « DevicePortalProvider », puis cliquez sur Créer. Il s’agira de l’application qui contient le service d’application. Vous devrez peut-être mettre à jour Visual Studio ou installer le SDK Windows le plus récent.
Vous devrez ajouter du code à votre fichier package.appxmanifest pour faire fonctionner votre application en tant que plug-in du Portail d'appareil. Commencez par ajouter les définitions d’espace de noms suivants au début du fichier. Ajoutez également ces définitions à l’attribut IgnorableNamespaces
.
<Package
...
xmlns:rescap="http://schemas.microsoft.com/appx/manifest/foundation/windows10/restrictedcapabilities"
xmlns:uap4="http://schemas.microsoft.com/appx/manifest/uap/windows10/4"
IgnorableNamespaces="uap mp rescap uap4">
...
Pour déclarer votre application en tant que fournisseur du Portail d'appareil, vous devez créer un service d’application, ainsi qu’une extension de fournisseur du Portail d'appareil utilisant ce service. Ajoutez à la fois l’extension windows.appService et l’extension windows.devicePortalProvider dans l’élément Extensions
sous Application
. Vérifiez que les attributs AppServiceName
correspondent dans les deux extensions. Cette opération indique au service du Portail d'appareil que ce service d’application peut être lancé pour gérer les requêtes sur l’espace de noms du gestionnaire.
...
<Application
Id="App"
Executable="$targetnametoken$.exe"
EntryPoint="DevicePortalProvider.App">
...
<Extensions>
<uap:Extension Category="windows.appService" EntryPoint="MySampleProvider.SampleProvider">
<uap:AppService Name="com.sampleProvider.wdp" />
</uap:Extension>
<uap4:Extension Category="windows.devicePortalProvider">
<uap4:DevicePortalProvider
DisplayName="My Device Portal Provider Sample App"
AppServiceName="com.sampleProvider.wdp"
HandlerRoute="/MyNamespace/api/" />
</uap4:Extension>
</Extensions>
</Application>
...
L’attribut HandlerRoute
référence l’espace de noms REST revendiqué par votre application. Toutes les requêtes HTTP sur cet espace de noms (implicitement suivi d’un caractère générique) qui sont reçues par le service du Portail d'appareil seront envoyées à votre application pour être gérées. Dans ce cas, toutes les requêtes HTTP correctement authentifiées à destination de <ip_address>/MyNamespace/api/*
seront envoyées à votre application. Les conflits entre les itinéraires de gestionnaire sont réglés par le biais d’une vérification du type « priorité à l’itinéraire le plus long » : autrement dit, l’itinéraire qui correspond le mieux aux requêtes est sélectionné. Par exemple, une requête adressée à « /MyNamespace/api/foo » sera mise en correspondance avec un fournisseur contenant « /MyNamespace/api » plutôt qu’avec un fournisseur indiquant simplement « /MyNamespace ».
Cette approche requiert deux nouvelles fonctionnalités. que vous devrez également ajouter à votre fichier package.appxmanifest.
...
<Capabilities>
...
<Capability Name="privateNetworkClientServer" />
<rescap:Capability Name="devicePortalProvider" />
</Capabilities>
...
Notes
La fonctionnalité « devicePortalProvider » est restreinte (« rescap »), ce qui signifie que vous devez obtenir une approbation préalable auprès du Windows Store avant de pouvoir y publier votre application. Toutefois, ceci ne vous empêche pas de tester votre application localement par le biais d’un chargement indépendant. Pour plus d’informations sur les fonctionnalités restreintes, consultez l’article Déclarations des fonctionnalités d’application.
Pour la configuration de la connexion du Portail d'appareil, votre application doit raccorder une connexion de service d’application à partir du service Device Portal à l’instance Device Portal en cours d’exécution au sein de votre application. Pour effectuer cette opération, ajoutez un nouveau composant WinRT à votre application avec une classe qui implémente IBackgroundTask.
using Windows.System.Diagnostics.DevicePortal;
using Windows.ApplicationModel.Background;
namespace MySampleProvider {
// Implementing a DevicePortalConnection in a background task
public sealed class SampleProvider : IBackgroundTask {
BackgroundTaskDeferral taskDeferral;
DevicePortalConnection devicePortalConnection;
//...
}
Assurez-vous que son nom correspond à l’espace de noms et au nom de classe configurés par AppService EntryPoint (« MySampleProvider.SampleProvider »). Lorsque vous adressez votre première requête à votre fournisseur du Portail d'appareil, le Portail d'appareil remise (stash) la requête, lance la tâche en arrière-plan de votre application, appelle sa méthode Run et transmet une interface IBackgroundTaskInstance. Votre application l’utilise ensuite pour configurer une instance DevicePortalConnection.
// Implement background task handler with a DevicePortalConnection
public void Run(IBackgroundTaskInstance taskInstance) {
// Take a deferral to allow the background task to continue executing
this.taskDeferral = taskInstance.GetDeferral();
taskInstance.Canceled += TaskInstance_Canceled;
// Create a DevicePortal client from an AppServiceConnection
var details = taskInstance.TriggerDetails as AppServiceTriggerDetails;
var appServiceConnection = details.AppServiceConnection;
this.devicePortalConnection = DevicePortalConnection.GetForAppServiceConnection(appServiceConnection);
// Add Closed, RequestReceived handlers
devicePortalConnection.Closed += DevicePortalConnection_Closed;
devicePortalConnection.RequestReceived += DevicePortalConnection_RequestReceived;
}
Deux événements doivent être gérés par l’application pour terminer la boucle de traitement de la demande : Fermé à chaque fois que le service du portail d’appareil s’arrête et RequestReceived, qui regroupe les requêtes HTTP entrantes et fournit la fonctionnalité principale du fournisseur du portail d’appareil.
L’événement RequestReceived se déclenche chaque fois qu’une requête HTTP est effectuée sur l’itinéraire du gestionnaire spécifié de votre plug-in. La boucle de gestion des requêtes pour les fournisseurs du Portail d'appareil est comparable à celle de NodeJS Express : les objets de requête et de réponse sont fournis en même temps que l’événement, et le gestionnaire répond en remplissant l’objet de réponse. Dans les fournisseurs du Portail d'appareil, l’événement RequestReceived et ses gestionnaires utilisent les objets Windows.Web.Http.HttpRequestMessage et HttpResponseMessage.
// Sample RequestReceived echo handler: respond with an HTML page including the query and some additional process information.
private void DevicePortalConnection_RequestReceived(DevicePortalConnection sender, DevicePortalConnectionRequestReceivedEventArgs args)
{
var req = args.RequestMessage;
var res = args.ResponseMessage;
if (req.RequestUri.AbsolutePath.EndsWith("/echo"))
{
// construct an html response message
string con = "<h1>" + req.RequestUri.AbsoluteUri + "</h1><br/>";
var proc = Windows.System.Diagnostics.ProcessDiagnosticInfo.GetForCurrentProcess();
con += String.Format("This process is consuming {0} bytes (Working Set)<br/>", proc.MemoryUsage.GetReport().WorkingSetSizeInBytes);
con += String.Format("The process PID is {0}<br/>", proc.ProcessId);
con += String.Format("The executable filename is {0}", proc.ExecutableFileName);
res.Content = new Windows.Web.HttpStringContent(con);
res.Content.Headers.ContentType = new Windows.Web.Http.Headers.HttpMediaTypeHeaderValue("text/html");
res.StatusCode = Windows.Web.Http.HttpStatusCode.Ok;
}
//...
}
Dans cet exemple de gestionnaire de requêtes, nous commençons par extraire les objets de requête et de réponse du paramètre args, puis nous créons une chaîne contenant l’URL de la requête et une mise en forme HTML supplémentaire. Cette chaîne est ajoutée à l’objet Response sous la forme d’une instance HttpStringContent. D’autres classes IHttpContent, telles que celles pour « String » et « Buffer », sont également autorisées.
La réponse est ensuite définie en tant que réponse HTTP et reçoit un code d’état 200 (OK). Elle devrait s’afficher comme prévu dans le navigateur qui a effectué l’appel d’origine. Notez que lorsque le gestionnaire d’événements RequestReceived répond, le message de réponse est automatiquement renvoyé à l’agent utilisateur : aucune méthode « send » supplémentaire n’est nécessaire.
Le contenu statique peut être traité directement à partir d’un dossier de votre package, ce qui facilite sensiblement l’ajout d’une interface utilisateur à votre fournisseur. Le moyen le plus simple de traiter du contenu statique consiste à créer dans votre projet un dossier de contenu pouvant être mappé sur une URL.
Ensuite, ajoutez un gestionnaire d’itinéraires à votre gestionnaire d’événements RequestReceived qui détecte les itinéraires de contenu statique et mappe une requête en conséquence.
if (req.RequestUri.LocalPath.ToLower().Contains("/www/")) {
var filePath = req.RequestUri.AbsolutePath.Replace('/', '\\').ToLower();
filePath = filePath.Replace("\\backgroundprovider", "")
try {
var fileStream = Windows.ApplicationModel.Package.Current.InstalledLocation.OpenStreamForReadAsync(filePath).GetAwaiter().GetResult();
res.StatusCode = HttpStatusCode.Ok;
res.Content = new HttpStreamContent(fileStream.AsInputStream());
res.Content.Headers.ContentType = new HttpMediaTypeHeaderValue("text/html");
} catch(FileNotFoundException e) {
string con = String.Format("<h1>{0} - not found</h1>\r\n", filePath);
con += "Exception: " + e.ToString();
res.Content = new Windows.Web.Http.HttpStringContent(con);
res.StatusCode = Windows.Web.Http.HttpStatusCode.NotFound;
res.Content.Headers.ContentType = new Windows.Web.Http.Headers.HttpMediaTypeHeaderValue("text/html");
}
}
Assurez-vous que tous les fichiers figurant dans le dossier de contenu sont marqués en tant que « Contenu » et sont définis sur « Copier si plus récent » ou sur « Toujours copier » dans le menu Propriétés de Visual Studio. Cette opération garantit que les fichiers seront présents dans votre package AppX lorsque vous le déploierez.
Le contenu statique traité par un fournisseur du Portail d'appareil est traité sur le même port que le service principal du Portail d'appareil. Cela signifie que vous pouvez référencer les éléments JS et CSS existants inclus dans le portail d'appareil avec de simples balises <link>
et <script>
dans votre code HTML. En règle générale, nous vous suggérons d’utiliser rest.js, qui inclut dans un wrapper la totalité des API REST principales du Portail d'appareil dans un objet webbRest pratique, ainsi que le fichier common.css, qui vous permettra de styliser votre contenu pour l’adapter au reste de l’interface utilisateur du Portail d'appareil. La page index.html illustrée ci-dessous vous en fournit un exemple. Elle utilise rest.js pour récupérer le nom de l’appareil et les processus en cours d’exécution à partir du Portail d'appareil.
Plus important encore, l’utilisation des méthodes HttpPost/DeleteExpect200 sur webbRest assurera automatiquement la gestion des attaques de falsification de requête intersites (CSRF, Cross Site Request Forgery) à votre intention, ce qui permet à votre page web d’appeler les API REST de modification d’état.
Notes
Le contenu statique fourni dans le Portail d'appareil n’est pas protégé contre les changements importants. Bien que les API ne soient pas susceptibles de changer fréquemment, il peut arriver qu’elles le soient, notamment dans les fichiers common.js et controls.js, que votre fournisseur ne doit pas utiliser.
Pour déboguer votre tâche en arrière-plan, vous devez modifier la façon dont Visual Studio exécute votre code. Pour inspecter le mode de gestion des requêtes HTTP par votre fournisseur, suivez la procédure ci-après de débogage d’une connexion de service d’application :
- Dans le menu Déboguer, sélectionnez Propriétés DevicePortalProvider.
- Sous l’onglet Débogage, dans la section Action de démarrage, sélectionnez « Ne pas lancer, mais déboguer mon code au démarrage ».
- Définissez un point d’arrêt dans la fonction du gestionnaire RequestReceived.
Notes
Assurez-vous que l’architecture de la build correspond exactement à celle de la cible. Si vous utilisez un PC 64 bits, vous devez effectuer le déploiement à l’aide d’une build AMD64. 4. Appuyez sur F5 pour déployer votre application. 5. Désactivez le Portail d'appareil, puis réactivez-le pour qu’il trouve votre application (cette opération n’est requise que si vous modifiez le manifeste de votre application ; dans les autres cas, il vous suffit de procéder à un nouveau déploiement et d’ignorer cette étape). 6. Dans votre navigateur, accédez à l’espace de noms de votre fournisseur ; le point d’arrêt devrait alors être atteint.