using System.Reflection; using Autofac; using Autofac.Extensions.DependencyInjection; using ce.autofac.extension; using IdentityServer4.AccessTokenValidation; using Infrastructure; using Infrastructure.Cache; using Infrastructure.CloudSdk.minio; using Infrastructure.CloudSdk.mqttmessagecenter; using Infrastructure.Extensions.AutofacManager; using Infrastructure.Helpers; using Infrastructure.Middleware; using Microsoft.AspNetCore.DataProtection; using Microsoft.AspNetCore.Http.Features; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Server.Kestrel.Core; using Microsoft.AspNetCore.StaticFiles; using Microsoft.Extensions.FileProviders; using Microsoft.OpenApi.Models; using Newtonsoft.Json; using Npgsql; using OpenAuth.App; using OpenAuth.App.BaseApp.ImMsgManager; using OpenAuth.App.HostedService; using OpenAuth.App.ServiceApp; using OpenAuth.Repository; using OpenAuth.WebApi.Model; using OpenAuth.WebApi.Model.mqtt; using OpenAuth.WebApi.Model.RabbitMQService; using SqlSugar; using Swashbuckle.AspNetCore.SwaggerUI; using Yitter.IdGenerator; namespace OpenAuth.WebApi { public class Startup { public IConfiguration Configuration { get; } public Startup(IConfiguration configuration) { Configuration = configuration; } public Startup(ConfigurationManager configuration) { Configuration = configuration; } public void ConfigureServices(IServiceCollection services) { services.AddHttpContextAccessor(); services.AddSingleton(); services.AddSingleton(); // minio client services.AddSingleton(_ => new MinioService()); #region log4net //在startup中需要强制创建log4net //var loggerFactory = LoggerFactory.Create(builder => { builder.AddLog4Net(); }); //ILogger logger = loggerFactory.CreateLogger(); services.AddLogging(loggingBuilder => { loggingBuilder.ClearProviders(); loggingBuilder.AddLog4Net(); }); #endregion #region identityServer var identityServer = ((ConfigurationSection)Configuration.GetSection("AppSetting:IdentityServerUrl")).Value; if (!string.IsNullOrEmpty(identityServer)) { services.AddAuthorization(); services.AddAuthentication(IdentityServerAuthenticationDefaults.AuthenticationScheme) .AddJwtBearer(options => { options.Authority = identityServer; options.RequireHttpsMetadata = false; // 指定是否为HTTPS options.Audience = "openauthapi"; }); } #endregion #region MiniProfiler // 添加MiniProfiler服务 //services.AddMiniProfiler(options => //{ // // 设定访问分析结果URL的路由基地址 // options.RouteBasePath = "/profiler"; // options.ColorScheme = StackExchange.Profiling.ColorScheme.Auto; // options.PopupRenderPosition = StackExchange.Profiling.RenderPosition.BottomLeft; // options.PopupShowTimeWithChildren = true; // options.PopupShowTrivial = true; // options.SqlFormatter = new StackExchange.Profiling.SqlFormatters.InlineFormatter(); // // options.IgnoredPaths.Add("/swagger/"); //}).AddEntityFramework(); //显示SQL语句及耗时 #endregion #region swagger //添加swagger services.AddSwaggerGen(option => { #region 注释 //foreach (var controller in GetControllers()) //{ // var groupname = GetSwaggerGroupName(controller); // option.SwaggerDoc(groupname, new OpenApiInfo // { // Version = "v1", // Title = groupname, // Description = "by hopetry" // }); //} //foreach (var name in Directory.GetFiles(AppContext.BaseDirectory, "*.*", // SearchOption.AllDirectories).Where(f => Path.GetExtension(f).ToLower() == ".xml")) //{ // option.IncludeXmlComments(name, includeControllerXmlComments: true); //} #endregion option.CustomSchemaIds(type => type.FullName); option.SwaggerDoc("v1", new OpenApiInfo { Version = "v1", Title = "低空态势感知平台 API", Description = "by hopetry" }); //option.IncludeXmlComments(Path.Combine(AppContext.BaseDirectory, "OpenAuth.WebApi.xml"), true); option.IncludeXmlComments(Path.Combine(AppContext.BaseDirectory, "OpenAuth.WebApi.xml"), true); option.OperationFilter(); // 添加httpHeader参数 }); //services.AddSwaggerGen(option => //{ // option.CustomSchemaIds(type => type.FullName); // option.SwaggerDoc("v1", new OpenApiInfo // { // Version = "v1", // Title = "无人值守 WebAPI", // Description = "by HopetryTech" // }); // option.CustomOperationIds(apiDesc => // { // var controllerAction = apiDesc.ActionDescriptor as ControllerActionDescriptor; // return controllerAction.ControllerName + "-" + controllerAction.ActionName; // }); // option.IncludeXmlComments(Path.Combine(AppContext.BaseDirectory, "OpenAuth.WebApi.xml"), true); // option.OperationFilter(); // 添加httpHeader参数 //}); #endregion #region mqtt services.AddSingleton(); services.AddSingleton(); services.AddSingleton(); services.AddSingleton(); services.AddSingleton(); //services.AddSingleton(); services.AddSingleton(); services.AddHostedService(); #endregion #region rabbitmq services.AddSingleton(); services.AddHostedService(sp => sp.GetRequiredService()); #endregion #region AppSetting services.Configure(Configuration.GetSection("AppSetting")); #endregion #region 限制文件大小 services.Configure(options => { options.Limits.MaxRequestBodySize = long.MaxValue; }); services.Configure(x => { x.ValueLengthLimit = int.MaxValue; x.MultipartBodyLengthLimit = long.MaxValue; x.MemoryBufferThreshold = int.MaxValue; }); #endregion #region Controllers services.AddControllers(option => { option.Filters.Add(); //option.Filters.Add(); //option.Filters.Add(); }) .ConfigureApiBehaviorOptions(options => { // 禁用自动模态验证 // options.SuppressModelStateInvalidFilter = true; //启动WebAPI自动模态验证,处理返回值 options.InvalidModelStateResponseFactory = context => { var problems = new CustomBadRequest(context); return new BadRequestObjectResult(problems); }; }).AddNewtonsoftJson(options => { //忽略循环引用 options.SerializerSettings.ReferenceLoopHandling = ReferenceLoopHandling.Ignore; //不使用驼峰样式的key //options.SerializerSettings.ContractResolver = new DefaultContractResolver(); options.SerializerSettings.DateFormatString = "yyyy-MM-dd HH:mm:ss"; }); #endregion #region MemoryCache services.AddMemoryCache(); services.AddSingleton(); #endregion #region Cors services.AddCors(options => { //options.AddPolicy("CorsPolicy", // builder => builder.SetIsOriginAllowed(origin => true) // .AllowAnyMethod() // .AllowAnyHeader() // .AllowCredentials() // ); options.AddPolicy("SignalR", builder => builder.SetIsOriginAllowed(origin => true) .AllowAnyMethod() .AllowAnyHeader() .AllowCredentials() ); //options.AddPolicy("CorsPolicy", // builder => builder.AllowAnyOrigin() // .AllowAnyMethod() // .AllowAnyHeader()); }); #endregion #region SqlSugar ////没有直接 using SqlSugar,因为如果引用命名空间的话,会和Microsoft的一个GetTypeInfo存在二义性,所以就直接这么使用了 services.AddScoped(s => { StaticConfig.CustomSnowFlakeFunc = () => { return YitIdHelper.NextId(); }; var sqlSugar = new SqlSugarClient(new ConnectionConfig() { DbType = DbType.PostgreSQL, ConnectionString = Configuration.GetConnectionString("OpenAuthDBContext"), IsAutoCloseConnection = true, MoreSettings = new ConnMoreSettings() { PgSqlIsAutoToLower = false, PgSqlIsAutoToLowerCodeFirst = false } }, //var sqlSugar = new SqlSugarClient(new ConnectionConfig() //{ // DbType = SqlSugar.DbType.Kdbndp, // ConnectionString = Configuration.GetConnectionString("OpenAuthDBContext"), // IsAutoCloseConnection = true, // MoreSettings = new SqlSugar.ConnMoreSettings() // { // //PgSqlIsAutoToLower = false, // //PgSqlIsAutoToLowerCodeFirst = false, // IsAutoToUpper = false, // DatabaseModel = DbType.PostgreSQL // } //}, db => { NpgsqlConnection.GlobalTypeMapper.UseNetTopologySuite(); //单例参数配置,所有上下文生效 db.Aop.OnLogExecuting = (sql, pars) => { //获取原生SQL推荐 5.1.4.63 性能OK //UtilMethods.GetNativeSql(sql, pars); //获取无参数化SQL 影响性能只适合调试 //Console.WriteLine(UtilMethods.GetSqlString(DbType.PostgreSQL, sql, pars)); //Console.WriteLine(sql + "\r\n" + db.Utilities.SerializeObject(pars.ToDictionary(it => it.ParameterName, it => it.Value))); //LogHelper.LogInformation(sql + "\r\n" +db.Utilities.SerializeObject(pars.ToDictionary(it => it.ParameterName, it => it.Value))); //Console.WriteLine(); }; }); return sqlSugar; }); services.AddScoped>(s => new SugarUnitOfWork(s.GetService())); #endregion #region HttpClient services.AddHttpClient(); #endregion #region DataProtection services.AddDataProtection().PersistKeysToFileSystem(new DirectoryInfo(Configuration["DataProtection"])); #endregion #region Quartz services.AddHostedService(); #endregion #region SignalR services.AddSignalR(); #endregion } public void ConfigureContainer(ContainerBuilder builder) { AutofacExt.InitAutofac(builder); } // This method gets called by the runtime. Use this method to configure the HTTP request pipeline. //public void Configure(IApplicationBuilder app, IHostEnvironment env, ILogger loggerFactory) public void Configure(IApplicationBuilder app, IHostEnvironment env) { //loggerFactory.AddLog4Net(); IocManager.Instance.Container = app.ApplicationServices.GetAutofacRoot(); if (env.IsDevelopment()) { app.UseDeveloperExceptionPage(); } app.UseMiniProfiler(); //可以访问根目录下面的静态文件 var staticfile = new StaticFileOptions { FileProvider = new PhysicalFileProvider(AppContext.BaseDirectory), OnPrepareResponse = (ctx) => { //可以在这里为静态文件添加其他http头信息,默认添加跨域信息 ctx.Context.Response.Headers["Access-Control-Allow-Origin"] = "*"; }, ContentTypeProvider = new FileExtensionContentTypeProvider( new Dictionary { { ".amr", "audio/AMR" }, { ".mp3", "audio/mpeg" }, { ".mp4", "video/mp4" }, { ".apk", "application/vnd.android.package-archive" }, { ".cpg", "application/vnd.android.package-archive" }, { ".dbf", "application/vnd.android.package-archive" }, { ".shp", "application/vnd.android.package-archive" }, { ".shx", "application/vnd.android.package-archive" }, { ".zip", "application/zip" }, { ".jpeg", "image/jpeg" }, { ".jpg", "image/jpg" }, { ".png", "image/png" }, { ".docx", "application/msword" }, { ".doc", "application/msword" }, { ".txt", "application/octet-stream" }, { ".xlsx", "application/octet-stream" }, { ".xls", "application/vnd.ms-excel" }, { ".pdf", "application/pdf" }, { ".js","application/javascript" }, { ".cjs","application/javascript" }, { ".pt", "application/octet-stream" }, }) }; app.UseStaticFiles(staticfile); //todo:测试可以允许任意跨域,正式环境要加权限 app.UseRouting(); app.UseCors("SignalR"); app.UseAuthentication(); // 启用日志追踪记录和异常友好提示 app.UseLogMiddleware(); //配置ServiceProvider AutofacContainerModule.ConfigServiceProvider(app.ApplicationServices); //app.UseSwagger(); app.UseSwagger(options => { options.PreSerializeFilters.Add((swagger, httpReq) => { if (httpReq.Headers.ContainsKey("X-Request-Uri")) { var index = httpReq.Headers["X-Request-Uri"].ToString().IndexOf("/swagger/"); if (index > 0) { var serverUrl = $"{httpReq.Headers["X-Request-Uri"].ToString().Substring(0, index)}/"; swagger.Servers = new List { new OpenApiServer { Url = serverUrl } }; } } }); }); // Enable middleware to serve swagger-ui (HTML, JS, CSS, etc.), // specifying the Swagger JSON endpoint. app.UseSwaggerUI(c => { //foreach (var controller in GetControllers()) //{ // var groupname = GetSwaggerGroupName(controller); // c.SwaggerEndpoint($"{groupname}/swagger.json", groupname); //} c.SwaggerEndpoint("v1/swagger.json", "V1 Docs"); c.DocExpansion(DocExpansion.List); //默认展开列表 c.OAuthClientId("OpenAuth.WebApi"); //oauth客户端名称 c.OAuthAppName("开源版webapi认证"); // 描述 }); app.UseSwaggerUI(c => { c.SwaggerEndpoint("v1/swagger.json", "V1 Docs"); c.DocExpansion(DocExpansion.None); c.OAuthClientId("OpenAuth.WebApi"); //oauth客户端名称 c.OAuthAppName("开源版webapi认证"); // 描述 }); app.UseEndpoints(endpoints => { endpoints.MapHub("/chathub").RequireCors("SignalR"); // 配置 SignalR Hub 的路由 endpoints.MapControllers(); //endpoints.MapHub("/chathub"); // 配置 SignalR Hub 的路由 // endpoints.MapControllers().RequireCors("CorsPolicy"); }); } /// /// 获取控制器对应的swagger分组值 /// private string GetSwaggerGroupName(Type controller) { var groupname = controller.Name.Replace("Controller", ""); var apisetting = controller.GetCustomAttribute(typeof(ApiExplorerSettingsAttribute)); if (apisetting != null) { groupname = ((ApiExplorerSettingsAttribute)apisetting).GroupName; } return groupname; } /// /// 获取所有的控制器 /// private List GetControllers() { Assembly asm = Assembly.GetExecutingAssembly(); var controlleractionlist = asm.GetTypes() .Where(type => typeof(ControllerBase).IsAssignableFrom(type)) .OrderBy(x => x.Name).ToList(); return controlleractionlist; } } }