Scrivere codice in un connettore personalizzato
Il codice personalizzato trasforma i payload di richieste e risposte oltre l'ambito di modelli di criteri esistenti. Il codice utilizzato avrà la precedenza sulla definizione senza codice.
Per altre informazioni, vedi Creare un connettore personalizzato da zero.
Classe di script
Il codice deve implementare un metodo denominato ExecuteAsync, che viene chiamato durante il runtime. Puoi creare altri metodi in questa classe come necessario e chiamarli dal metodo ExecuteAsync. Il nome della classe deve essere Script e deve implementare ScriptBase.
public class Script : ScriptBase
{
public override Task<HttpResponseMessage> ExecuteAsync()
{
// Your code here
}
}
Definizione di classi e interfacce di supporto
Le seguenti classi e interfacce sono referenziate dalla classe Script. Possono essere utilizzate per il test e la compilazione locali.
public abstract class ScriptBase
{
// Context object
public IScriptContext Context { get; }
// CancellationToken for the execution
public CancellationToken CancellationToken { get; }
// Helper: Creates a StringContent object from the serialized JSON
public static StringContent CreateJsonContent(string serializedJson);
// Abstract method for your code
public abstract Task<HttpResponseMessage> ExecuteAsync();
}
public interface IScriptContext
{
// Correlation Id
string CorrelationId { get; }
// Connector Operation Id
string OperationId { get; }
// Incoming request
HttpRequestMessage Request { get; }
// Logger instance
ILogger Logger { get; }
// Used to send an HTTP request
// Use this method to send requests instead of HttpClient.SendAsync
Task<HttpResponseMessage> SendAsync(
HttpRequestMessage request,
CancellationToken cancellationToken);
}
Esempi
Script Hello World
Questo script di esempio restituirà sempre Hello World come risposta per tutte le richieste.
public override async Task<HttpResponseMessage> ExecuteAsync()
{
// Create a new response
var response = new HttpResponseMessage();
// Set the content
// Initialize a new JObject and call .ToString() to get the serialized JSON
response.Content = CreateJsonContent(new JObject
{
["greeting"] = "Hello World!",
}.ToString());
return response;
}
Script Regex
L'esempio seguente utilizza del testo per la corrispondenza con l'espressione regex e restituisce il risultato della corrispondenza nella risposta.
public override async Task<HttpResponseMessage> ExecuteAsync()
{
// Check if the operation ID matches what is specified in the OpenAPI definition of the connector
if (this.Context.OperationId == "RegexIsMatch")
{
return await this.HandleRegexIsMatchOperation().ConfigureAwait(false);
}
// Handle an invalid operation ID
HttpResponseMessage response = new HttpResponseMessage(HttpStatusCode.BadRequest);
response.Content = CreateJsonContent($"Unknown operation ID '{this.Context.OperationId}'");
return response;
}
private async Task<HttpResponseMessage> HandleRegexIsMatchOperation()
{
HttpResponseMessage response;
// We assume the body of the incoming request looks like this:
// {
// "textToCheck": "<some text>",
// "regex": "<some regex pattern>"
// }
var contentAsString = await this.Context.Request.Content.ReadAsStringAsync().ConfigureAwait(false);
// Parse as JSON object
var contentAsJson = JObject.Parse(contentAsString);
// Get the value of text to check
var textToCheck = (string)contentAsJson["textToCheck"];
// Create a regex based on the request content
var regexInput = (string)contentAsJson["regex"];
var rx = new Regex(regexInput);
JObject output = new JObject
{
["textToCheck"] = textToCheck,
["isMatch"] = rx.IsMatch(textToCheck),
};
response = new HttpResponseMessage(HttpStatusCode.OK);
response.Content = CreateJsonContent(output.ToString());
return response;
}
Script di inoltro
L'esempio seguente inoltra la richiesta in ingresso al back-end.
public override async Task<HttpResponseMessage> ExecuteAsync()
{
// Check if the operation ID matches what is specified in the OpenAPI definition of the connector
if (this.Context.OperationId == "ForwardAsPostRequest")
{
return await this.HandleForwardOperation().ConfigureAwait(false);
}
// Handle an invalid operation ID
HttpResponseMessage response = new HttpResponseMessage(HttpStatusCode.BadRequest);
response.Content = CreateJsonContent($"Unknown operation ID '{this.Context.OperationId}'");
return response;
}
private async Task<HttpResponseMessage> HandleForwardOperation()
{
// Example case: If your OpenAPI definition defines the operation as 'GET', but the backend API expects a 'POST',
// use this script to change the HTTP method.
this.Context.Request.Method = HttpMethod.Post;
// Use the context to forward/send an HTTP request
HttpResponseMessage response = await this.Context.SendAsync(this.Context.Request, this.CancellationToken).ConfigureAwait(continueOnCapturedContext: false);
return response;
}
Script di inoltro e trasformazione
L'esempio seguente inoltra la richiesta in ingresso e trasforma la risposta restituita dal back-end.
public override async Task<HttpResponseMessage> ExecuteAsync()
{
// Check if the operation ID matches what is specified in the OpenAPI definition of the connector
if (this.Context.OperationId == "ForwardAndTransformRequest")
{
return await this.HandleForwardAndTransformOperation().ConfigureAwait(false);
}
// Handle an invalid operation ID
HttpResponseMessage response = new HttpResponseMessage(HttpStatusCode.BadRequest);
response.Content = CreateJsonContent($"Unknown operation ID '{this.Context.OperationId}'");
return response;
}
private async Task<HttpResponseMessage> HandleForwardAndTransformOperation()
{
// Use the context to forward/send an HTTP request
HttpResponseMessage response = await this.Context.SendAsync(this.Context.Request, this.CancellationToken).ConfigureAwait(continueOnCapturedContext: false);
// Do the transformation if the response was successful, otherwise return error responses as-is
if (response.IsSuccessStatusCode)
{
var responseString = await response.Content.ReadAsStringAsync().ConfigureAwait(continueOnCapturedContext: false);
// Example case: response string is some JSON object
var result = JObject.Parse(responseString);
// Wrap the original JSON object into a new JSON object with just one key ('wrapped')
var newResult = new JObject
{
["wrapped"] = result,
};
response.Content = CreateJsonContent(newResult.ToString());
}
return response;
}
Spazi dei nomi supportati
Non tutti gli spazi dei nomi C# sono supportati. Attualmente, puoi utilizzare soltanto le funzioni dei seguenti spazi dei nomi.
using System;
using System.Collections;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.IO.Compression;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Net.Http.Headers;
using System.Net.Security;
using System.Security.Authentication;
using System.Security.Cryptography;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading;
using System.Threading.Tasks;
using System.Web;
using System.Xml;
using System.Xml.Linq;
using System.Drawing;
using System.Drawing.Drawing2D;
using System.Drawing.Imaging;
using Microsoft.Extensions.Logging;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
Esempi di GitHub
Per esempi del connettore DocuSign, vai a Connettori Power Platform in GitHub.
Domande frequenti sul codice personalizzato
Per saperne di più sul codice personalizzato, vedi Passaggio 4: (facoltativo) utilizzare il supporto per codice personalizzato.
D: È possibile utilizzare più script per connettore personalizzato?
R: No, è supportato un solo file di script per connettore personalizzato.
D: Durante l'aggiornamento del connettore personalizzato viene visualizzato un errore interno. Quale potrebbe essere il problema?
R: Molto probabilmente il problema riguarda la compilazione del codice. In futuro, visualizzeremo l'elenco completo degli errori di compilazione per migliorare questa esperienza. Al momento è consigliabile utilizzare le classi di supporto per testare gli errori di compilazione localmente come soluzione alternativa.
D: Posso aggiungere la registrazione al mio codice e ottenere un'analisi per il debug?
R: Non attualmente, ma il supporto per questa operazione verrà aggiunto in futuro.
D: Come posso testare il codice nel frattempo?
R: Testalo localmente e assicurati di poter compilare il codice usando solo gli spazi dei nomi forniti in Spazi dei nomi supportati. Per informazioni sui test locali, vedi Scrivere codice in un connettore personalizzato.
D: Ci sono limiti?
R: Sì. L'esecuzione dello script deve terminare nel giro di 5 secondi e la dimensione del file di script non può essere superiore a 1 MB.
D: Posso creare il mio client http nel codice script?
R: Attualmente sì, ma non in futuro. Il modo consigliato è usare il metodo this.Context.SendAsync.
D: Posso usare il codice personalizzato con il gateway dati locale?
R: Non attualmente, no.
Supporto di Rete virtuale
Quando il connettore viene usato in un ambiente Power Platform collegato a una rete virtuale, si applicano le limitazioni:
- Context.SendAsync usa un endpoint pubblico, quindi non può accedere ai dati dagli endpoint privati esposti nella rete virtuale.
Problemi noti e limitazioni generali
L'intestazione OperationId può essere restituita in formato codificato base64 in alcune regioni. Se il valore di OperationId è richiesto per un'implementazione, questo deve essere decodificato in base64 per l'uso in maniera simile alla seguente.
public override async Task<HttpResponseMessage> ExecuteAsync()
{
string realOperationId = this.Context.OperationId;
// Resolve potential issue with base64 encoding of the OperationId
// Test and decode if it's base64 encoded
try {
byte[] data = Convert.FromBase64String(this.Context.OperationId);
realOperationId = System.Text.Encoding.UTF8.GetString(data);
}
catch (FormatException ex) {}
// Check if the operation ID matches what is specified in the OpenAPI definition of the connector
if (realOperationId == "RegexIsMatch")
// Refer to the original examples above for remaining details
}
Passaggio successivo
Creare un connettore personalizzato da zero.
Inviare commenti
L'invio da parte degli utenti di feedback sui problemi riscontrati con la piattaforma di connettori o di idee su nuove funzionalità è molto apprezzato. Per fornire un feedback, vai a Inviare problemi o ottenere assistenza per i connettori e seleziona il tipo di commenti.