c# - 为什么 TimerTrigger 在基本 WebJobs SDK v3 主机应用程序中不要求存储帐户?

标签 c# azure azure-webjobs azure-webjobssdk

我正在使用 WebJobs SDK v3.0.5,使用一个非常简单的 .NET Core 2.2 控制台项目,如下所示:

TimerHost.csproj

<Project Sdk="Microsoft.NET.Sdk">
  <PropertyGroup>
    <OutputType>Exe</OutputType>
    <TargetFramework>netcoreapp2.2</TargetFramework>
    <LangVersion>7.1</LangVersion>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="Microsoft.Azure.WebJobs" Version="3.0.5" />
    <PackageReference Include="Microsoft.Azure.WebJobs.Extensions" Version="3.0.2" />
    <PackageReference Include="Microsoft.Extensions.Logging.Console" Version="2.2.0" />
  </ItemGroup>

  <ItemGroup>
    <None Update="appsettings.json">
      <CopyToOutputDirectory>Always</CopyToOutputDirectory>
    </None>
  </ItemGroup>
</Project>

Program.cs

using System.Threading.Tasks;
using Microsoft.Azure.WebJobs;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;

namespace TimerHost
{
    public class Program
    {
        public static async Task Main()
        {
            var builder = new HostBuilder();
            var host = builder
                .UseEnvironment("Development")
                .ConfigureServices((context, services) =>
                {
                    services.AddSingleton(context.Configuration);
                })
                .ConfigureWebJobs(webJobsBuilder =>
                {
                    webJobsBuilder
                        .AddAzureStorageCoreServices()
                        .AddTimers();
                })
                .ConfigureLogging((context, b) =>
                {
                    b.SetMinimumLevel(LogLevel.Debug);
                    b.AddConsole();
                })
                .UseConsoleLifetime()
                .Build();

            await host.RunAsync();
        }
    }

    public static class Function
    {
        public static void Run([TimerTrigger("*/10 * * * * *")] TimerInfo timer, ILogger logger)
        {
            logger.LogInformation($"Running job for timer. Next 3 runs are: {timer.FormatNextOccurrences(3)}");
        }
    }
}

appsettings.json

{
}

触发器运行良好。但是,根据最新文档 ( https://learn.microsoft.com/en-us/azure/app-service/webjobs-sdk-how-to#multiple-instances ),计时器应作为单例隐式运行,这意味着它应使用 Azure 存储帐户来支持分布式锁定。

在本地使用 Azure Functions 时,我希望提供如下设置:

{
  "AzureWebJobsStorage": "UseDevelopmentStorage=true"
}

否则我实际上无法运行函数,我收到一条错误,指出需要此设置,但是在控制台主机示例中,我根本没有收到任何错误。

有人可以解释为什么控制台主机不需要使用默认存储帐户吗?在这种情况下,计时器如何保持单例行为?

最佳答案

我花了一些时间从我的应用程序中调试 WebJobs SDK 源代码,并找到了有关幕后情况的更多信息:

  • 如果配置中未定义 AzureWebJobsStorage 应用设置,则 SDK 将回退到使用内存中分布式锁管理器来执行计时器和单例触发器。没有与此回退相关的日志记录,默认的锁定管理器仅适合本地开发。

  • 可以通过像 Azure Functions 一样设置连接字符串来使用 Azure 存储模拟器,只需确保已重新生成项目,以便将 appsettings.json 文件传播到项目输出文件夹,这让我有点困惑。

  • 如果 AzureWebJobsStorage 的值不是有效的本地或基于云的存储帐户连接字符串,则不会发出错误或日志条目 - 配置将只是默默地回退到内存中锁定管理器。

关于c# - 为什么 TimerTrigger 在基本 WebJobs SDK v3 主机应用程序中不要求存储帐户?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55055586/

相关文章:

c# - C++ 调用 C# 选项

azure - 将 Azure 触发的 WebJob 超时设置为无限

c# - LINQ 投影上的代码覆盖率

c# - 如何在访问多个不同的数据库时获得最佳性能

c# - EF Core 中的一对零或一对一关系

c# - 无法在 Azure C# 上创建存储帐户

node.js - 如何在 Azure Easy API 中启用 Swagger UI

c# - 将信息记录到 azure webjobs 日志文件中?

azure-webjobs - 重新部署网站后,在连续 Azure WebJob 上运行的进程会发生什么情况?

c# - Azure WebJob 服务总线在出现错误时在队列前面重新排队消息