using System.Diagnostics; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Mvc; using Microsoft.EntityFrameworkCore; using bitforum.Models; using bitforum.Models.Page.Faq; using bitforum.Repository; using bitforum.Constants; namespace bitforum.Controllers.Page.Faq { [Authorize] [Route("Page")] public class CategoryController : Controller { private readonly ILogger _logger; private readonly IRedisRepository _redisRepository; private readonly DefaultDbContext _db; private readonly string _ViewPath = "~/Views/Page/Faq/Category/Index.cshtml"; public CategoryController(ILogger logger, IRedisRepository redisRepository, DefaultDbContext db) { _logger = logger; _redisRepository = redisRepository; _db = db; } [ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)] public IActionResult Error() { return View(new ErrorViewModel { RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier }); } [HttpGet("Faq")] public IActionResult Index() { ViewBag.FaqCategories = _db.FaqCategory.Include(c => c.FaqItem).OrderBy(c => c.Order).ToList(); ViewBag.Total = ViewBag.FaqCategories?.Count ?? 0; return View(_ViewPath); } [HttpPost("Faq")] public async Task Save([FromForm] List request) { using var transaction = await _db.Database.BeginTransactionAsync(); try { if (request == null || !request.Any()) { // 전체 삭제 var faqCategories = await _db.FaqCategory.ToListAsync(); if (faqCategories.Any()) { _db.FaqCategory.RemoveRange(faqCategories); await _db.SaveChangesAsync(); } await transaction.CommitAsync(); return RedirectToAction("Index"); } if (!ModelState.IsValid) { throw new Exception("유효성 검사에 실패하였습니다."); } var requestIDs = request.Select(x => x.ID).ToList(); // 요청 데이터의 ID 목록 var existingIDs = await _db.FaqCategory.Select(c => c.ID).ToListAsync(); // 데이터베이스에 존재하는 ID 목록 var IDsToDelete = existingIDs.Except(requestIDs).ToList(); // 삭제 대상 ID: 요청 데이터에 없는 항목 // 삭제 대상 항목 제거 if (IDsToDelete.Any()) { var selectedRows = await _db.FaqCategory.Where(c => IDsToDelete.Contains(c.ID) && !c.FaqItem.Any()).ToListAsync(); _db.FaqCategory.RemoveRange(selectedRows); } foreach (var row in request) { // 중복 확인 if (await _db.FaqCategory.AnyAsync(c => c.Code == row.Code && c.ID != row.ID)) { throw new Exception($"{row.Code} `Code`는 이미 존재합니다."); } if (row.ID == 0) { row.CreatedAt = DateTime.UtcNow; await _db.FaqCategory.AddAsync(row); } else { var faqCategory = await _db.FaqCategory.FirstOrDefaultAsync(c => c.ID == row.ID); if (faqCategory == null) { throw new Exception($"ID {row.ID}에 해당하는 정보가 없습니다."); } // 기존 엔터티 업데이트 faqCategory.Code = row.Code; faqCategory.Subject = row.Subject; faqCategory.Order = row.Order; faqCategory.IsActive = row.IsActive; faqCategory.UpdatedAt = DateTime.UtcNow; _db.FaqCategory.Update(faqCategory); } } await transaction.CommitAsync(); int affectedRows = await _db.SaveChangesAsync(); if (affectedRows <= 0) { throw new Exception("저장 중 오류가 발생했습니다."); } await _redisRepository.DeleteAsync(RedisConst.BannerKey); string message = "FAQ 분류가 저장되었습니다."; TempData["SuccessMessage"] = message; _logger.LogInformation(message); return RedirectToAction("Index"); } catch (Exception e) { await transaction.RollbackAsync(); TempData["ErrorMessages"] = e.Message; _logger.LogError(e, e.Message); return Index(); } } } }