Startup.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. using Lib.AspNetCore.ServerSentEvents;
  2. using Microsoft.AspNetCore.Authentication.JwtBearer;
  3. using Microsoft.AspNetCore.Builder;
  4. using Microsoft.AspNetCore.Hosting;
  5. using Microsoft.AspNetCore.Http;
  6. using Microsoft.AspNetCore.Mvc;
  7. using Microsoft.AspNetCore.SpaServices;
  8. using Microsoft.Extensions.Configuration;
  9. using Microsoft.Extensions.DependencyInjection;
  10. using Microsoft.Extensions.Hosting;
  11. using Microsoft.IdentityModel.Tokens;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.IdentityModel.Tokens.Jwt;
  15. using System.Linq;
  16. using System.Threading.Tasks;
  17. using TEAMModelOS.SDK.DI.Multiple;
  18. using TEAMModelBI.Models;
  19. using TEAMModelOS.Models;
  20. using TEAMModelOS.SDK;
  21. using TEAMModelOS.SDK.DI;
  22. using TEAMModelOS.SDK.Extension;
  23. using TEAMModelOS.SDK.Filter;
  24. using TEAMModelOS.SDK.Helper.Common.ReflectorExtensions;
  25. using TEAMModelOS.SDK.Models;
  26. using VueCliMiddleware;
  27. using System.Net.Http;
  28. using TEAMModelOS.Filter;
  29. using TEAMModelOS.SDK.DI.IPIP;
  30. namespace TEAMModelBI
  31. {
  32. public class Startup
  33. {
  34. public IWebHostEnvironment environment { get; set; }
  35. readonly string MyAllowSpecificOrigins = "_myAllowSpecificOrigins";
  36. public Startup(IConfiguration configuration,IWebHostEnvironment env)
  37. {
  38. Configuration = configuration;
  39. environment = env;
  40. }
  41. public IConfiguration Configuration { get; }
  42. // This method gets called by the runtime. Use this method to add services to the container.
  43. public void ConfigureServices(IServiceCollection services)
  44. {
  45. // true,默認情況下,聲明映射將以舊格式映射聲明名稱,以適應較早的SAML應用程序,RoleClaimType = 'http://schemas.microsoft.com/ws/2008/06/identity/claims/role'
  46. // false,RoleClaimType = 'roles'
  47. JwtSecurityTokenHandler.DefaultMapInboundClaims = false;
  48. services.AddAuthentication(options => options.DefaultScheme = JwtBearerDefaults.AuthenticationScheme)
  49. .AddJwtBearer(options => //AzureADJwtBearer
  50. {
  51. //options.SaveToken = true; //驗證令牌由服務器生成才有效,不適用於服務重啟或分布式架構
  52. options.Authority = Configuration["Option:Authority"];
  53. options.Audience = Configuration["Option:Audience"];
  54. options.RequireHttpsMetadata = true;
  55. options.TokenValidationParameters = new TokenValidationParameters
  56. {
  57. RoleClaimType = "roles",
  58. ValidAudiences = new string[] { Configuration["Option:Audience"], $"api://{Configuration["Option:Audience"]}" }
  59. };
  60. options.Events = new JwtBearerEvents();
  61. //下列事件有需要紀錄則打開
  62. //options.Events.OnMessageReceived = async context => { await Task.FromResult(0); };
  63. //options.Events.OnForbidden = async context => { await Task.FromResult(0); };
  64. //options.Events.OnChallenge = async context => { await Task.FromResult(0); };
  65. //options.Events.OnAuthenticationFailed = async context => { await Task.FromResult(0); };
  66. options.Events.OnTokenValidated = async context =>
  67. {
  68. if (!context.Principal.Claims.Any(x => x.Type.Equals("http://schemas.microsoft.com/identity/claims/scope")) //ClaimConstants.Scope
  69. && !context.Principal.Claims.Any(y => y.Type.Equals("roles"))) //ClaimConstants.Roles //http://schemas.microsoft.com/ws/2008/06/identity/claims/role
  70. {
  71. //TODO 需處理額外授權非角色及範圍的訪問異常紀錄
  72. throw new UnauthorizedAccessException("Neither scope or roles claim was found in the bearer token.");
  73. }
  74. await Task.FromResult(0);
  75. };
  76. });
  77. //設定跨域請求
  78. services.AddCors(options =>
  79. {
  80. options.AddPolicy(MyAllowSpecificOrigins,
  81. builder =>
  82. {
  83. builder.WithOrigins("http://teammodelos-test.chinacloudsites.cn",
  84. "https://www.teammodel.cn", "https://localhost:5001",
  85. "http://localhost:5000")
  86. .AllowAnyHeader()
  87. .AllowAnyMethod();
  88. });
  89. });
  90. //Table和blob注入
  91. List<(string name, string connectionString)> storageConnects = new();
  92. storageConnects.Add(("Default", Configuration.GetValue<string>("Azure:Storage:ConnectionString"))); //大路站ClientString
  93. storageConnects.Add(("Global", Configuration.GetValue<string>("GlobalAzure:Storage:ConnectionString"))); //国际站ClientString
  94. storageConnects.Add(("LogChina", Configuration.GetValue<string>("Azure:LogStorage:ConnectionString"))); //防火墙日志 大陆站ClientString
  95. storageConnects.Add(("LogGlobal", Configuration.GetValue<string>("GlobalAzure:LogStorage:ConnectionString"))); //防火墙日志 国际站ClientString
  96. storageConnects.Add(("CoreServiceV2", Configuration.GetValue<string>("CoreServiceV2:Storage:ConnectionString"))); //CoreService V2
  97. services.AddMultipleAzureStorage(storageConnects);
  98. //cosmosDB注入
  99. List<(string name,string connectionString)> cosmosDBConnects = new();
  100. cosmosDBConnects.Add(("Default", Configuration.GetValue<string>("Azure:Cosmos:ConnectionString"))); //大陆站ConnectString
  101. cosmosDBConnects.Add(("Global", Configuration.GetValue<string>("GlobalAzure:Cosmos:ConnectionString"))); //国际站ConnectString
  102. cosmosDBConnects.Add(("CoreServiceV1", Configuration.GetValue<string>("CoreServiceV1:Cosmos:ConnectionString"))); //CoreService V1 read only
  103. cosmosDBConnects.Add(("CoreServiceV2", Configuration.GetValue<string>("CoreServiceV2:Cosmos:ConnectionString"))); //CoreService V2
  104. cosmosDBConnects.Add(("CoreServiceV2CnRead", Configuration.GetValue<string>("CoreServiceV2:CosmosCnRead:ConnectionString"))); //CoreService V2 read only
  105. services.AddMultipleAzureCosmos(cosmosDBConnects);
  106. //redis注入
  107. List<(string name, string connectionString)> redisConnects = new();
  108. redisConnects.Add(("Default", Configuration.GetValue<string>("Azure:Redis:ConnectionString")));
  109. redisConnects.Add(("Global", Configuration.GetValue<string>("GlobalAzure:Redis:ConnectionString")));
  110. services.AddMultipleAzureRedis(redisConnects);
  111. //serverBus 注入
  112. List<(string name, string connectionString)> funConnects = new();
  113. funConnects.Add(("Default", Configuration.GetValue<string>("Azure:ServiceBus:ConnectionString")));
  114. funConnects.Add(("Global", Configuration.GetValue<string>("GlobalAzure:ServiceBus:ConnectionString")));
  115. services.AddMultipleAzureServiceBus(funConnects);
  116. //单一注入
  117. //services.AddAzureStorage(Configuration.GetValue<string>("Azure:Storage:ConnectionString"));
  118. //services.AddAzureCosmos(Configuration.GetValue<string>("Azure:Cosmos:ConnectionString"));
  119. //services.AddAzureRedis(Configuration.GetValue<string>("Azure:Redis:ConnectionString"));
  120. //services.AddAzureServiceBus(Configuration.GetValue<string>("Azure:ServiceBus:ConnectionString"));
  121. services.AddSnowflakeId(Convert.ToInt64(Configuration.GetValue<string>("Option:LocationNum")), 1);
  122. services.AddHttpClient();
  123. services.AddHttpClient<DingDing>();
  124. services.AddSingleton(new City(@"Services/ipip.ipdb"));
  125. //services.AddCoreAPIHttpService(Configuration);
  126. services.AddHttpClient<CoreAPIHttpService>().ConfigureHttpMessageHandlerBuilder(builder =>
  127. {
  128. builder.PrimaryHandler = new HttpClientHandler
  129. {
  130. ServerCertificateCustomValidationCallback = (message, cert, chain, errors) => true
  131. };
  132. });
  133. services.AddHttpClient<HttpTrigger>();
  134. services.AddMemoryCache();
  135. services.AddControllers().AddJsonOptions(options => { options.JsonSerializerOptions.IgnoreNullValues = false; });
  136. services.Configure<SysConfig>(Configuration.GetSection("sysConfig")); //注册连接
  137. //HttpContextAccessor,并用来访问HttpContext。(提供組件或非控制器服務存取HttpContext)
  138. services.AddHttpContextAccessor();
  139. services.Configure<Option>(options => Configuration.GetSection("Option").Bind(options));
  140. services.AddControllers();
  141. string path = $"{ environment.ContentRootPath}/JsonFile/Core";
  142. services.AddIPSearcher(path);
  143. services.AddSpaStaticFiles(configuration =>
  144. {
  145. configuration.RootPath = "ClientApp";
  146. });
  147. services.AddServerSentEvents(o =>
  148. {
  149. o.KeepaliveMode = ServerSentEventsKeepaliveMode.Always;
  150. o.OnClientConnected = async (service, client) =>
  151. {
  152. //if (client.Request.Headers.TryGetValue("X-Auth-Name", out StringValues name))
  153. //{
  154. // client.Client.SetProperty("NAME", name.ToString());
  155. //}
  156. //if (client.Request.Headers.TryGetValue("X-Auth-DID", out StringValues did))
  157. //{
  158. // client.Client.SetProperty("DID", did.ToString());
  159. //}
  160. //if (client.Request.Headers.TryGetValue("X-Auth-CID", out StringValues cid))
  161. //{
  162. // client.Client.SetProperty("CID", cid.ToString());
  163. //}
  164. //if (client.Request.Headers.TryGetValue("X-Auth-PIN", out StringValues pin))
  165. //{
  166. // client.Client.SetProperty("PIN", pin.ToString());
  167. //}
  168. //if (client.Request.Headers.TryGetValue("X-Auth-APP", out StringValues app))
  169. //{
  170. // client.Client.SetProperty("APP", app.ToString());
  171. //}
  172. await client.Client.SendEventAsync(new { sid = client.Client.Id.ToString() }.ToJsonString());
  173. };
  174. });
  175. //等保安全性验证。
  176. services.AddScoped<SecurityHeadersAttribute>();
  177. services.AddMvcFilter<RequestAuditFilter>();
  178. services.AddAntiforgery(options => options.HeaderName = "X-CSRF-TOKEN");
  179. }
  180. // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
  181. public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
  182. {
  183. if (env.IsDevelopment())
  184. {
  185. app.UseDeveloperExceptionPage();
  186. }
  187. app.UseRouting();
  188. //以下需要按照順序載入中間件 如果应用调用 UseStaticFiles,请将 UseStaticFiles 置于 UseRouting之前。
  189. app.UseStaticFiles();
  190. app.UseSpaStaticFiles();
  191. app.UseCors(MyAllowSpecificOrigins); //使用跨域設定
  192. app.UseHttpsRedirection(); //開發中暫時關掉
  193. //如果应用使用身份验证/授权功能(如 AuthorizePage 或 [Authorize]),请将对 UseAuthentication 和 UseAuthorization的
  194. //调用放在之后、UseRouting 和 UseCors,但在 UseEndpoints之前
  195. app.UseAuthentication();
  196. app.UseAuthorization();
  197. app.UseEndpoints(endpoints =>
  198. {
  199. endpoints.MapControllers();
  200. endpoints.MapServerSentEvents("/service/sse", new ServerSentEventsOptions
  201. {
  202. //Authorization = ServerSentEventsAuthorization.Default,
  203. OnPrepareAccept = response =>
  204. {
  205. response.Headers.Append("Cache-Control", "no-cache");
  206. response.Headers.Append("X-Accel-Buffering", "no");
  207. response.Headers.Append("Content-Type", "text/event-stream");
  208. response.Headers.Append("Access-Control-Allow-Origin", "*");
  209. }
  210. });
  211. #if DEBUG
  212. endpoints.MapToVueCliProxy(
  213. "{*path}",
  214. new SpaOptions { SourcePath = "ClientApp" },
  215. npmScript: (System.Diagnostics.Debugger.IsAttached) ? "serve" : null,
  216. // regex: "Compiled successfully",
  217. forceKill: true
  218. );
  219. #else
  220. endpoints.MapFallbackToFile("index.html");
  221. #endif
  222. });
  223. }
  224. }
  225. }