Startup.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Security.Claims;
  5. using System.Text;
  6. using System.Text.Encodings.Web;
  7. using System.Text.Unicode;
  8. using HaBookCms.AzureCosmos.ServiceExtension;
  9. using HaBookCms.AzureStorage.ServiceExtension;
  10. using HaBookCms.Common.LogHelper;
  11. using HaBookCms.ContextConfig.Attributes;
  12. using HaBookCms.ContextConfig.Exceptions;
  13. using HaBookCms.Jwt.Filter;
  14. using HaBookCms.Jwt.Model;
  15. using HaBookCms.RedisStorage.Cache;
  16. using HaBookCms.ServiceOptions.Options;
  17. using log4net;
  18. using log4net.Config;
  19. using log4net.Repository;
  20. using Microsoft.AspNetCore.Authentication.JwtBearer;
  21. using Microsoft.AspNetCore.Builder;
  22. using Microsoft.AspNetCore.Hosting;
  23. using Microsoft.AspNetCore.Http;
  24. using Microsoft.AspNetCore.HttpOverrides;
  25. using Microsoft.AspNetCore.Mvc;
  26. using Microsoft.Extensions.Configuration;
  27. using Microsoft.Extensions.DependencyInjection;
  28. using Microsoft.IdentityModel.Tokens;
  29. namespace HaBookCms.Contest
  30. {
  31. public class Startup
  32. {
  33. /// <summary>
  34. /// log4net 仓储库
  35. /// </summary>
  36. public static ILoggerRepository repository { get; set; }
  37. public Startup(IConfiguration configuration, IHostingEnvironment env)
  38. {
  39. Configuration = configuration;
  40. var builder = new ConfigurationBuilder()
  41. .SetBasePath(env.ContentRootPath)
  42. .AddJsonFile("appsettings.json", optional: true, reloadOnChange: true);
  43. this.Configuration = builder.Build();
  44. //log4net
  45. repository = LogManager.CreateRepository("HaBookCms");
  46. //指定配置文件
  47. XmlConfigurator.Configure(repository, new FileInfo("Log4net.config"));
  48. BaseConfigModel.SetBaseConfig(Configuration, env.ContentRootPath, env.WebRootPath);
  49. }
  50. public IConfiguration Configuration { get; }
  51. // This method gets called by the runtime. Use this method to add services to the container.
  52. public void ConfigureServices(IServiceCollection services)
  53. {//Configuration["AppSettings:Azure:TableStorageConnection"]
  54. services.AddAzureTableStorage().AddAzureBlobStorage().AddConnection(new AzureStorageOptions
  55. {
  56. ConnectionString = Configuration["AppSettings:Azure:StorageConnection"]
  57. });
  58. services.Configure<CookiePolicyOptions>(options =>
  59. {
  60. // This lambda determines whether user consent for non-essential cookies is needed for a given request.
  61. options.CheckConsentNeeded = context => true;
  62. options.MinimumSameSitePolicy = SameSiteMode.None;
  63. });
  64. services.AddAzureCosmosDB().AddCosmosDBConnection(new AzureCosmosDBOptions() {
  65. ConnectionString = Configuration["AppSettings:Azure:CosmosDBConnection:AccountEndpoint"],
  66. ConnectionKey= Configuration["AppSettings:Azure:CosmosDBConnection:AccountKey"],
  67. Database = Configuration["AppSettings:Azure:CosmosDBConnection:Database"],
  68. });
  69. services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);
  70. //services.AddMvc().AddMvcOptions(
  71. // option =>
  72. // {
  73. // option.OutputFormatters.Clear();
  74. // option.OutputFormatters.Add(new MessagePackOutputFormatter(ContractlessStandardResolver.Instance));
  75. // option.InputFormatters.Clear();
  76. // option.InputFormatters.Add(new MessagePackInputFormatter(ContractlessStandardResolver.Instance));
  77. // });
  78. //解决视图输出内容中文编码问题
  79. services.AddSingleton(HtmlEncoder.Create(UnicodeRanges.All));
  80. services.AddSingleton<IHttpContextAccessor, HttpContextAccessor>();
  81. #region 认证
  82. JwtAuthConfigModel jwtConfig = new JwtAuthConfigModel();
  83. // services.AddAuthentication(CookieAuthenticationDefaults.AuthenticationScheme)
  84. //.AddCookie(CookieAuthenticationDefaults.AuthenticationScheme, o =>
  85. // {
  86. // o.LoginPath = new PathString("/api/Users/checkLogin");
  87. // })
  88. services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
  89. .AddJwtBearer(JwtBearerDefaults.AuthenticationScheme, o =>
  90. {
  91. o.TokenValidationParameters = new TokenValidationParameters
  92. {
  93. ValidIssuer = jwtConfig.Issuer,
  94. ValidAudience = jwtConfig.Audience,
  95. IssuerSigningKey = new SymmetricSecurityKey(Encoding.ASCII.GetBytes(jwtConfig.JWTSecretKey)),
  96. /***********************************TokenValidationParameters的参数默认值***********************************/
  97. RequireSignedTokens = true,
  98. // SaveSigninToken = false,
  99. // ValidateActor = false,
  100. // 将下面两个参数设置为false,可以不验证Issuer和Audience,但是不建议这样做。
  101. ValidateAudience = true,
  102. ValidateIssuer = true,
  103. ValidateIssuerSigningKey = true,
  104. // 是否要求Token的Claims中必须包含 Expires
  105. RequireExpirationTime = true,
  106. // 允许的服务器时间偏移量
  107. // ClockSkew = TimeSpan.FromSeconds(300),
  108. ClockSkew = TimeSpan.Zero,
  109. // 是否验证Token有效期,使用当前时间与Token的Claims中的NotBefore和Expires对比
  110. ValidateLifetime = true
  111. };
  112. });
  113. #endregion
  114. var signingCredentials = new SigningCredentials(new SymmetricSecurityKey(Encoding.ASCII.GetBytes(jwtConfig.JWTSecretKey)), SecurityAlgorithms.HmacSha256);
  115. // 如果要数据库动态绑定,这里先留个空,后边处理器里动态赋值
  116. var permission = new List<Permission>();
  117. // 角色与接口的权限要求参数
  118. var permissionRequirement = new PermissionRequirement(
  119. "/api/denied",// 拒绝授权的跳转地址(目前无用)
  120. permission,
  121. ClaimTypes.Role,//基于角色的授权
  122. jwtConfig.Issuer,//发行人
  123. jwtConfig.Audience,//听众
  124. signingCredentials,//签名凭据
  125. expiration: TimeSpan.FromSeconds(60 * 2)//接口的过期时间
  126. );
  127. // services.AddSingleton<PermissionRequirement, PermissionRequirement>();
  128. #region 授权
  129. services.AddAuthorization(options =>
  130. {
  131. options.AddPolicy("RequireContestWeb", policy => policy.RequireRole("ContestWeb").Build());
  132. options.AddPolicy("RequireApp", policy => policy.RequireRole("App").Build());
  133. options.AddPolicy("RequireAdmin", policy => policy.RequireRole("Admin").Build());
  134. options.AddPolicy("RequireAdminOrApp", policy => policy.RequireRole("Admin,App").Build());
  135. // 自定义权限要求
  136. options.AddPolicy("Permission",
  137. policy => policy.Requirements.Add(permissionRequirement));
  138. });
  139. #endregion
  140. //log日志注入
  141. services.AddSingleton<ILoggerHelper, LogHelper>();
  142. #region 缓存 读取配置是否使用哪种缓存模式
  143. services.AddMemoryCache();
  144. if (Convert.ToBoolean(Configuration["Cache:IsUseRedis"]))
  145. {
  146. services.AddSingleton<ICacheService, RedisCacheService>();
  147. }
  148. else
  149. {
  150. services.AddSingleton<ICacheService, MemoryCacheService>();
  151. }
  152. #endregion
  153. #region 缓存 RedisCache
  154. //将Redis分布式缓存服务添加到服务中
  155. services.AddDistributedRedisCache(options =>
  156. {
  157. //用于连接Redis的配置
  158. options.Configuration = "localhost";// Configuration.GetConnectionString("RedisConnectionString");
  159. //Redis实例名RedisDistributedCache
  160. options.InstanceName = "RedisInstance";
  161. });
  162. #endregion
  163. #region CORS
  164. services.AddCors(c =>
  165. {
  166. c.AddPolicy("Any", policy =>
  167. {
  168. policy.AllowAnyOrigin()
  169. .AllowAnyMethod()
  170. .AllowAnyHeader()
  171. .AllowCredentials();
  172. });
  173. c.AddPolicy("Limit", policy =>
  174. {
  175. policy
  176. .WithOrigins("localhost:63969")
  177. .WithMethods("get", "post", "put", "delete")
  178. //.WithHeaders("Authorization");
  179. .AllowAnyHeader();
  180. });
  181. });
  182. #endregion
  183. #region 性能 压缩
  184. services.AddResponseCompression();
  185. #endregion
  186. //跨域//设置了允许所有来源
  187. services.AddCors(options =>
  188. options.AddPolicy("any",
  189. builder => builder.AllowAnyMethod().AllowAnyHeader().AllowAnyOrigin().AllowCredentials()));
  190. services.AddMvc(options =>
  191. {
  192. options.Filters.Add(new AllowCorsAttribute());
  193. });
  194. }
  195. // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
  196. public void Configure(IApplicationBuilder app, IHostingEnvironment env)
  197. {
  198. if (env.IsDevelopment())
  199. {
  200. app.UseDeveloperExceptionPage();
  201. //app.UseWebpackDevMiddleware(new WebpackDevMiddlewareOptions
  202. //{
  203. // HotModuleReplacement = true
  204. //});
  205. }
  206. else
  207. {
  208. app.UseExceptionHandler("/Home/Error");
  209. // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
  210. app.UseHsts();
  211. }
  212. #region 解决Ubuntu Nginx 代理不能获取IP问题
  213. app.UseForwardedHeaders(new ForwardedHeadersOptions
  214. {
  215. ForwardedHeaders = ForwardedHeaders.XForwardedFor | ForwardedHeaders.XForwardedProto
  216. });
  217. #endregion
  218. app.UseHttpsRedirection();
  219. app.UseStaticFiles(new StaticFileOptions
  220. {
  221. ServeUnknownFileTypes = true
  222. //ContentTypeProvider = new FileExtensionContentTypeProvider(new Dictionary<string, string>
  223. //{
  224. // { ".apk","application/vnd.android.package-archive"},
  225. // { ".nupkg","application/zip"}
  226. //}) //支持特殊文件下载处理
  227. });
  228. //自定义异常处理
  229. app.UseMiddleware<ExceptionFilter>();
  230. //认证
  231. app.UseAuthentication();
  232. //授权
  233. app.UseMiddleware<JwtAuthorizationFilter>();
  234. //性能压缩
  235. app.UseResponseCompression();
  236. app.UseCookiePolicy();
  237. app.UseMvc(routes =>
  238. {
  239. routes.MapRoute(
  240. name: "default",
  241. template: "{controller=Home}/{action=Index}/{id?}");
  242. routes.MapSpaFallbackRoute(
  243. name: "spa-fallback",
  244. defaults: new { controller = "Home", action = "Index" });
  245. });
  246. }
  247. }
  248. }