hrms-api-recruit/Services/RecruitService.cs

554 lines
27 KiB
C#
Raw Normal View History

2023-03-17 14:24:43 +07:00
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.EntityFrameworkCore;
using BMA.EHR.Recruit.Service.Data;
using BMA.EHR.Recruit.Service.Models.Recruits;
using BMA.EHR.Recruit.Service.Core;
using BMA.EHR.MetaData.Service.Models;
using BMA.EHR.Domain.Models.Placement;
using BMA.EHR.Recurit.Service.Data;
using System.Security.Claims;
2024-07-03 00:42:41 +07:00
using System.Net.Http.Headers;
using Newtonsoft.Json;
2025-07-15 15:54:50 +07:00
using System.Globalization;
2025-10-11 18:56:47 +07:00
using BMA.EHR.Recruit.Service.Requests.Recruits;
2023-03-17 14:24:43 +07:00
namespace BMA.EHR.Recruit.Service.Services
{
public class RecruitService
{
private readonly ApplicationDbContext _context;
private readonly MetadataDbContext _contextMetadata;
2025-01-05 21:59:00 +07:00
private readonly OrgDbContext _contextOrg;
private readonly MinIOService _minIOService;
private readonly IHttpContextAccessor _httpContextAccessor;
2024-07-03 00:42:41 +07:00
private readonly IConfiguration _configuration;
public RecruitService(ApplicationDbContext context,
2025-01-05 21:59:00 +07:00
MetadataDbContext contextMetadata,
OrgDbContext contextOrg,
IHttpContextAccessor httpContextAccessor,
MinIOService minIOService,
IConfiguration configuration)
{
_context = context;
_contextMetadata = contextMetadata;
2025-01-05 21:59:00 +07:00
_contextOrg = contextOrg;
_minIOService = minIOService;
_httpContextAccessor = httpContextAccessor;
2024-07-03 00:42:41 +07:00
_configuration = configuration;
}
#region " Properties "
private string? UserId => _httpContextAccessor?.HttpContext?.User?.FindFirst(ClaimTypes.NameIdentifier)?.Value;
private string? FullName => _httpContextAccessor?.HttpContext?.User?.FindFirst("name")?.Value;
2024-07-03 00:42:41 +07:00
private string? token => _httpContextAccessor?.HttpContext?.Request.Headers["Authorization"];
#endregion
public int GetExamCount(string citizenId)
{
try
{
var count = _context.Recruits.AsQueryable()
.Where(x => x.CitizenId == citizenId)
.Count();
return count;
}
catch
{
throw;
}
}
public async Task<string> GetExamAttributeAsync(Guid period, Guid exam)
{
try
{
var payment = await _context.RecruitPayments.AsQueryable()
.Include(x => x.Recruit)
.ThenInclude(x => x.RecruitImport)
.Where(x => x.Recruit.Id == exam)
.Where(x => x.Recruit.RecruitImport.Id == period)
.FirstOrDefaultAsync();
return payment != null ? "มีคุณสมบัติ" : "ไม่มีคุณสมบัติ";
}
catch
{
throw;
}
}
public bool CheckValidCertificate(DateTime certDate, int nextYear = 5)
{
var valid = true;
if (DateTime.Now.Date > certDate.Date.AddYears(nextYear))
valid = false;
return valid;
}
public async Task UpdateDocAsync(Guid ImportId, IFormFileCollection files)
{
var periodExam = await _context.RecruitImports.AsQueryable()
.FirstOrDefaultAsync(x => x.Id == ImportId);
if (periodExam == null)
throw new Exception(GlobalMessages.DataNotFound);
foreach (var file in files)
{
var doc = await _minIOService.UploadFileAsync(file);
var periodExamDocument = new RecruitImportDocument
{
RecruitImportId = ImportId,
DocumentId = doc.Id,
};
await _context.RecruitImportDocuments.AddAsync(periodExamDocument);
}
await _context.SaveChangesAsync();
}
public async Task UpdateImageAsync(Guid ImportId, IFormFileCollection files)
{
var periodExam = await _context.RecruitImports.AsQueryable()
.FirstOrDefaultAsync(x => x.Id == ImportId);
if (periodExam == null)
throw new Exception(GlobalMessages.DataNotFound);
foreach (var file in files)
{
var doc = await _minIOService.UploadFileAsync(file);
var periodExamImage = new RecruitImportImage
{
RecruitImportId = ImportId,
DocumentId = doc.Id,
};
await _context.RecruitImportImages.AddAsync(periodExamImage);
}
await _context.SaveChangesAsync();
}
public async Task DeleteImageAsync(Guid id)
{
var image = await _context.RecruitImportImages.AsQueryable()
.Include(x => x.Document)
.FirstOrDefaultAsync(x => x.Id == id);
if (image == null)
throw new Exception(GlobalMessages.DataNotFound);
var doc_id = image.Document.Id;
_context.RecruitImportImages.Remove(image);
await _context.SaveChangesAsync();
await _minIOService.DeleteFileAsync(doc_id);
}
public async Task DeleteDocAsync(Guid id)
{
var doc = await _context.RecruitImportDocuments.AsQueryable()
.Include(x => x.Document)
.FirstOrDefaultAsync(x => x.Id == id);
if (doc == null)
throw new Exception(GlobalMessages.DataNotFound);
var doc_id = doc.Document.Id;
_context.RecruitImportDocuments.Remove(doc);
await _context.SaveChangesAsync();
await _minIOService.DeleteFileAsync(doc_id);
}
2025-10-11 18:37:46 +07:00
public async Task UpdateAsyncRecruitToPlacement(Guid examId, DateTime accountStartDate)
{
2025-01-05 21:59:00 +07:00
try
{
2025-10-11 18:56:47 +07:00
// 🚀 Prepare HTTP client once
var httpClient1 = new HttpClient();
httpClient1.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", token?.Replace("Bearer ", ""));
httpClient1.DefaultRequestHeaders.Add("api_key", _configuration["API_KEY"]);
2025-10-12 13:59:35 +07:00
var apiUrl1 = $"{_configuration["API"]}/org/pos/level";
2025-10-11 18:56:47 +07:00
var response1 = await httpClient1.GetStringAsync(apiUrl1);
2025-10-11 19:16:50 +07:00
var posOptions = JsonConvert.DeserializeObject<RecruitPosRequest>(response1);
2025-10-11 18:56:47 +07:00
2025-01-05 21:59:00 +07:00
var recruitImport = await _context.RecruitImports.AsQueryable()
.FirstOrDefaultAsync(x => x.Id == examId);
2024-07-03 00:42:41 +07:00
2025-01-05 21:59:00 +07:00
if (recruitImport == null)
throw new Exception(GlobalMessages.DataNotFound);
2024-07-03 00:42:41 +07:00
2025-01-05 21:59:00 +07:00
var _placement = await _contextMetadata.Placements.AsQueryable()
.FirstOrDefaultAsync(x => x.PlacementType.Name == "สอบแข่งขัน" && x.RefId == recruitImport.Id);
2025-10-12 14:53:35 +07:00
if (_placement != null)
throw new Exception("รอบการสอบนี้ได้ทำการบรรจุไปแล้ว");
2025-10-11 18:37:46 +07:00
// 🚀 Pre-load all lookup data once
var placementTypesCache = await _contextMetadata.PlacementTypes.ToListAsync();
var provincesCache = await _contextOrg.province.ToListAsync();
var districtsCache = await _contextOrg.district.ToListAsync();
var subDistrictsCache = await _contextOrg.subDistrict.ToListAsync();
var educationLevelsCache = await _contextOrg.educationLevel.ToListAsync();
2025-01-05 21:59:00 +07:00
var placement = new Placement
{
2025-01-05 21:59:00 +07:00
Name = recruitImport.Name,
RefId = recruitImport.Id,
2025-10-11 18:37:46 +07:00
Round = recruitImport.Order.ToString() ?? "",
Year = recruitImport.Year,
2025-01-05 21:59:00 +07:00
Number = await _context.Recruits.AsQueryable().Where(x => x.RecruitImport == recruitImport).CountAsync(),
2025-10-11 18:37:46 +07:00
PlacementType = placementTypesCache.FirstOrDefault(x => x.Name.Trim().ToUpper().Contains("สอบแข่งขัน")) ?? placementTypesCache.First(),
StartDate = accountStartDate,
EndDate = accountStartDate.AddYears(2).AddDays(-1),
CreatedAt = DateTime.Now,
CreatedUserId = UserId ?? "",
CreatedFullName = FullName ?? "",
LastUpdatedAt = DateTime.Now,
LastUpdateUserId = UserId ?? "",
LastUpdateFullName = FullName ?? "",
};
2025-01-05 21:59:00 +07:00
await _contextMetadata.Placements.AddAsync(placement);
2025-10-11 18:37:46 +07:00
// 🚀 Load all related data with single queries
2025-01-05 21:59:00 +07:00
var candidates = await _context.Recruits.AsQueryable()
.Include(x => x.Addresses)
.Include(x => x.Certificates)
.Include(x => x.Educations)
.Include(x => x.Occupations)
.Where(x => x.RecruitImport == recruitImport)
.ToListAsync();
2025-10-11 18:37:46 +07:00
var scoreImport = await _context.ScoreImports.AsQueryable()
.FirstOrDefaultAsync(x => x.RecruitImport == recruitImport);
var recruitScores = await _context.RecruitScores.AsQueryable()
.Where(x => x.ScoreImport == scoreImport && x.ExamStatus == "ผ่าน")
.ToListAsync();
2025-10-11 18:56:47 +07:00
var recruitScoresDict = recruitScores
.Where(x => !string.IsNullOrWhiteSpace(x.ExamId))
.ToDictionary(x => x.ExamId, x => x);
2025-10-11 18:37:46 +07:00
// 🚀 Prepare HTTP client once
var httpClient = new HttpClient();
httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", token?.Replace("Bearer ", ""));
httpClient.DefaultRequestHeaders.Add("api_key", _configuration["API_KEY"]);
// 🚀 Batch HTTP requests
var orgTasks = candidates.Select(async candidate =>
2025-01-05 21:59:00 +07:00
{
2025-10-11 18:56:47 +07:00
if (string.IsNullOrWhiteSpace(candidate.CitizenId))
return new { CitizenId = candidate.CitizenId ?? "", org = (dynamic?)null };
2025-01-05 21:59:00 +07:00
var apiUrl = $"{_configuration["API"]}/org/profile/citizenid/position/{candidate.CitizenId}";
2025-10-11 18:37:46 +07:00
try
{
var response = await httpClient.GetStringAsync(apiUrl);
2025-10-11 18:56:47 +07:00
return new { CitizenId = candidate.CitizenId, org = JsonConvert.DeserializeObject<dynamic>(response) };
2025-10-11 18:37:46 +07:00
}
catch
2025-01-05 21:59:00 +07:00
{
2025-10-11 18:56:47 +07:00
return new { CitizenId = candidate.CitizenId ?? "", org = (dynamic?)null };
2025-01-05 21:59:00 +07:00
}
2025-10-11 18:37:46 +07:00
}).ToList();
var orgResults = await Task.WhenAll(orgTasks);
2025-10-11 18:56:47 +07:00
var orgDict = orgResults.ToDictionary(x => x.CitizenId ?? "", x => x.org);
2025-10-11 18:37:46 +07:00
// 🚀 Prepare batch inserts
var placementProfiles = new List<PlacementProfile>();
var placementEducations = new List<PlacementEducation>();
/*Comment ข้อมูลใบประกอบวิชาชีพ เพราะในไฟล์นำเข้ายังไม่มีคอลัมน์ที่ระบุข้อมูลส่วนนี้*/
//var placementCertificates = new List<PlacementCertificate>();
2025-10-11 18:37:46 +07:00
foreach (var candidate in candidates)
{
2025-10-11 18:56:47 +07:00
if (string.IsNullOrWhiteSpace(candidate.ExamId) ||
!recruitScoresDict.TryGetValue(candidate.ExamId, out var recruitScore))
2025-01-05 21:59:00 +07:00
continue;
2025-10-11 18:37:46 +07:00
2025-10-11 18:56:47 +07:00
var org = orgDict.TryGetValue(candidate.CitizenId ?? "", out var orgValue) ? orgValue : null;
2025-10-11 18:37:46 +07:00
var isOfficer = org?.result != null;
// 🚀 Cache repeated calculations
var firstAddress = candidate.Addresses?.FirstOrDefault();
var firstEducation = candidate.Educations?.FirstOrDefault();
var firstCertificate = candidate.Certificates?.FirstOrDefault();
var firstOccupation = candidate.Occupations?.FirstOrDefault();
var registAddress = BuildAddress(firstAddress?.Address, firstAddress?.Moo, firstAddress?.Soi, firstAddress?.Road);
var currentAddress = BuildAddress(firstAddress?.Address1, firstAddress?.Moo1, firstAddress?.Soi1, firstAddress?.Road1);
2025-10-11 19:16:50 +07:00
// หาค่า posLevelName หลังสุด
var posLevelObject = posOptions?.result?.FirstOrDefault(x =>
!string.IsNullOrWhiteSpace(x.posLevelName) &&
!string.IsNullOrWhiteSpace(candidate.PositionName) &&
candidate.PositionName.Contains(x.posLevelName));
// เก็บเฉพาะค่า posLevelName
var posLevelName = posLevelObject?.posLevelName;
// สร้างตัวแปร PositionName ที่ตัดค่า posLevelName ออก
var positionNameWithoutLevel = candidate.PositionName ?? "";
if (!string.IsNullOrWhiteSpace(posLevelName))
{
positionNameWithoutLevel = positionNameWithoutLevel.Replace(posLevelName, "").Trim();
}
2025-01-05 21:59:00 +07:00
var placementProfile = new PlacementProfile
{
Placement = placement,
2025-10-11 19:49:30 +07:00
PositionCandidate = positionNameWithoutLevel ?? "",
2025-10-12 13:55:35 +07:00
PositionType = posLevelObject?.posTypes?.posTypeName ?? "",
2025-10-11 19:16:50 +07:00
PositionLevel = posLevelName ?? "",
2025-10-11 18:56:47 +07:00
Prefix = candidate.Prefix ?? "",
Firstname = candidate.FirstName ?? "",
Lastname = candidate.LastName ?? "",
Gender = candidate.Gendor ?? "",
Nationality = candidate.National ?? "",
Race = candidate.Race ?? "",
Religion = candidate.Religion ?? "",
2025-01-05 21:59:00 +07:00
DateOfBirth = candidate.DateOfBirth,
2025-10-11 18:56:47 +07:00
Relationship = candidate.Marry ?? "",
CitizenId = candidate.CitizenId ?? "",
2025-10-11 18:37:46 +07:00
CitizenProvinceId = provincesCache.FirstOrDefault(x => x.name == candidate.CitizenCardIssuer)?.Id,
2025-01-05 21:59:00 +07:00
CitizenDate = candidate.CitizenCardExpireDate,
2025-10-11 19:49:30 +07:00
Telephone = firstAddress?.Telephone ?? "",
MobilePhone = firstAddress?.Mobile ?? "",
RegistAddress = registAddress ?? "",
2025-10-11 18:37:46 +07:00
RegistProvinceId = provincesCache.FirstOrDefault(x => x.name == firstAddress?.Province)?.Id,
RegistDistrictId = districtsCache.FirstOrDefault(x => x.name == firstAddress?.Amphur)?.Id,
RegistSubDistrictId = subDistrictsCache.FirstOrDefault(x => x.name == firstAddress?.District)?.Id,
2025-10-11 19:49:30 +07:00
RegistZipCode = firstAddress?.ZipCode ?? "",
2025-01-05 21:59:00 +07:00
RegistSame = false,
2025-10-11 18:37:46 +07:00
CurrentAddress = currentAddress,
CurrentProvinceId = provincesCache.FirstOrDefault(x => x.name == firstAddress?.Province1)?.Id,
CurrentDistrictId = districtsCache.FirstOrDefault(x => x.name == firstAddress?.Amphur1)?.Id,
CurrentSubDistrictId = subDistrictsCache.FirstOrDefault(x => x.name == firstAddress?.District1)?.Id,
CurrentZipCode = firstAddress?.ZipCode1,
Marry = candidate.Marry?.Contains("สมรส") ?? false,
2025-01-05 21:59:00 +07:00
OccupationPositionType = "other",
2025-10-11 19:49:30 +07:00
OccupationTelephone = firstOccupation?.Telephone ?? "",
OccupationPosition = firstOccupation?.Position ?? "",
2025-10-11 18:56:47 +07:00
PointTotalA = recruitScore.FullA, // non-nullable int
PointA = recruitScore.SumA, // non-nullable double
PointTotalB = recruitScore.FullB ?? 0, // nullable int?
PointB = recruitScore.SumB ?? 0, // nullable double?
PointTotalC = recruitScore.FullC, // non-nullable int
PointC = recruitScore.SumC, // non-nullable double
ExamNumber = !string.IsNullOrWhiteSpace(recruitScore.Number) && int.TryParse(recruitScore.Number, out int n) ? n : null,
2025-01-05 21:59:00 +07:00
ExamRound = null,
IsRelief = false,
PlacementStatus = "UN-CONTAIN",
2025-10-11 18:56:47 +07:00
Pass = recruitScore.ExamStatus ?? "",
2025-01-05 21:59:00 +07:00
RemarkHorizontal = "โดยมีเงื่อนไขว่าต้องปฏิบัติงานให้กรุงเทพมหานครเป็นระยะเวลาไม่น้อยกว่า ๕ ปี นับแต่วันที่ได้รับการบรรจุและแต่งตั้ง โดยห้ามโอนไปหน่วยงานหรือส่วนราชการอื่น เว้นเเต่ลาออกจากราชการ",
2025-10-11 18:37:46 +07:00
Amount = org?.result?.amount,
PositionSalaryAmount = org?.result?.positionSalaryAmount,
MouthSalaryAmount = org?.result?.mouthSalaryAmount,
2025-01-05 21:59:00 +07:00
CreatedAt = DateTime.Now,
CreatedUserId = UserId ?? "",
CreatedFullName = FullName ?? "",
LastUpdatedAt = DateTime.Now,
LastUpdateUserId = UserId ?? "",
LastUpdateFullName = FullName ?? "",
2025-10-11 18:37:46 +07:00
IsOfficer = isOfficer,
2025-10-11 19:49:30 +07:00
profileId = org?.result?.profileId ?? "",
2025-10-11 18:37:46 +07:00
IsOld = org?.result != null,
AmountOld = org?.result?.amount,
2025-10-11 19:49:30 +07:00
nodeOld = org?.result?.node ?? "",
nodeIdOld = org?.result?.nodeId ?? "",
posmasterIdOld = org?.result?.posmasterId ?? "",
rootOld = org?.result?.root ?? "",
rootIdOld = org?.result?.rootId ?? "",
rootShortNameOld = org?.result?.rootShortName ?? "",
child1Old = org?.result?.child1 ?? "",
child1IdOld = org?.result?.child1Id ?? "",
child1ShortNameOld = org?.result?.child1ShortName ?? "",
child2Old = org?.result?.child2 ?? "",
child2IdOld = org?.result?.child2Id ?? "",
child2ShortNameOld = org?.result?.child2ShortName ?? "",
child3Old = org?.result?.child3 ?? "",
child3IdOld = org?.result?.child3Id ?? "",
child3ShortNameOld = org?.result?.child3ShortName ?? "",
child4Old = org?.result?.child4 ?? "",
child4IdOld = org?.result?.child4Id ?? "",
child4ShortNameOld = org?.result?.child4ShortName ?? "",
orgRevisionIdOld = org?.result?.orgRevisionId ?? "",
2025-10-12 14:43:42 +07:00
posMasterNoOld = org?.result?.posMasterNo,
2025-10-11 19:49:30 +07:00
positionNameOld = org?.result?.position ?? "",
posTypeIdOld = org?.result?.posTypeId ?? "",
posTypeNameOld = org?.result?.posTypeName ?? "",
posLevelIdOld = org?.result?.posLevelId ?? "",
posLevelNameOld = org?.result?.posLevelName ?? "",
2025-01-05 21:59:00 +07:00
};
2025-10-11 18:37:46 +07:00
placementProfiles.Add(placementProfile);
2025-01-05 21:59:00 +07:00
var placementEducation = new PlacementEducation
{
PlacementProfile = placementProfile,
2025-10-11 18:37:46 +07:00
EducationLevelId = educationLevelsCache.FirstOrDefault(x => x.name == firstEducation?.HighDegree)?.Id,
EducationLevelName = educationLevelsCache.FirstOrDefault(x => x.name == firstEducation?.HighDegree)?.name,
2025-10-11 19:49:30 +07:00
Field = firstEducation?.Major ?? "",
Gpa = firstEducation == null || firstEducation?.GPA == null ? "" : firstEducation.GPA.ToString(),
Institute = firstEducation?.University ?? "",
Degree = firstEducation?.Degree ?? "",
2025-10-11 18:37:46 +07:00
FinishDate = firstEducation?.BachelorDate,
2025-01-05 21:59:00 +07:00
IsDate = true,
CreatedAt = DateTime.Now,
CreatedUserId = UserId ?? "",
LastUpdatedAt = DateTime.Now,
LastUpdateUserId = UserId ?? "",
CreatedFullName = FullName ?? "",
LastUpdateFullName = FullName ?? "",
};
2025-10-11 18:37:46 +07:00
placementEducations.Add(placementEducation);
/*Comment ข้อมูลใบประกอบวิชาชีพ เพราะในไฟล์นำเข้ายังไม่มีคอลัมน์ที่ระบุข้อมูลส่วนนี้*/
//var placementCertificate = new PlacementCertificate
//{
// PlacementProfile = placementProfile,
// CertificateNo = firstCertificate?.CertificateNo ?? "",
// IssueDate = firstCertificate?.IssueDate,
// ExpireDate = firstCertificate?.ExpiredDate,
// CertificateType = firstCertificate?.Description ?? "",
// CreatedAt = DateTime.Now,
// CreatedUserId = UserId ?? "",
// LastUpdatedAt = DateTime.Now,
// LastUpdateUserId = UserId ?? "",
// CreatedFullName = FullName ?? "",
// LastUpdateFullName = FullName ?? "",
//};
//placementCertificates.Add(placementCertificate);
2025-01-05 21:59:00 +07:00
}
2025-10-11 18:37:46 +07:00
// 🚀 Batch insert all records
await _contextMetadata.PlacementProfiles.AddRangeAsync(placementProfiles);
await _contextMetadata.PlacementEducations.AddRangeAsync(placementEducations);
/*Comment ข้อมูลใบประกอบวิชาชีพ เพราะในไฟล์นำเข้ายังไม่มีคอลัมน์ที่ระบุข้อมูลส่วนนี้*/
//await _contextMetadata.PlacementCertificates.AddRangeAsync(placementCertificates);
2025-10-11 18:37:46 +07:00
// 🚀 Single SaveChanges at the end
await _contextMetadata.SaveChangesAsync();
httpClient.Dispose();
2025-01-05 21:59:00 +07:00
}
catch
{
throw;
}
}
2025-10-11 18:37:46 +07:00
private string BuildAddress(string? address, string? moo, string? soi, string? road)
{
var parts = new List<string>();
if (!string.IsNullOrWhiteSpace(address)) parts.Add(address);
if (!string.IsNullOrWhiteSpace(moo)) parts.Add($"หมู่ {moo}");
if (!string.IsNullOrWhiteSpace(soi)) parts.Add($"ซอย {soi}");
if (!string.IsNullOrWhiteSpace(road)) parts.Add($"ถนน {road}");
return string.Join(" ", parts);
}
public DateTime? CheckDateTime(string Date, string Formate)
2025-07-15 15:54:50 +07:00
{
// ตอนนี้ทำไว้ให้รองรับแค่ "dd/MM/yyyy", "yyyy-MM-dd"
Date = Date.Trim();
2025-07-15 15:54:50 +07:00
if (string.IsNullOrWhiteSpace(Date))
return null;
2025-07-15 15:54:50 +07:00
2025-09-05 12:09:16 +07:00
// จะเข้าเฉพาะกรณีที่ string เป็นตัวเลข เช่น "35635", "44561.5"
if (double.TryParse(Date, out double oaDate))
{
try
{
Date = DateTime.FromOADate(oaDate).ToString(Formate);
}
catch
{
Date = DateTime.MinValue.ToString(Formate);
}
}
2025-07-15 15:54:50 +07:00
string[] parts = Date.Trim().Replace("-", "/").Split("/");
if (parts.Length != 3)
return null;
2025-07-15 15:54:50 +07:00
int year;
int month;
int day;
switch (Formate)
{
case "dd/MM/yyyy":
if (int.TryParse(parts[2], out year) && year > 2500)
{
year -= 543;
}
else if (!int.TryParse(parts[2], out year))
{
return null;
2025-07-15 15:54:50 +07:00
}
if (!int.TryParse(parts[1], out month))
return null;
2025-07-15 15:54:50 +07:00
if (!int.TryParse(parts[0], out day))
return null;
2025-07-15 15:54:50 +07:00
break;
case "yyyy-MM-dd":
if (int.TryParse(parts[0], out year) && year > 2500)
{
year -= 543;
}
else if (!int.TryParse(parts[0], out year))
{
return null;
2025-07-15 15:54:50 +07:00
}
if (!int.TryParse(parts[1], out month))
return null;
2025-07-15 15:54:50 +07:00
if (!int.TryParse(parts[2], out day))
return null;
2025-07-15 15:54:50 +07:00
break;
default:
return null;
2025-07-15 15:54:50 +07:00
}
if (month < 1 || month > 12)
month = 1;
int maxDay = DateTime.DaysInMonth(year, month);
if (day < 1)
day = 1;
else if (day > maxDay)
day = maxDay;
var normalDate = $"{(day >= 1 && day <= 9 ? $"0{day}" : day)}/{(month >= 1 && month <= 9 ? $"0{month}" : month)}/{year}";
2025-07-15 15:54:50 +07:00
if (DateTime.TryParseExact(normalDate, "dd/MM/yyyy", CultureInfo.InvariantCulture, DateTimeStyles.None, out DateTime parsedDate))
{
return parsedDate;
}
return null;
2025-07-15 15:54:50 +07:00
}
}
2023-03-17 14:24:43 +07:00
}