Handler.cs 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. using SharedKernel.Extensions;
  2. using Application.Abstractions.Data;
  3. using MediatR;
  4. using Microsoft.EntityFrameworkCore;
  5. namespace Application.Features.Member.Logs.Login.Search;
  6. public sealed class Handler(IAppDbContext db) : IRequestHandler<Query, Response>
  7. {
  8. public async Task<Response> Handle(Query request, CancellationToken ct)
  9. {
  10. var query = db.MemberLoginLog.AsNoTracking().Include(x => x.Member).AsQueryable();
  11. if (!string.IsNullOrWhiteSpace(request.Keyword))
  12. {
  13. query = request.Search switch
  14. {
  15. 1 => query.Where(x => x.MemberID.HasValue && x.MemberID.Value.ToString().Contains(request.Keyword)),
  16. 2 => query.Where(x => x.Member != null && x.Member.Email.Contains(request.Keyword)),
  17. 3 => query.Where(x => x.Member != null && x.Member.Name != null && x.Member.Name.Contains(request.Keyword)),
  18. _ => query
  19. };
  20. }
  21. if (!string.IsNullOrWhiteSpace(request.StartAt) && DateTime.TryParse(request.StartAt, out var startAt))
  22. {
  23. query = query.Where(x => x.CreatedAt >= startAt);
  24. }
  25. if (!string.IsNullOrWhiteSpace(request.EndAt) && DateTime.TryParse(request.EndAt, out var endAt))
  26. {
  27. query = query.Where(x => x.CreatedAt <= endAt);
  28. }
  29. var total = await query.CountAsync(ct);
  30. var skip = (request.PageNum - 1) * request.PerPage;
  31. var list = await query
  32. .OrderByDescending(x => x.ID)
  33. .Skip(skip)
  34. .Take(request.PerPage)
  35. .Select(x => new
  36. {
  37. x.ID,
  38. x.MemberID,
  39. MemberName = x.Member != null ? x.Member.Name : null,
  40. x.Account,
  41. x.Success,
  42. x.Reason,
  43. x.IpAddress,
  44. x.UserAgent,
  45. x.CreatedAt
  46. })
  47. .ToListAsync(ct);
  48. var rows = list.Select((x, idx) => new Response.Row
  49. {
  50. Num = total - skip - idx,
  51. ID = x.ID,
  52. MemberID = x.MemberID,
  53. MemberName = x.MemberName,
  54. Account = x.Account,
  55. Success = x.Success,
  56. Reason = x.Reason,
  57. IpAddress = x.IpAddress,
  58. UserAgent = x.UserAgent,
  59. Browser = ParseUserAgent(x.UserAgent, "browser"),
  60. OS = ParseUserAgent(x.UserAgent, "os"),
  61. Device = ParseUserAgent(x.UserAgent, "device"),
  62. CreatedAt = x.CreatedAt
  63. }).ToList();
  64. return new Response
  65. {
  66. Total = total,
  67. List = rows
  68. };
  69. }
  70. private static string? ParseUserAgent(string? userAgent, string type)
  71. {
  72. if (string.IsNullOrWhiteSpace(userAgent))
  73. {
  74. return null;
  75. }
  76. return type switch
  77. {
  78. "browser" => ExtractBrowser(userAgent),
  79. "os" => ExtractOS(userAgent),
  80. "device" => ExtractDevice(userAgent),
  81. _ => null
  82. };
  83. }
  84. private static string ExtractBrowser(string ua)
  85. {
  86. if (ua.Contains("Edg/")) return "Edge";
  87. if (ua.Contains("Chrome/")) return "Chrome";
  88. if (ua.Contains("Firefox/")) return "Firefox";
  89. if (ua.Contains("Safari/") && !ua.Contains("Chrome")) return "Safari";
  90. if (ua.Contains("MSIE") || ua.Contains("Trident/")) return "IE";
  91. return "Unknown";
  92. }
  93. private static string ExtractOS(string ua)
  94. {
  95. if (ua.Contains("Windows NT 10")) return "Windows 10";
  96. if (ua.Contains("Windows NT 6.3")) return "Windows 8.1";
  97. if (ua.Contains("Windows NT 6.1")) return "Windows 7";
  98. if (ua.Contains("Windows")) return "Windows";
  99. if (ua.Contains("Mac OS X")) return "macOS";
  100. if (ua.Contains("Android")) return "Android";
  101. if (ua.Contains("iPhone") || ua.Contains("iPad")) return "iOS";
  102. if (ua.Contains("Linux")) return "Linux";
  103. return "Unknown";
  104. }
  105. private static string ExtractDevice(string ua)
  106. {
  107. if (ua.Contains("Mobile") || ua.Contains("Android") && !ua.Contains("Tablet")) return "Mobile";
  108. if (ua.Contains("Tablet") || ua.Contains("iPad")) return "Tablet";
  109. return "Desktop";
  110. }
  111. }