ErrorController.cs 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. using FreeRedis;
  2. using Hangfire;
  3. using Masuit.MyBlogs.Core.Common;
  4. using Masuit.MyBlogs.Core.Configs;
  5. using Masuit.MyBlogs.Core.Extensions.Firewall;
  6. using Masuit.Tools.AspNetCore.Mime;
  7. using Masuit.Tools.Core.Validator;
  8. using Masuit.Tools.Logging;
  9. using Microsoft.AspNetCore.Diagnostics;
  10. using Microsoft.AspNetCore.Mvc;
  11. using Microsoft.EntityFrameworkCore;
  12. using Microsoft.Net.Http.Headers;
  13. using System.Diagnostics;
  14. using System.Text;
  15. using System.Web;
  16. using Masuit.MyBlogs.Core.Common.Mails;
  17. using SameSiteMode = Microsoft.AspNetCore.Http.SameSiteMode;
  18. namespace Masuit.MyBlogs.Core.Controllers;
  19. /// <summary>
  20. /// 错误页
  21. /// </summary>
  22. [ApiExplorerSettings(IgnoreApi = true)]
  23. public sealed class ErrorController : Controller
  24. {
  25. public IRedisClient RedisClient { get; set; }
  26. /// <summary>
  27. /// 404
  28. /// </summary>
  29. /// <returns></returns>
  30. [Route("error"), Route("{*url}", Order = 99999), ResponseCache(Duration = 36000)]
  31. public ActionResult Index()
  32. {
  33. Response.StatusCode = 404;
  34. string accept = Request.Headers[HeaderNames.Accept] + "";
  35. return true switch
  36. {
  37. _ when accept.StartsWith("image") => File("/Assets/images/404/4044.jpg", ContentType.Jpeg),
  38. _ when Request.HasJsonContentType() || Request.Method == HttpMethods.Post => Json(new
  39. {
  40. StatusCode = 404,
  41. Success = false,
  42. Message = "页面未找到!"
  43. }),
  44. _ => View("Index")
  45. };
  46. }
  47. /// <summary>
  48. /// 503
  49. /// </summary>
  50. /// <returns></returns>
  51. [Route("ServiceUnavailable")]
  52. public async Task<ActionResult> ServiceUnavailable()
  53. {
  54. var feature = HttpContext.Features.Get<IExceptionHandlerPathFeature>();
  55. if (feature != null)
  56. {
  57. string err;
  58. var ip = HttpContext.Connection.RemoteIpAddress;
  59. switch (feature.Error)
  60. {
  61. case DbUpdateConcurrencyException ex:
  62. err = $"数据库并发更新异常,更新表:{ex.Entries.Select(e => e.Metadata.Name)},请求路径({Request.Method}):{Request.Scheme}://{Request.Host}{HttpUtility.UrlDecode(feature.Path)}{Request.QueryString},客户端用户代理:{Request.Headers[HeaderNames.UserAgent]},客户端IP:{ip}\t{ex.InnerException?.Message},请求参数:\n{await GetRequestBody(Request)}\n堆栈信息:";
  63. LogManager.Error(err, ex.Demystify());
  64. break;
  65. case DbUpdateException ex:
  66. err = $"数据库更新时异常,更新表:{ex.Entries.Select(e => e.Metadata.Name)},请求路径({Request.Method}):{Request.Scheme}://{Request.Host}{HttpUtility.UrlDecode(feature.Path)}{Request.QueryString} ,客户端用户代理:{Request.Headers[HeaderNames.UserAgent]},客户端IP:{ip}\t{ex.InnerException?.Message},请求参数:\n{await GetRequestBody(Request)}\n堆栈信息:";
  67. LogManager.Error(err, ex.Demystify());
  68. break;
  69. case AggregateException ex:
  70. LogManager.Debug("↓↓↓" + ex.Message + "↓↓↓");
  71. ex.Flatten().Handle(e =>
  72. {
  73. LogManager.Error($"异常源:{e.Source},异常类型:{e.GetType().Name},请求路径({Request.Method}):{Request.Scheme}://{Request.Host}{HttpUtility.UrlDecode(feature.Path)}{Request.QueryString} ,客户端用户代理:{Request.Headers[HeaderNames.UserAgent]},客户端IP:{ip}\t", e.Demystify());
  74. return true;
  75. });
  76. var body = await GetRequestBody(Request);
  77. if (!string.IsNullOrEmpty(body))
  78. {
  79. LogManager.Debug("↑↑↑请求参数:\n" + body);
  80. }
  81. break;
  82. case AccessDenyException:
  83. var entry = ip.GetIPLocation();
  84. var tips = Template.Create(CommonHelper.SystemSettings.GetOrAdd("AccessDenyTips", @"<h4>遇到了什么问题?</h4>
  85. <h4>基于主观因素考虑,您所在的地区暂时不允许访问本站,如有疑问,请联系站长!或者请联系站长开通本站的访问权限!</h4>")).Set("clientip", ip.ToString()).Set("location", entry.Address).Set("network", entry.Network).Render();
  86. Response.StatusCode = 403;
  87. return View("AccessDeny", tips);
  88. case TempDenyException:
  89. Response.StatusCode = 429;
  90. return Request.HasJsonContentType() || Request.Method == HttpMethods.Post ? Json(new
  91. {
  92. StatusCode = 429,
  93. Success = false,
  94. Message = $"检测到您的IP({ip})访问过于频繁,已被本站暂时禁止访问,请稍后再试!"
  95. }) : View("TempDeny");
  96. default:
  97. LogManager.Error($"异常源:{feature.Error.Source},异常类型:{feature.Error.GetType().Name},请求路径({Request.Method}):{Request.Scheme}://{Request.Host}{HttpUtility.UrlDecode(feature.Path)}{Request.QueryString} ,客户端用户代理:{Request.Headers[HeaderNames.UserAgent]},客户端IP:{ip},请求参数:\n{await GetRequestBody(Request)}\n堆栈信息:", feature.Error.Demystify());
  98. break;
  99. }
  100. }
  101. Response.StatusCode = 503;
  102. return Request.HasJsonContentType() || Request.Method == HttpMethods.Post ? Json(new
  103. {
  104. StatusCode = 503,
  105. Success = false,
  106. Message = "服务器发生错误!"
  107. }) : View();
  108. }
  109. private static async Task<string> GetRequestBody(HttpRequest req)
  110. {
  111. if (req.ContentLength > 5120)
  112. {
  113. return "请求体超长";
  114. }
  115. req.Body.Seek(0, SeekOrigin.Begin);
  116. using var sr = new StreamReader(req.Body, Encoding.UTF8, false);
  117. var body = await sr.ReadToEndAsync();
  118. body = HttpUtility.UrlDecode(body);
  119. req.Body.Position = 0;
  120. return body;
  121. }
  122. /// <summary>
  123. /// 网站升级中
  124. /// </summary>
  125. /// <returns></returns>
  126. [Route("ComingSoon"), ResponseCache(Duration = 360000)]
  127. public ActionResult ComingSoon()
  128. {
  129. return View();
  130. }
  131. /// <summary>
  132. /// 检查访问密码
  133. /// </summary>
  134. /// <param name="email"></param>
  135. /// <param name="token"></param>
  136. /// <returns></returns>
  137. [HttpPost, ValidateAntiForgeryToken, AllowAccessFirewall]
  138. public ActionResult CheckViewToken(string email, string token)
  139. {
  140. if (string.IsNullOrEmpty(token))
  141. {
  142. return ResultData(null, false, "请输入访问密码!");
  143. }
  144. var s = RedisClient.Get("token:" + email);
  145. if (!token.Equals(s))
  146. {
  147. return ResultData(null, false, "访问密码不正确!");
  148. }
  149. Response.Cookies.Append("Email", email, new CookieOptions
  150. {
  151. Expires = DateTime.Now.AddYears(1),
  152. SameSite = SameSiteMode.Lax
  153. });
  154. Response.Cookies.Append("FullAccessToken", email.MDString(AppConfig.BaiduAK), new CookieOptions
  155. {
  156. Expires = DateTime.Now.AddYears(1),
  157. SameSite = SameSiteMode.Lax
  158. });
  159. return ResultData(null);
  160. }
  161. /// <summary>
  162. /// 检查授权邮箱
  163. /// </summary>
  164. /// <param name="userInfoService"></param>
  165. /// <param name="email"></param>
  166. /// <returns></returns>
  167. [HttpPost, ValidateAntiForgeryToken, AllowAccessFirewall, ResponseCache(Duration = 100, VaryByQueryKeys = new[] { "email" })]
  168. public ActionResult GetViewToken([FromServices] IUserInfoService userInfoService, string email)
  169. {
  170. var validator = new IsEmailAttribute();
  171. if (!validator.IsValid(email))
  172. {
  173. return ResultData(null, false, validator.ErrorMessage);
  174. }
  175. if (RedisClient.Exists("get:" + email))
  176. {
  177. RedisClient.Expire("get:" + email, 120);
  178. return ResultData(null, false, "发送频率限制,请在2分钟后重新尝试发送邮件!请检查你的邮件,若未收到,请检查你的邮箱地址或邮件垃圾箱!");
  179. }
  180. if (!userInfoService.Any(b => b.Email == email))
  181. {
  182. return ResultData(null, false, "您目前没有权限访问这个链接,请联系站长开通访问权限!");
  183. }
  184. var token = SnowFlake.GetInstance().GetUniqueShortId(6);
  185. RedisClient.Set("token:" + email, token, 86400);
  186. BackgroundJob.Enqueue<IMailSender>(sender => sender.Send(Request.Host + "博客访问验证码", $"{Request.Host}本次验证码是:<span style='color:red'>{token}</span>,有效期为24h,请按时使用!", email, HttpContext.Connection.RemoteIpAddress.ToString()));
  187. RedisClient.Set("get:" + email, token, 120);
  188. return ResultData(null);
  189. }
  190. /// <summary>
  191. /// 响应数据
  192. /// </summary>
  193. /// <param name="data">数据</param>
  194. /// <param name="success">响应状态</param>
  195. /// <param name="message">响应消息</param>
  196. /// <returns></returns>
  197. public ActionResult ResultData(object data, bool success = true, string message = "")
  198. {
  199. return Ok(new
  200. {
  201. Success = success,
  202. Message = message,
  203. Data = data
  204. });
  205. }
  206. }