你当前正在访问 Microsoft Azure Global Edition 技术文档网站。 如果需要访问由世纪互联运营的 Microsoft Azure 中国技术文档网站,请访问 https://docs.azure.cn

HTTP 功能

Durable Functions 提供多个功能来方便用户将持久性业务流程和实体整合到 HTTP 工作流中。 本文将会详细介绍其中的某些功能。

公开 HTTP API

可以使用 HTTP 请求来调用和管理业务流程与实体。 Durable Functions 扩展公开内置的 HTTP API。 它还提供相应的 API 用于从 HTTP 触发的函数内部来与业务流程和实体交互。

内置的 HTTP API

Durable Functions 扩展自动将一组 HTTP API 添加到 Azure Functions 宿主。 使用这些 API,无需编写任何代码即可与业务流程和实体交互并对其进行管理。

支持以下内置 HTTP API。

有关 Durable Functions 扩展公开的所有内置 HTTP API 的完整说明,请参阅 HTTP API 文章

HTTP API URL 发现

业务流程客户端绑定公开可以生成便捷 HTTP 响应有效负载的 API。 例如,它可以创建一个响应,其中包含特定业务流程实例的管理 API 的链接。 以下 HTTP 触发器函数示例演示如何对新的业务流程实例使用此 API:

// Copyright (c) .NET Foundation. All rights reserved.
// Licensed under the MIT License. See LICENSE in the project root for license information.

using System.Net.Http;
using System.Threading.Tasks;
using Microsoft.Azure.WebJobs;
using Microsoft.Azure.WebJobs.Extensions.DurableTask;
using Microsoft.Azure.WebJobs.Extensions.Http;
using Microsoft.Extensions.Logging;

namespace VSSample
{
    public static class HttpStart
    {
        [FunctionName("HttpStart")]
        public static async Task<HttpResponseMessage> Run(
            [HttpTrigger(AuthorizationLevel.Function, methods: "post", Route = "orchestrators/{functionName}")] HttpRequestMessage req,
            [DurableClient] IDurableClient starter,
            string functionName,
            ILogger log)
        {
            // Function input comes from the request content.
            object eventData = await req.Content.ReadAsAsync<object>();
            string instanceId = await starter.StartNewAsync(functionName, eventData);

            log.LogInformation($"Started orchestration with ID = '{instanceId}'.");

            return starter.CreateCheckStatusResponse(req, instanceId);
        }
    }
}

可以通过任何 HTTP 客户端启动使用上面所示 HTTP 触发器函数的业务流程协调程序函数。 以下 cURL 命令启动名为 DoWork 的业务流程协调程序函数:

curl -X POST https://localhost:7071/orchestrators/DoWork -H "Content-Length: 0" -i

下面是使用 abc123 作为 ID 的业务流程的示例响应。 为简明起见,此处删除了一些细节。

HTTP/1.1 202 Accepted
Content-Type: application/json; charset=utf-8
Location: http://localhost:7071/runtime/webhooks/durabletask/instances/abc123?code=XXX
Retry-After: 10

{
    "id": "abc123",
    "purgeHistoryDeleteUri": "http://localhost:7071/runtime/webhooks/durabletask/instances/abc123?code=XXX",
    "sendEventPostUri": "http://localhost:7071/runtime/webhooks/durabletask/instances/abc123/raiseEvent/{eventName}?code=XXX",
    "statusQueryGetUri": "http://localhost:7071/runtime/webhooks/durabletask/instances/abc123?code=XXX",
    "terminatePostUri": "http://localhost:7071/runtime/webhooks/durabletask/instances/abc123/terminate?reason={text}&code=XXX"
}

在以上示例中,以 Uri 结尾的每个字段对应于内置的 HTTP API。 可以使用这些 API 来管理目标业务流程实例。

注意

Webhook URL 的格式取决于所运行 Azure Functions 主机的版本。 前面的示例适用于 Azure Functions 2.0 主机。

有关所有内置 HTTP API 的介绍,请参阅 HTTP API 参考

异步操作跟踪

前面提到的 HTTP 响应旨在通过 Durable Functions 实现长时间运行的 HTTP 异步 API。 此模式有时称为“轮询使用者模式”。 客户端/服务器流工作方式如下:

  1. 客户端发出 HTTP 请求,以启动长时间运行的进程,例如业务流程协调程序函数。
  2. 目标 HTTP 触发器返回 HTTP 202 响应,其中包含值为“statusQueryGetUri”的 Location 标头。
  3. 客户端轮询 Location 标头中的 URL。 客户端会继续看到包含 Location 标头的 HTTP 202 响应。
  4. 实例完成或失败后,Location 标头中的终结点返回 HTTP 200。

此协议允许通过外部客户端或可轮询 HTTP 终结点并遵循 Location 标头的服务协调长时间运行的进程。 此模式的客户端和服务器实现内置于 Durable Functions HTTP API 中。

注意

默认情况下,Azure 逻辑应用提供的所有基于 HTTP 的操作都支持标准异步操作模式。 使用此功能,可在逻辑应用工作流中嵌入长时间运行的持久函数。 在 Azure 逻辑应用工作流操作和触发器文档中可以找到有关异步 HTTP 模式的逻辑应用支持的更多详细信息。

注意

可以从任何函数类型(而不仅仅是 HTTP 触发的函数)来与业务流程交互。

有关如何使用客户端 API 管理业务流程和实体的详细信息,请参阅实例管理文章

使用 HTTP API

根据协调程序函数代码约束中所述,业务流程协调程序函数无法直接执行 I/O。 这些函数通常调用执行 I/O 操作的活动函数

从 Durable Functions 2.0 开始,业务流程原生可以通过业务流程触发器绑定来使用 HTTP API。

以下示例代码演示了一个发出出站 HTTP 请求的业务流程协调程序函数:

[FunctionName(nameof(CheckSiteAvailable))]
public static async Task CheckSiteAvailable(
    [OrchestrationTrigger] IDurableOrchestrationContext context)
{
    Uri url = context.GetInput<Uri>();

    // Makes an HTTP GET request to the specified endpoint
    DurableHttpResponse response = 
        await context.CallHttpAsync(HttpMethod.Get, url);

    if (response.StatusCode >= 400)
    {
        // handling of error codes goes here
    }
}

注意

你可能想知道此功能为何使用 DurableHttpRequest 和 DurableHttpResponse 类型,而不是内置的 .NET HttpRequestMessage 和 HttpResponseMessage 类型。

这是有意设计的。 主要原因在于,自定义类型有助于确保用户不会对内部 HTTP 客户端支持的行为做出错误的假设。 使用特定于 Durable Functions 的类型还可以简化 API 设计。 此外,使用这些类型可以更轻松地实现托管标识集成轮询使用者模式等特殊功能。

使用“调用 HTTP”操作可以在业务流程协调程序函数中执行以下操作:

  • 直接从业务流程函数调用 HTTP API,但存在后面所述的一些限制。
  • 自动支持客户端 HTTP 202 状态轮询模式。
  • 使用 Azure 托管标识对其他 Azure 终结点发出授权的 HTTP 调用。

直接从业务流程协调程序函数使用 HTTP API 的功能旨在为一组特定的常见方案提供便利。 可以使用活动函数自行实现所有这些功能。 在许多情况下,活动函数可以提供更大的灵活性。

HTTP 202 处理(仅限.NET 进程内)

“调用 HTTP”API 可自动实现客户端的轮询使用者模式。 如果被调用的 API 返回带有 Location 标头的 HTTP 202 响应,则业务流程协调程序函数将自动轮询 Location 资源,直到收到非 202 响应。 此响应是返回到业务流程协调程序函数代码的响应。

注意

  1. 业务流程协调程序函数原生还支持服务器端“轮询使用者模式”,如异步操作跟踪中所述。 此项支持意味着,一个函数应用中的业务流程可以轻松协调其他函数应用中的业务流程协调程序函数。 这类似于子业务流程的概念,但支持跨应用的通信。 此项支持对于微服务式的应用开发特别有用。
  2. 内置 HTTP 轮询模式目前仅在 .NET 进程内主机中可用。

托管标识

Durable Functions 原生支持调用接受 Microsoft Entra 授权令牌的 API。 此项支持使用 Azure 托管标识来获取这些令牌。

以下代码是业务流程协调程序函数的一个示例。 该函数发出经过身份验证的调用来使用 Azure 资源管理器虚拟机 REST API 重启虚拟机。

[FunctionName("RestartVm")]
public static async Task RunOrchestrator(
    [OrchestrationTrigger] IDurableOrchestrationContext context)
{
    string subscriptionId = "mySubId";
    string resourceGroup = "myRG";
    string vmName = "myVM";
    string apiVersion = "2019-03-01";
    
    // Automatically fetches an Azure AD token for resource = https://management.core.windows.net/.default
    // and attaches it to the outgoing Azure Resource Manager API call.
    var restartRequest = new DurableHttpRequest(
        HttpMethod.Post, 
        new Uri($"https://management.azure.com/subscriptions/{subscriptionId}/resourceGroups/{resourceGroup}/providers/Microsoft.Compute/virtualMachines/{vmName}/restart?api-version={apiVersion}"),
        tokenSource: new ManagedIdentityTokenSource("https://management.core.windows.net/.default"));
    DurableHttpResponse restartResponse = await context.CallHttpAsync(restartRequest);
    if (restartResponse.StatusCode != HttpStatusCode.OK)
    {
        throw new ArgumentException($"Failed to restart VM: {restartResponse.StatusCode}: {restartResponse.Content}");
    }
}

在以上示例中,tokenSource 参数配置为获取 Azure 资源管理器的 Microsoft Entra 令牌。 该令牌由资源 URI https://management.core.windows.net/.default 标识。 该示例假设当前函数应用在本地运行,或者已使用托管标识部署为函数应用。 假设本地标识或托管标识有权管理指定资源组 myRG 中的 VM。

在运行时,配置的令牌源会自动返回 OAuth 2.0 访问令牌。 然后,源会将该令牌作为持有者令牌添加到传出请求的 Authorization 标头中。 相比于将授权标头手动添加到 HTTP 请求,此模型是一种改进,原因如下:

  • 可自动处理令牌刷新。 无需担心令牌过期。
  • 永远不会以持久性业务流程状态存储令牌。
  • 无需编写任何代码即可管理令牌获取。

可以在预编译的 C# RestartVMs 示例中找到更完整的示例。

托管标识并不局限于 Azure 资源管理。 可以使用托管标识来访问接受 Microsoft Entra 持有者令牌的任何 API,包括 Microsoft 提供的 Azure 服务,以及合作伙伴提供的 Web 应用。 合作伙伴的 Web 应用甚至可以是其他函数应用。 有关支持使用 Microsoft Entra 进行身份验证的 Microsoft Azure 服务的列表,请参阅支持 Microsoft Entra 身份验证的 Azure 服务

限制

内置的 HTTP API 调用支持只是一项便利功能。 它不一定适用于所有方案。

业务流程协调程序函数发送的 HTTP 请求及其响应将在 Durable Functions 存储提供程序中序列化并保存为队列消息。 此永久性排队行为确保 HTTP 调用在业务流程重播中可靠且安全。 但是,永久性队列行为也存在一些限制:

  • 与本机 HTTP 客户端相比,每个 HTTP 请求会造成更大的延迟。
  • 根据配置的存储提供程序,大型请求或响应消息可能会显著降低业务流程性能。 例如,使用 Azure 存储时,过大而无法适应 Azure 队列消息的 HTTP 有效负载会被压缩并存储在 Azure Blob 存储中。
  • 不支持流式处理、分块和二进制有效负载。
  • 自定义 HTTP 客户端行为的功能受到限制。

如果上述任何限制可能会影响你的用例,请考虑改用活动函数和特定于语言的 HTTP 客户端库来发出出站 HTTP 调用。

可扩展性(仅限 .NET 进程内)

对于进程内工作器,可以使用 Azure Functions .NET 依赖项注入来自定义业务流程内部 HTTP 客户端的行为。 此功能可用于做出轻微的行为更改。 使用此功能还可以通过注入 mock 对象,来对 HTTP 客户端进行单元测试。

以下示例演示如何使用依赖项注入为调用外部 HTTP 终结点的业务流程协调程序函数禁用 TLS/SSL 证书验证。

public class Startup : FunctionsStartup
{
    public override void Configure(IFunctionsHostBuilder builder)
    {
        // Register own factory
        builder.Services.AddSingleton<
            IDurableHttpMessageHandlerFactory,
            MyDurableHttpMessageHandlerFactory>();
    }
}

public class MyDurableHttpMessageHandlerFactory : IDurableHttpMessageHandlerFactory
{
    public HttpMessageHandler CreateHttpMessageHandler()
    {
        // Disable TLS/SSL certificate validation (not recommended in production!)
        return new HttpClientHandler
        {
            ServerCertificateCustomValidationCallback =
                HttpClientHandler.DangerousAcceptAnyServerCertificateValidator,
        };
    }
}

后续步骤