You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
364 lines
15 KiB
364 lines
15 KiB
using Hangfire.Redis;
|
|
using Lion.AbpPro.CAP.EntityFrameworkCore;
|
|
using Swagger;
|
|
using Volo.Abp.BackgroundJobs.Hangfire;
|
|
using Volo.Abp.Timing;
|
|
|
|
namespace Lion.AbpPro
|
|
{
|
|
[DependsOn(
|
|
typeof(AbpProHttpApiModule),
|
|
typeof(SharedHostingMicroserviceModule),
|
|
typeof(AbpAspNetCoreMvcUiMultiTenancyModule),
|
|
typeof(AbpProEntityFrameworkCoreModule),
|
|
typeof(AbpAspNetCoreAuthenticationJwtBearerModule),
|
|
typeof(AbpAspNetCoreSerilogModule),
|
|
typeof(AbpAccountWebModule),
|
|
typeof(AbpProApplicationModule),
|
|
typeof(LionAbpProCapModule),
|
|
typeof(LionAbpProCapEntityFrameworkCoreModule),
|
|
typeof(AbpAspNetCoreMvcUiBasicThemeModule),
|
|
typeof(AbpCachingStackExchangeRedisModule),
|
|
typeof(AbpBackgroundJobsHangfireModule)
|
|
)]
|
|
public class AbpProHttpApiHostModule : AbpModule
|
|
{
|
|
public override void OnPostApplicationInitialization(ApplicationInitializationContext context)
|
|
{
|
|
// 应用程序初始化的时候注册hangfire
|
|
//context.CreateRecurringJob();
|
|
base.OnPostApplicationInitialization(context);
|
|
}
|
|
|
|
public override void ConfigureServices(ServiceConfigurationContext context)
|
|
{
|
|
var configuration = context.Services.GetConfiguration();
|
|
ConfigureCache(context);
|
|
ConfigureSwaggerServices(context);
|
|
ConfigureJwtAuthentication(context, configuration);
|
|
ConfigureHangfire(context);
|
|
ConfigureMiniProfiler(context);
|
|
ConfigureIdentity(context);
|
|
ConfigureCap(context);
|
|
ConfigureAuditLog(context);
|
|
ConfigurationSignalR(context);
|
|
}
|
|
|
|
public override void OnApplicationInitialization(ApplicationInitializationContext context)
|
|
{
|
|
var app = context.GetApplicationBuilder();
|
|
var configuration = context.GetConfiguration();
|
|
app.UseAbpRequestLocalization();
|
|
app.UseCorrelationId();
|
|
app.UseStaticFiles();
|
|
if (configuration.GetValue("MiniProfiler:Enabled", false))
|
|
{
|
|
app.UseMiniProfiler();
|
|
}
|
|
|
|
app.UseRouting();
|
|
app.UseCors(AbpProHttpApiHostConst.DefaultCorsPolicyName);
|
|
app.UseAuthentication();
|
|
|
|
if (MultiTenancyConsts.IsEnabled)
|
|
{
|
|
app.UseMultiTenancy();
|
|
}
|
|
|
|
app.UseAuthorization();
|
|
app.UseSwagger();
|
|
app.UseAbpSwaggerUI(options =>
|
|
{
|
|
options.SwaggerEndpoint("/swagger/AbpPro/swagger.json", "AbpPro API");
|
|
options.DocExpansion(DocExpansion.None);
|
|
options.DefaultModelsExpandDepth(-1);
|
|
});
|
|
|
|
app.UseAuditing();
|
|
app.UseAbpSerilogEnrichers();
|
|
|
|
app.UseUnitOfWork();
|
|
app.UseConfiguredEndpoints(endpoints => { endpoints.MapHealthChecks("/health"); });
|
|
app.UseHangfireDashboard("/hangfire", new DashboardOptions()
|
|
{
|
|
Authorization = new[] { new CustomHangfireAuthorizeFilter() },
|
|
IgnoreAntiforgeryToken = true
|
|
});
|
|
|
|
if (configuration.GetValue("Consul:Enabled", false))
|
|
{
|
|
app.UseConsul();
|
|
}
|
|
}
|
|
|
|
|
|
#region 私有配置
|
|
|
|
private void ConfigureHangfire(ServiceConfigurationContext context)
|
|
{
|
|
var redisStorageOptions = new RedisStorageOptions()
|
|
{
|
|
Db = context.Services.GetConfiguration().GetValue<int>("Hangfire:Redis:DB")
|
|
};
|
|
|
|
Configure<AbpBackgroundJobOptions>(options => { options.IsJobExecutionEnabled = true; });
|
|
context.Services.AddHangfire(config =>
|
|
{
|
|
config.UseRedisStorage(ConnectionMultiplexer.Connect(context.Services.GetConfiguration().GetValue<string>("Hangfire:Redis:Host")), redisStorageOptions)
|
|
.WithJobExpirationTimeout(TimeSpan.FromDays(7));
|
|
var delaysInSeconds = new[] { 10, 60, 60 * 3 }; // 重试时间间隔
|
|
const int Attempts = 3; // 重试次数
|
|
config.UseFilter(new AutomaticRetryAttribute() { Attempts = Attempts, DelaysInSeconds = delaysInSeconds });
|
|
//config.UseFilter(new AutoDeleteAfterSuccessAttribute(TimeSpan.FromDays(7)));
|
|
config.UseFilter(new JobRetryLastFilter(Attempts));
|
|
});
|
|
}
|
|
|
|
/// <summary>
|
|
/// 配置MiniProfiler
|
|
/// </summary>
|
|
private void ConfigureMiniProfiler(ServiceConfigurationContext context)
|
|
{
|
|
if (context.Services.GetConfiguration().GetValue("MiniProfiler:Enabled", false))
|
|
{
|
|
context.Services.AddMiniProfiler(options => options.RouteBasePath = "/profiler").AddEntityFramework();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// 配置JWT
|
|
/// </summary>
|
|
private void ConfigureJwtAuthentication(ServiceConfigurationContext context, IConfiguration configuration)
|
|
{
|
|
context.Services.AddAuthentication(options =>
|
|
{
|
|
options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
|
|
options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
|
|
})
|
|
.AddJwtBearer(options =>
|
|
{
|
|
options.TokenValidationParameters =
|
|
new TokenValidationParameters()
|
|
{
|
|
// 是否开启签名认证
|
|
ValidateIssuerSigningKey = true,
|
|
ValidateIssuer = true,
|
|
ValidateAudience = true,
|
|
ValidateLifetime = true,
|
|
//ClockSkew = TimeSpan.Zero,
|
|
ValidIssuer = configuration["Jwt:Issuer"],
|
|
ValidAudience = configuration["Jwt:Audience"],
|
|
IssuerSigningKey =
|
|
new SymmetricSecurityKey(
|
|
Encoding.ASCII.GetBytes(configuration["Jwt:SecurityKey"]))
|
|
};
|
|
|
|
options.Events = new JwtBearerEvents
|
|
{
|
|
OnMessageReceived = currentContext =>
|
|
{
|
|
var path = currentContext.HttpContext.Request.Path;
|
|
if (path.StartsWithSegments("/login"))
|
|
{
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
var accessToken = string.Empty;
|
|
if (currentContext.HttpContext.Request.Headers.ContainsKey("Authorization"))
|
|
{
|
|
accessToken = currentContext.HttpContext.Request.Headers["Authorization"];
|
|
if (!string.IsNullOrWhiteSpace(accessToken))
|
|
{
|
|
accessToken = accessToken.Split(" ").LastOrDefault();
|
|
}
|
|
}
|
|
|
|
if (accessToken.IsNullOrWhiteSpace())
|
|
{
|
|
accessToken = currentContext.Request.Query["access_token"].FirstOrDefault();
|
|
}
|
|
|
|
if (accessToken.IsNullOrWhiteSpace())
|
|
{
|
|
accessToken = currentContext.Request.Cookies[AbpProHttpApiHostConst.DefaultCookieName];
|
|
}
|
|
|
|
currentContext.Token = accessToken;
|
|
currentContext.Request.Headers.Remove("Authorization");
|
|
currentContext.Request.Headers.Add("Authorization", $"Bearer {accessToken}");
|
|
|
|
return Task.CompletedTask;
|
|
}
|
|
};
|
|
});
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Redis缓存
|
|
/// </summary>
|
|
private void ConfigureCache(ServiceConfigurationContext context)
|
|
{
|
|
Configure<AbpDistributedCacheOptions>(
|
|
options => { options.KeyPrefix = "AbpPro:"; });
|
|
var configuration = context.Services.GetConfiguration();
|
|
var redis = ConnectionMultiplexer.Connect(configuration["Redis:Configuration"]);
|
|
context.Services
|
|
.AddDataProtection()
|
|
.PersistKeysToStackExchangeRedis(redis, "AbpPro-Protection-Keys");
|
|
}
|
|
|
|
/// <summary>
|
|
/// 配置Identity
|
|
/// </summary>
|
|
private void ConfigureIdentity(ServiceConfigurationContext context)
|
|
{
|
|
context.Services.Configure<IdentityOptions>(options => { options.Lockout = new LockoutOptions() { AllowedForNewUsers = false }; });
|
|
}
|
|
|
|
private void ConfigurationSignalR(ServiceConfigurationContext context)
|
|
{
|
|
var redisConnection = context.Services.GetConfiguration()["Redis:Configuration"];
|
|
|
|
if (redisConnection.IsNullOrWhiteSpace())
|
|
{
|
|
throw new UserFriendlyException(message: "Redis连接字符串未配置.");
|
|
}
|
|
|
|
context.Services.AddSignalR().AddStackExchangeRedis(redisConnection, options => { options.Configuration.ChannelPrefix = "Lion.AbpPro"; });
|
|
}
|
|
|
|
private void ConfigureSwaggerServices(ServiceConfigurationContext context)
|
|
{
|
|
context.Services.AddSwaggerGen(
|
|
options =>
|
|
{
|
|
// 文件下载类型
|
|
options.MapType<FileContentResult>(() => new OpenApiSchema() { Type = "file" });
|
|
|
|
options.SwaggerDoc("AbpPro",
|
|
new OpenApiInfo { Title = "LionAbpPro API", Version = "v1" });
|
|
options.DocInclusionPredicate((docName, description) => true);
|
|
options.EnableAnnotations(); // 启用注解
|
|
options.DocumentFilter<HiddenAbpDefaultApiFilter>();
|
|
options.SchemaFilter<EnumSchemaFilter>();
|
|
// 加载所有xml注释,这里会导致swagger加载有点缓慢
|
|
var xmlPaths = Directory.GetFiles(AppContext.BaseDirectory, "*.xml");
|
|
foreach (var xml in xmlPaths)
|
|
{
|
|
options.IncludeXmlComments(xml, true);
|
|
}
|
|
|
|
options.AddSecurityDefinition(JwtBearerDefaults.AuthenticationScheme,
|
|
new OpenApiSecurityScheme()
|
|
{
|
|
Description = "直接在下框输入JWT生成的Token",
|
|
Name = "Authorization",
|
|
In = ParameterLocation.Header,
|
|
Type = SecuritySchemeType.Http,
|
|
Scheme = JwtBearerDefaults.AuthenticationScheme,
|
|
BearerFormat = "JWT"
|
|
});
|
|
options.AddSecurityRequirement(new OpenApiSecurityRequirement
|
|
{
|
|
{
|
|
new OpenApiSecurityScheme
|
|
{
|
|
Reference = new OpenApiReference
|
|
{
|
|
Type = ReferenceType.SecurityScheme, Id = "Bearer"
|
|
}
|
|
},
|
|
new List<string>()
|
|
}
|
|
});
|
|
|
|
options.AddSecurityDefinition("ApiKey", new OpenApiSecurityScheme()
|
|
{
|
|
Type = SecuritySchemeType.ApiKey,
|
|
In = ParameterLocation.Header,
|
|
Name = "Accept-Language",
|
|
Description = "多语言设置,系统预设语言有zh-Hans、en,默认为zh-Hans",
|
|
});
|
|
|
|
options.AddSecurityRequirement(new OpenApiSecurityRequirement
|
|
{
|
|
{
|
|
new OpenApiSecurityScheme
|
|
{
|
|
Reference = new OpenApiReference
|
|
{ Type = ReferenceType.SecurityScheme, Id = "ApiKey" }
|
|
},
|
|
Array.Empty<string>()
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
|
|
private void ConfigureCap(ServiceConfigurationContext context)
|
|
{
|
|
var configuration = context.Services.GetConfiguration();
|
|
var enabled = configuration.GetValue("Cap:Enabled", false);
|
|
if (enabled)
|
|
{
|
|
context.AddAbpCap(capOptions =>
|
|
{
|
|
capOptions.SetCapDbConnectionString(configuration["ConnectionStrings:Default"]);
|
|
capOptions.UseEntityFramework<AbpProDbContext>();
|
|
capOptions.UseRabbitMQ(option =>
|
|
{
|
|
option.HostName = configuration.GetValue<string>("Cap:RabbitMq:HostName");
|
|
option.UserName = configuration.GetValue<string>("Cap:RabbitMq:UserName");
|
|
option.Password = configuration.GetValue<string>("Cap:RabbitMq:Password");
|
|
});
|
|
|
|
var hostingEnvironment = context.Services.GetHostingEnvironment();
|
|
bool auth = !hostingEnvironment.IsDevelopment();
|
|
capOptions.UseDashboard(options =>
|
|
{
|
|
options.UseAuth = auth;
|
|
options.AuthorizationPolicy = LionAbpProCapPermissions.CapManagement.Cap;
|
|
});
|
|
});
|
|
}
|
|
else
|
|
{
|
|
context.AddAbpCap(capOptions =>
|
|
{
|
|
capOptions.UseInMemoryStorage();
|
|
capOptions.UseInMemoryMessageQueue();
|
|
var hostingEnvironment = context.Services.GetHostingEnvironment();
|
|
var auth = !hostingEnvironment.IsDevelopment();
|
|
capOptions.UseDashboard(options => { options.UseAuth = auth; });
|
|
});
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// 审计日志
|
|
/// </summary>
|
|
private void ConfigureAuditLog(ServiceConfigurationContext context)
|
|
{
|
|
Configure<AbpAuditingOptions>
|
|
(
|
|
options =>
|
|
{
|
|
options.IsEnabled = true;
|
|
options.EntityHistorySelectors.AddAllEntities();
|
|
options.ApplicationName = "Lion.AbpPro";
|
|
}
|
|
);
|
|
|
|
Configure<AbpAspNetCoreAuditingOptions>(
|
|
options =>
|
|
{
|
|
options.IgnoredUrls.Add("/AuditLogs/page");
|
|
options.IgnoredUrls.Add("/hangfire/stats");
|
|
options.IgnoredUrls.Add("/cap");
|
|
});
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|