using DotNetCore.CAP; using LINGYUN.Abp.BackgroundTasks; using LINGYUN.Abp.ExceptionHandling; using LINGYUN.Abp.ExceptionHandling.Emailing; using LINGYUN.Abp.Localization.CultureMap; using LINGYUN.Abp.Serilog.Enrichers.Application; using LINGYUN.Abp.Serilog.Enrichers.UniqueId; using LINGYUN.Abp.Webhooks; using LINGYUN.Abp.Webhooks.BackgroundJobs; using LINGYUN.Abp.WebhooksManagement; using LINGYUN.Abp.WebhooksManagement.Localization; using LINGYUN.Abp.Wrapper; using Medallion.Threading; using Medallion.Threading.Redis; using Microsoft.AspNetCore.Authentication.JwtBearer; using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.DataProtection; using Microsoft.AspNetCore.Routing; using Microsoft.Extensions.Caching.StackExchangeRedis; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.DependencyInjection.Extensions; using Microsoft.IdentityModel.Logging; using Microsoft.OpenApi.Models; using OpenTelemetry.Metrics; using OpenTelemetry.Resources; using OpenTelemetry.Trace; using Quartz; using StackExchange.Redis; using System; using System.Collections.Generic; using System.Text.Encodings.Web; using System.Text.Unicode; using Volo.Abp; using Volo.Abp.AspNetCore.Mvc; using Volo.Abp.Auditing; using Volo.Abp.Caching; using Volo.Abp.EntityFrameworkCore; using Volo.Abp.FeatureManagement; using Volo.Abp.GlobalFeatures; using Volo.Abp.Http.Client; using Volo.Abp.Json; using Volo.Abp.Json.SystemTextJson; using Volo.Abp.Localization; using Volo.Abp.MultiTenancy; using Volo.Abp.PermissionManagement; using Volo.Abp.Quartz; using Volo.Abp.Security.Claims; using Volo.Abp.SettingManagement; using Volo.Abp.Threading; using Volo.Abp.Timing; using Volo.Abp.VirtualFileSystem; namespace LY.MicroService.WebhooksManagement; public partial class WebhooksManagementHttpApiHostModule { public static string ApplicationName { get; set; } = "WebhookService"; private static readonly OneTimeRunner OneTimeRunner = new OneTimeRunner(); private void PreConfigureFeature() { OneTimeRunner.Run(() => { GlobalFeatureManager.Instance.Modules.Editions().EnableAll(); }); } private void PreForwardedHeaders() { } private void PreConfigureApp(IConfiguration configuration) { AbpSerilogEnrichersConsts.ApplicationName = ApplicationName; PreConfigure(options => { // 以开放端口区别 options.SnowflakeIdOptions.WorkerId = 28; options.SnowflakeIdOptions.WorkerIdBits = 5; options.SnowflakeIdOptions.DatacenterId = 1; }); if (configuration.GetValue("App:ShowPii")) { IdentityModelEventSource.ShowPII = true; } } private void PreConfigureCAP(IConfiguration configuration) { PreConfigure(options => { options .UseMySql(sqlOptions => { configuration.GetSection("CAP:MySql").Bind(sqlOptions); }) .UseRabbitMQ(rabbitMQOptions => { configuration.GetSection("CAP:RabbitMQ").Bind(rabbitMQOptions); }) .UseDashboard(); }); } private void PreConfigureQuartz(IConfiguration configuration) { PreConfigure(options => { // 如果使用持久化存储, 则配置quartz持久层 if (configuration.GetSection("Quartz:UsePersistentStore").Get()) { var settings = configuration.GetSection("Quartz:Properties").Get>(); if (settings != null) { foreach (var setting in settings) { options.Properties[setting.Key] = setting.Value; } } options.Configurator += (config) => { config.UsePersistentStore(store => { store.UseProperties = false; store.UseNewtonsoftJsonSerializer(); }); }; } }); } private void ConfigureDbContext() { // 配置Ef Configure(options => { options.UseMySQL(); //options.Configure(cfg => //{ // cfg.UseMySQL(); // cfg.DbContextOptions.EnableSensitiveDataLogging(); //}); }); } private void ConfigureBackgroundTasks(IServiceCollection services, IConfiguration configuration) { var webhooksOptions = services.ExecutePreConfiguredActions(); Configure(options => { options.NodeName = ApplicationName; options.JobDispatcherSelectors.AddJob( job => { job.NodeName = ApplicationName; // 需要间隔时长控制 job.Interval = 10; job.MaxCount = webhooksOptions.MaxSendAttemptCount; job.MaxTryCount = webhooksOptions.MaxSendAttemptCount; // 需要锁定作业 job.LockTimeOut = webhooksOptions.TimeoutDuration.TotalSeconds.To(); if (webhooksOptions.IsAutomaticSubscriptionDeactivationEnabled) { job.MaxCount = webhooksOptions.MaxConsecutiveFailCountBeforeDeactivateSubscription; } }); //options.JobDispatcherSelectors.AddNamespace( // "LINGYUN.Abp.Webhooks.BackgroundJobs", // job => // { // // more // }); }); } private void ConfigureJsonSerializer(IConfiguration configuration) { // 统一时间日期格式 Configure(options => { var jsonConfiguration = configuration.GetSection("Json"); if (jsonConfiguration.Exists()) { jsonConfiguration.Bind(options); } }); // 中文序列化的编码问题 Configure(options => { options.JsonSerializerOptions.Encoder = JavaScriptEncoder.Create(UnicodeRanges.All); }); } private void ConfigureDistributedLock(IServiceCollection services, IConfiguration configuration) { var distributedLockEnabled = configuration["DistributedLock:IsEnabled"]; if (distributedLockEnabled.IsNullOrEmpty() || bool.Parse(distributedLockEnabled)) { var redis = ConnectionMultiplexer.Connect(configuration["DistributedLock:Redis:Configuration"]); services.AddSingleton(_ => new RedisDistributedSynchronizationProvider(redis.GetDatabase())); } } private void ConfigureOpenTelemetry(IServiceCollection services, IConfiguration configuration) { var openTelemetryEnabled = configuration["OpenTelemetry:IsEnabled"]; if (openTelemetryEnabled.IsNullOrEmpty() || bool.Parse(openTelemetryEnabled)) { services.AddOpenTelemetry() .ConfigureResource(resource => { resource.AddService(ApplicationName); }) .WithTracing(tracing => { tracing.AddHttpClientInstrumentation(); tracing.AddAspNetCoreInstrumentation(); tracing.AddCapInstrumentation(); tracing.AddEntityFrameworkCoreInstrumentation(); tracing.AddSource(ApplicationName); var tracingOtlpEndpoint = configuration["OpenTelemetry:Otlp:Endpoint"]; if (!tracingOtlpEndpoint.IsNullOrWhiteSpace()) { tracing.AddOtlpExporter(otlpOptions => { otlpOptions.Endpoint = new Uri(tracingOtlpEndpoint); }); return; } var zipkinEndpoint = configuration["OpenTelemetry:ZipKin:Endpoint"]; if (!zipkinEndpoint.IsNullOrWhiteSpace()) { tracing.AddZipkinExporter(zipKinOptions => { zipKinOptions.Endpoint = new Uri(zipkinEndpoint); }); return; } }) .WithMetrics(metrics => { metrics.AddRuntimeInstrumentation(); metrics.AddHttpClientInstrumentation(); metrics.AddAspNetCoreInstrumentation(); }); } } private void ConfigureExceptionHandling() { // 自定义需要处理的异常 Configure(options => { // 加入需要处理的异常类型 options.Handlers.Add(); options.Handlers.Add(); options.Handlers.Add(); options.Handlers.Add(); options.Handlers.Add(); options.Handlers.Add(); }); // 自定义需要发送邮件通知的异常类型 Configure(options => { // 是否发送堆栈信息 options.SendStackTrace = true; // 未指定异常接收者的默认接收邮件 // 指定自己的邮件地址 }); } private void ConfigureAuditing(IConfiguration configuration) { Configure(options => { options.ApplicationName = ApplicationName; // 是否启用实体变更记录 var allEntitiesSelectorIsEnabled = configuration["Auditing:AllEntitiesSelector"]; if (allEntitiesSelectorIsEnabled.IsNullOrWhiteSpace() || (bool.TryParse(allEntitiesSelectorIsEnabled, out var enabled) && enabled)) { options.EntityHistorySelectors.AddAllEntities(); } }); } private void ConfigureTiming(IConfiguration configuration) { Configure(options => { configuration.GetSection("Clock").Bind(options); }); } private void ConfigureCaching(IConfiguration configuration) { Configure(options => { configuration.GetSection("DistributedCache").Bind(options); }); Configure(options => { var redisConfig = ConfigurationOptions.Parse(options.Configuration); options.ConfigurationOptions = redisConfig; options.InstanceName = configuration["Redis:InstanceName"]; }); } private void ConfigureMvc(IServiceCollection services, IConfiguration configuration) { Configure(options => { options.ExposeIntegrationServices = true; }); Configure(options => { options.EndpointConfigureActions.Add((builder) => { builder.Endpoints.MapHealthChecks(configuration["App:HealthChecks"] ?? "/healthz"); }); }); services.AddHealthChecks(); } private void ConfigureVirtualFileSystem() { Configure(options => { options.FileSets.AddEmbedded("LY.MicroService.WebhooksManagement"); }); } private void ConfigureMultiTenancy(IConfiguration configuration) { // 多租户 Configure(options => { options.IsEnabled = true; }); var tenantResolveCfg = configuration.GetSection("App:Domains"); if (tenantResolveCfg.Exists()) { Configure(options => { var domains = tenantResolveCfg.Get(); foreach (var domain in domains) { options.AddDomainTenantResolver(domain); } }); } } private void ConfigureIdentity(IConfiguration configuration) { Configure(options => { options.IsDynamicClaimsEnabled = true; options.RemoteRefreshUrl = configuration["App:RefreshClaimsUrl"] + options.RemoteRefreshUrl; }); } private void ConfigureSwagger(IServiceCollection services) { // Swagger services.AddSwaggerGen( options => { options.SwaggerDoc("v1", new OpenApiInfo { Title = "WebhooksManagement API", Version = "v1" }); options.DocInclusionPredicate((docName, description) => true); options.CustomSchemaIds(type => type.FullName); options.AddSecurityDefinition("Bearer", new OpenApiSecurityScheme { Description = "JWT Authorization header using the Bearer scheme. Example: \"Authorization: Bearer {token}\"", Name = "Authorization", In = ParameterLocation.Header, Scheme = "bearer", Type = SecuritySchemeType.Http, BearerFormat = "JWT" }); options.AddSecurityRequirement(new OpenApiSecurityRequirement { { new OpenApiSecurityScheme { Reference = new OpenApiReference { Type = ReferenceType.SecurityScheme, Id = "Bearer" } }, new string[] { } } }); options.OperationFilter(); }); } private void ConfigureFeatureManagement() { Configure(options => { options.IsDynamicFeatureStoreEnabled = true; }); } private void ConfigurePermissionManagement() { Configure(options => { options.IsDynamicPermissionStoreEnabled = true; }); } private void ConfigureSettingManagement() { Configure(options => { options.IsDynamicSettingStoreEnabled = true; }); } private void ConfigureWebhooks(IServiceCollection services) { Configure(options => { // 宿主应用中启用动态webhook options.IsDynamicWebhookStoreEnabled = true; }); // 宿主应用中使用默认发布者 services.Replace(ServiceDescriptor.Transient()); } private void ConfigureLocalization() { // 支持本地化语言类型 Configure(options => { options.Languages.Add(new LanguageInfo("en", "en", "English")); options.Languages.Add(new LanguageInfo("zh-Hans", "zh-Hans", "简体中文")); options.UsePersistence(); }); Configure(options => { var zhHansCultureMapInfo = new CultureMapInfo { TargetCulture = "zh-Hans", SourceCultures = new string[] { "zh", "zh_CN", "zh-CN" } }; options.CulturesMaps.Add(zhHansCultureMapInfo); options.UiCulturesMaps.Add(zhHansCultureMapInfo); }); } private void ConfigureSecurity(IServiceCollection services, IConfiguration configuration, bool isDevelopment = false) { services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme) .AddJwtBearer(options => { configuration.GetSection("AuthServer").Bind(options); }); if (!isDevelopment) { var redis = ConnectionMultiplexer.Connect(configuration["Redis:Configuration"]); services .AddDataProtection() .SetApplicationName("LINGYUN.Abp.Application") .PersistKeysToStackExchangeRedis(redis, "LINGYUN.Abp.Application:DataProtection:Protection-Keys"); } } private void ConfigureWrapper() { Configure(options => { options.IsEnabled = true; }); } private void PreConfigureWrapper() { //PreConfigure(options => //{ // options.ProxyRequestActions.Add( // (appid, httprequestmessage) => // { // httprequestmessage.Headers.TryAddWithoutValidation(AbpHttpWrapConsts.AbpDontWrapResult, "true"); // }); //}); // 服务间调用不包装 PreConfigure(options => { options.ProxyClientActions.Add( (_, _, client) => { client.DefaultRequestHeaders.TryAddWithoutValidation(AbpHttpWrapConsts.AbpDontWrapResult, "true"); }); }); } }