5053245575
**Architecture Refactor (SOLID Principles):** - Implement AdminDashboardController (REST API) - Add dashboard summary endpoint - Add upcoming filings endpoint - Add recent inquiries endpoint - Add monthly statistics endpoint **Database Layer (Repository Pattern):** - Extend IInquiryRepository with date range queries - Implement CountByDateRangeAsync - Implement CountByStatusAndDateAsync - Extend InquiryRepository with new methods **Service Layer (Single Responsibility):** - Extend AdminDashboardService with API methods - Add GetRecentInquiriesAsync - Add GetMonthlyStatsAsync with caching **Test Coverage:** - Update FakeInquiryRepository mock with new methods **SOLID Application:** ✓ Single Responsibility: Each class has one reason to change ✓ Open/Closed: Dashboard API can be extended without modifying existing code ✓ Dependency Inversion: Service depends on Repository abstraction ✓ Interface Segregation: API endpoints are focused and specific Status: ✓ Compiles successfully (0 errors, 0 warnings) Next phases: - Add remaining API controllers (Announcement, Client, FAQ, TaxFiling) - Refactor Blazor components to use API instead of services - Implement JWT token refresh mechanism - Add SignalR for change notifications Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
112 lines
4.7 KiB
C#
112 lines
4.7 KiB
C#
namespace TaxBaik.Application.Services;
|
|
|
|
using System.Text.RegularExpressions;
|
|
using Microsoft.Extensions.Caching.Memory;
|
|
using TaxBaik.Domain.Entities;
|
|
using TaxBaik.Domain.Enums;
|
|
using TaxBaik.Domain.Interfaces;
|
|
|
|
public class InquiryService(
|
|
IInquiryRepository repository,
|
|
IInquiryNotificationService notificationService,
|
|
IMemoryCache memoryCache)
|
|
{
|
|
private static readonly Regex PhoneRegex = new(@"^01[0-9]-\d{3,4}-\d{4}$");
|
|
|
|
public async Task<int> SubmitAsync(
|
|
string name, string phone, string serviceType, string message,
|
|
string? email = null, string? ipAddress = null, CancellationToken ct = default)
|
|
{
|
|
if (string.IsNullOrWhiteSpace(name))
|
|
throw new ValidationException("이름을 입력하세요.");
|
|
|
|
if (!PhoneRegex.IsMatch(phone))
|
|
throw new ValidationException("올바른 전화번호를 입력하세요. (예: 010-1234-5678)");
|
|
|
|
if (string.IsNullOrWhiteSpace(message))
|
|
throw new ValidationException("문의 내용을 입력하세요.");
|
|
|
|
var inquiry = new Inquiry
|
|
{
|
|
Name = name.Trim(),
|
|
Phone = phone.Trim(),
|
|
Email = string.IsNullOrWhiteSpace(email) ? null : email.Trim(),
|
|
ServiceType = serviceType ?? "기타",
|
|
Message = message.Trim(),
|
|
IpAddress = ipAddress,
|
|
Status = InquiryStatusMapper.ToStorageValue(InquiryStatus.New),
|
|
CreatedAt = DateTime.UtcNow
|
|
};
|
|
|
|
var inquiryId = await repository.CreateAsync(inquiry, ct);
|
|
await notificationService.NotifyCreatedAsync(inquiryId, inquiry.Name, inquiry.Phone, inquiry.ServiceType, inquiry.Message, inquiry.IpAddress, inquiry.CreatedAt, ct);
|
|
memoryCache.Remove(AdminDashboardService.CacheKey);
|
|
return inquiryId;
|
|
}
|
|
|
|
public async Task<Inquiry?> GetByIdAsync(int id, CancellationToken ct = default) =>
|
|
await repository.GetByIdAsync(id, ct);
|
|
|
|
public async Task<(IEnumerable<Inquiry>, int)> GetPagedAsync(
|
|
int page, int pageSize, string? status = null, CancellationToken ct = default) =>
|
|
await repository.GetPagedAsync(NormalizePage(page), NormalizePageSize(pageSize), NormalizeOptionalStatus(status), ct);
|
|
|
|
public Task<int> CountAsync(CancellationToken ct = default)
|
|
=> repository.CountAsync(ct);
|
|
|
|
public Task<int> CountThisMonthAsync(CancellationToken ct = default)
|
|
=> repository.CountThisMonthAsync(ct);
|
|
|
|
public Task<int> CountByStatusAsync(string status, CancellationToken ct = default)
|
|
=> repository.CountByStatusAsync(status, ct);
|
|
|
|
public Task<int> CountByDateRangeAsync(DateTime startDate, DateTime endDate, CancellationToken ct = default)
|
|
=> repository.CountByDateRangeAsync(startDate, endDate, ct);
|
|
|
|
public Task<int> CountByStatusAndDateAsync(string status, DateTime startDate, DateTime endDate, CancellationToken ct = default)
|
|
=> repository.CountByStatusAndDateAsync(status, startDate, endDate, ct);
|
|
|
|
public async Task UpdateAdminMemoAsync(int id, string? adminMemo, CancellationToken ct = default) =>
|
|
await repository.UpdateAdminMemoAsync(id, adminMemo, ct);
|
|
|
|
public async Task LinkClientAsync(int inquiryId, int clientId, CancellationToken ct = default) =>
|
|
await repository.LinkClientAsync(inquiryId, clientId, ct);
|
|
|
|
public async Task UpdateStatusAsync(int id, string status, string? changedBy = null, CancellationToken ct = default)
|
|
{
|
|
if (!InquiryStatusMapper.TryParse(status, out var parsed))
|
|
throw new ValidationException("지원하지 않는 문의 상태입니다.");
|
|
|
|
var inquiry = await repository.GetByIdAsync(id, ct);
|
|
if (inquiry == null)
|
|
return;
|
|
|
|
var previousStatus = inquiry.Status;
|
|
var newStatus = InquiryStatusMapper.ToStorageValue(parsed);
|
|
|
|
await repository.UpdateStatusAsync(id, newStatus, ct);
|
|
await notificationService.NotifyStatusChangedAsync(id, inquiry.Name, inquiry.Phone, inquiry.ServiceType, previousStatus, newStatus, changedBy, ct);
|
|
memoryCache.Remove(AdminDashboardService.CacheKey);
|
|
}
|
|
|
|
private static int NormalizePage(int page) => Math.Max(1, page);
|
|
|
|
private static int NormalizePageSize(int pageSize) => Math.Clamp(pageSize, 1, 100);
|
|
|
|
private static string? NormalizeOptionalStatus(string? status)
|
|
{
|
|
if (string.IsNullOrWhiteSpace(status))
|
|
return null;
|
|
|
|
if (!InquiryStatusMapper.TryParse(status, out var parsed))
|
|
throw new ValidationException("지원하지 않는 문의 상태입니다.");
|
|
|
|
return InquiryStatusMapper.ToStorageValue(parsed);
|
|
}
|
|
}
|
|
|
|
public class ValidationException : Exception
|
|
{
|
|
public ValidationException(string message) : base(message) { }
|
|
}
|