Startup.cs 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. using Admin.NET.Core;
  2. using AspNetCoreRateLimit;
  3. using Furion;
  4. using Furion.Logging;
  5. using Furion.SpecificationDocument;
  6. using IGeekFan.AspNetCore.Knife4jUI;
  7. using Microsoft.AspNetCore.Builder;
  8. using Microsoft.AspNetCore.Hosting;
  9. using Microsoft.AspNetCore.HttpOverrides;
  10. using Microsoft.Extensions.DependencyInjection;
  11. using Microsoft.Extensions.Hosting;
  12. using Microsoft.Extensions.Logging;
  13. using NETCore.MailKit.Extensions;
  14. using Newtonsoft.Json;
  15. using Newtonsoft.Json.Serialization;
  16. using OnceMi.AspNetCore.OSS;
  17. using System;
  18. using System.IO;
  19. using Yitter.IdGenerator;
  20. namespace Admin.NET.Web.Core;
  21. public class Startup : AppStartup
  22. {
  23. public void ConfigureServices(IServiceCollection services)
  24. {
  25. // 配置选项
  26. services.AddProjectOptions();
  27. // SqlSugar
  28. services.AddSqlSugar();
  29. // JWT
  30. services.AddJwt<JwtHandler>(enableGlobalAuthorize: true);
  31. // 允许跨域
  32. services.AddCorsAccessor();
  33. // 远程请求
  34. services.AddRemoteRequest();
  35. // 任务调度
  36. services.AddTaskScheduler();
  37. // 脱敏检测
  38. services.AddSensitiveDetection();
  39. // 结果拦截器
  40. services.AddMvcFilter<ResultFilter>();
  41. // 日志监听
  42. services.AddMonitorLogging();
  43. // 第三方授权登录
  44. services.AddAuthentication()
  45. .AddWeixin(options =>
  46. {
  47. var opt = App.GetOptions<OAuthOptions>();
  48. options.ClientId = opt.Weixin.ClientId;
  49. options.ClientSecret = opt.Weixin.ClientSecret;
  50. });
  51. // ElasticSearch
  52. services.AddElasticSearch();
  53. services.AddControllersWithViews()
  54. .AddAppLocalization()
  55. .AddNewtonsoftJson(options =>
  56. {
  57. options.SerializerSettings.ContractResolver = new CamelCasePropertyNamesContractResolver(); // 响应驼峰命名
  58. options.SerializerSettings.DateFormatString = "yyyy-MM-dd HH:mm:ss"; // 时间格式化
  59. options.SerializerSettings.ReferenceLoopHandling = ReferenceLoopHandling.Ignore; // 忽略循环引用
  60. // options.SerializerSettings.NullValueHandling = NullValueHandling.Ignore; // 忽略空值
  61. })
  62. .AddInjectWithUnifyResult<AdminResultProvider>();
  63. // 配置Nginx转发获取客户端真实IP
  64. // 注1:如果负载均衡不是在本机通过 Loopback 地址转发请求的,一定要加上options.KnownNetworks.Clear()和options.KnownProxies.Clear()
  65. // 注2:如果设置环境变量 ASPNETCORE_FORWARDEDHEADERS_ENABLED 为 True,则不需要下面的配置代码
  66. services.Configure<ForwardedHeadersOptions>(options =>
  67. {
  68. options.ForwardedHeaders = ForwardedHeaders.All;
  69. options.KnownNetworks.Clear();
  70. options.KnownProxies.Clear();
  71. });
  72. // 限流服务
  73. services.AddInMemoryRateLimiting();
  74. services.AddSingleton<IRateLimitConfiguration, RateLimitConfiguration>();
  75. // 事件总线
  76. services.AddEventBus(builder =>
  77. {
  78. // 订阅日志事件
  79. builder.AddSubscriber<LogEventSubscriber>();
  80. // 事件执行器(失败重试)
  81. builder.AddExecutor<RetryEventHandlerExecutor>();
  82. });
  83. // OSS对象存储(必须一个个赋值)
  84. var opt = App.GetOptions<OSSProviderOptions>();
  85. services.AddOSSService(opt.ProviderName, options =>
  86. {
  87. options.Provider = opt.Provider;
  88. options.Endpoint = opt.Endpoint;
  89. options.AccessKey = opt.AccessKey;
  90. options.SecretKey = opt.SecretKey;
  91. options.Region = opt.Region;
  92. options.IsEnableCache = opt.IsEnableCache;
  93. options.IsEnableHttps = opt.IsEnableHttps;
  94. });
  95. // 电子邮件
  96. services.AddMailKit(options =>
  97. {
  98. options.UseMailKit(App.GetOptions<EmailOptions>());
  99. });
  100. // Redis缓存
  101. services.AddCSRedisSetup();
  102. // 模板引擎
  103. services.AddViewEngine();
  104. // 即时通讯
  105. services.AddSignalR();
  106. // logo显示
  107. services.AddLogoDisplay();
  108. // 日志写入文件-消息、警告、错误
  109. Array.ForEach(new[] { LogLevel.Information, LogLevel.Warning, LogLevel.Error }, logLevel =>
  110. {
  111. services.AddFileLogging("logs/{0:yyyyMMdd}_{1}.log", options =>
  112. {
  113. options.FileNameRule = fileName => string.Format(fileName, DateTime.Now, logLevel.ToString()); // 每天创建一个文件
  114. options.WriteFilter = logMsg => logMsg.LogLevel == logLevel; // 日志级别
  115. options.FileSizeLimitBytes = 10 * 1024 * 1024; // 每个文件10M
  116. options.MaxRollingFiles = 30; // 只保留30个文件
  117. // 输出Json格式,对接阿里云日志、Elastaicsearch第三方日志
  118. options.MessageFormat = (logMsg) =>
  119. {
  120. return logMsg.WriteArray(writer =>
  121. {
  122. writer.WriteStringValue(DateTime.Now.ToString("o"));
  123. writer.WriteStringValue(logMsg.LogLevel.ToString());
  124. writer.WriteStringValue(logMsg.LogName);
  125. writer.WriteNumberValue(logMsg.EventId.Id);
  126. writer.WriteStringValue(logMsg.Message);
  127. writer.WriteStringValue(logMsg.Exception?.ToString());
  128. });
  129. };
  130. // 写入失败时启用备用文件
  131. options.HandleWriteError = (writeError) =>
  132. {
  133. writeError.UseRollbackFileName(Path.GetFileNameWithoutExtension(writeError.CurrentFileName) + "-oops" + Path.GetExtension(writeError.CurrentFileName));
  134. };
  135. });
  136. });
  137. // 日志写入数据库
  138. services.AddDatabaseLogging<DatabaseLoggingWriter>(options =>
  139. {
  140. options.MinimumLevel = LogLevel.Information;
  141. });
  142. //// 日志写入ElasticSearch
  143. //services.AddDatabaseLogging<ESLoggingWriter>(options =>
  144. //{
  145. // options.MinimumLevel = LogLevel.Information;
  146. //});
  147. // 设置雪花Id算法机器码
  148. YitIdHelper.SetIdGenerator(new IdGeneratorOptions
  149. {
  150. WorkerId = App.GetOptions<SnowIdOptions>().WorkerId
  151. });
  152. }
  153. public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
  154. {
  155. if (env.IsDevelopment())
  156. {
  157. app.UseDeveloperExceptionPage();
  158. app.UseForwardedHeaders();
  159. }
  160. else
  161. {
  162. app.UseExceptionHandler("/Home/Error");
  163. app.UseForwardedHeaders();
  164. app.UseHsts();
  165. }
  166. // 添加状态码拦截中间件
  167. app.UseUnifyResultStatusCodes();
  168. // 配置多语言
  169. app.UseAppLocalization();
  170. // 启用HTTPS
  171. app.UseHttpsRedirection();
  172. app.UseStaticFiles();
  173. app.UseRouting();
  174. app.UseCorsAccessor();
  175. // 限流组件(在跨域之后)
  176. app.UseIpRateLimiting();
  177. app.UseClientRateLimiting();
  178. app.UseAuthentication();
  179. app.UseAuthorization();
  180. // 配置Swagger-Knife4UI(路由前缀一致代表独立,不同则代表共存)
  181. app.UseKnife4UI(options =>
  182. {
  183. options.RoutePrefix = "kapi";
  184. foreach (var groupInfo in SpecificationDocumentBuilder.GetOpenApiGroups())
  185. {
  186. options.SwaggerEndpoint("/" + groupInfo.RouteTemplate, groupInfo.Title);
  187. }
  188. });
  189. app.UseInject();
  190. app.UseEndpoints(endpoints =>
  191. {
  192. // 注册集线器
  193. endpoints.MapHubs();
  194. endpoints.MapControllerRoute(
  195. name: "default",
  196. pattern: "{controller=Home}/{action=Index}/{id?}");
  197. });
  198. }
  199. }