diff --git a/BMA.EHR.Recruit.csproj b/BMA.EHR.Recruit.csproj
index bba397f..c148fa1 100644
--- a/BMA.EHR.Recruit.csproj
+++ b/BMA.EHR.Recruit.csproj
@@ -24,6 +24,8 @@
+
+
diff --git a/Controllers/RecruitController.cs b/Controllers/RecruitController.cs
index a673934..7d246ff 100644
--- a/Controllers/RecruitController.cs
+++ b/Controllers/RecruitController.cs
@@ -377,6 +377,7 @@ namespace BMA.EHR.Recruit.Controllers
try
{
var data = await _context.RecruitImports.AsQueryable()
+ .AsNoTracking()
.Include(x => x.RecruitImages)
.ThenInclude(x => x.Document)
.Include(x => x.RecruitDocuments)
@@ -2009,6 +2010,7 @@ namespace BMA.EHR.Recruit.Controllers
public async Task> ExportExamAsync(Guid id)
{
var data = await _context.RecruitImports.AsQueryable()
+ .AsNoTracking()
.Include(x => x.Recruits)
.FirstOrDefaultAsync(x => x.Id == id);
@@ -2517,6 +2519,7 @@ namespace BMA.EHR.Recruit.Controllers
public async Task GetCandidateNewListReportAsync(Guid id)
{
var data = await _context.Recruits.AsQueryable()
+ .AsNoTracking()
.Include(x => x.RecruitImport)
.Where(x => x.RecruitImport.Id == id)
.OrderBy(x => x.ExamId)
diff --git a/Services/ImportBackgroundService.cs b/Services/ImportBackgroundService.cs
index 0d3de83..0f827fb 100644
--- a/Services/ImportBackgroundService.cs
+++ b/Services/ImportBackgroundService.cs
@@ -4,13 +4,13 @@ using BMA.EHR.Recruit.Services;
using Microsoft.AspNetCore.Http;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Logging;
-using OfficeOpenXml;
using System.Net.Http.Headers;
using BMA.EHR.Recruit.Data;
-using EFCore.BulkExtensions;
using BMA.EHR.Recruit.Extensions;
+using EFCore.BulkExtensions;
using BMA.EHR.Recruit.Models.Recruits;
using BMA.EHR.Recruit.Requests.Recruits;
+using ExcelDataReader;
namespace BMA.EHR.Recruit.Services;
@@ -108,14 +108,18 @@ public class ImportBackgroundService : BackgroundService
var imported = await _context.RecruitImports.FindAsync(job.RecruitImportId);
if (imported == null) throw new Exception("RecruitImport not found");
- using var c_package = new ExcelPackage(new FileInfo(job.ImportFile));
- for (int i = 0; i < c_package.Workbook.Worksheets.Count; i++)
- {
- var workSheet = c_package.Workbook.Worksheets[i];
- var totalRows = workSheet.Dimension.Rows;
- var cols = workSheet.GetHeaderColumns();
+ System.Text.Encoding.RegisterProvider(System.Text.CodePagesEncodingProvider.Instance);
+ using var stream = System.IO.File.OpenRead(job.ImportFile);
+ using var reader = ExcelReaderFactory.CreateReader(stream);
+
+ do
+ {
+ // Read header row (row 1) to build column index map
+ if (!reader.Read()) continue;
+ var cols = new string[reader.FieldCount];
+ for (int c = 0; c < reader.FieldCount; c++)
+ cols[c] = reader.GetValue(c)?.ToString() ?? "";
- int row = 2;
int batchCount = 0;
const int batchSize = 500;
int totalProcessed = 0;
@@ -127,105 +131,105 @@ public class ImportBackgroundService : BackgroundService
var batchPayments = new List();
var batchCertificates = new List();
- while (row <= totalRows)
+ while (reader.Read())
{
- var cell1 = workSheet?.Cells[row, 1]?.GetValue();
- if (cell1 == "" || cell1 == null) break;
+ var cell1 = reader.GetValue(0)?.ToString();
+ if (string.IsNullOrEmpty(cell1)) break;
var r = new Models.Recruits.Recruit();
- r.ExamId = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.ExamID)]?.GetValue();
- r.CitizenId = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.PersonalID)]?.GetValue();
- r.Prefix = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Prefix)]?.GetValue();
- r.FirstName = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.FirstName)]?.GetValue();
- r.LastName = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.LastName)]?.GetValue();
- r.Gendor = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Gender)]?.GetValue();
- r.National = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.National)]?.GetValue().IsNull("");
- r.Race = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Race)]?.GetValue().IsNull("");
- r.Religion = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Religion)]?.GetValue().IsNull("");
- r.DateOfBirth = Convert.ToDateTime(workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.DateOfBirth)]?.GetValue().ToDateTime(DateTimeFormat.Ymd, "-"));
- r.Marry = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Marry)]?.GetValue();
+ r.ExamId = GetCellValue(reader, cols, CandidateFileHeader.ExamID);
+ r.CitizenId = GetCellValue(reader, cols, CandidateFileHeader.PersonalID);
+ r.Prefix = GetCellValue(reader, cols, CandidateFileHeader.Prefix);
+ r.FirstName = GetCellValue(reader, cols, CandidateFileHeader.FirstName);
+ r.LastName = GetCellValue(reader, cols, CandidateFileHeader.LastName);
+ r.Gendor = GetCellValue(reader, cols, CandidateFileHeader.Gender);
+ r.National = GetCellValue(reader, cols, CandidateFileHeader.National).IsNull("");
+ r.Race = GetCellValue(reader, cols, CandidateFileHeader.Race).IsNull("");
+ r.Religion = GetCellValue(reader, cols, CandidateFileHeader.Religion).IsNull("");
+ r.DateOfBirth = Convert.ToDateTime(GetCellValue(reader, cols, CandidateFileHeader.DateOfBirth).ToDateTime(DateTimeFormat.Ymd, "-"));
+ r.Marry = GetCellValue(reader, cols, CandidateFileHeader.Marry);
r.Isspecial = "N";
- r.CitizenCardIssuer = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.PersonalCardIssue)]?.GetValue();
- r.CitizenCardExpireDate = Convert.ToDateTime(workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.PersonalCardExpireDate)]?.GetValue().ToDateTime(DateTimeFormat.Ymd, "-"));
- r.ApplyDate = (DateTime)workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.ApplyDate)]?.GetValue();
- r.PositionName = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.PositionName)]?.GetValue().IsNull("");
- r.PositionType = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.PositionType)]?.GetValue().IsNull("");
- r.PositionLevel = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.PositionLevel)]?.GetValue().IsNull("");
+ r.CitizenCardIssuer = GetCellValue(reader, cols, CandidateFileHeader.PersonalCardIssue);
+ r.CitizenCardExpireDate = Convert.ToDateTime(GetCellValue(reader, cols, CandidateFileHeader.PersonalCardExpireDate).ToDateTime(DateTimeFormat.Ymd, "-"));
+ r.ApplyDate = GetCellDateTime(reader, cols, CandidateFileHeader.ApplyDate) ?? DateTime.MinValue;
+ r.PositionName = GetCellValue(reader, cols, CandidateFileHeader.PositionName).IsNull("");
+ r.PositionType = GetCellValue(reader, cols, CandidateFileHeader.PositionType).IsNull("");
+ r.PositionLevel = GetCellValue(reader, cols, CandidateFileHeader.PositionLevel).IsNull("");
// address
var address = new RecruitAddress()
{
- Address = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Address)]?.GetValue() ?? "",
- Moo = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Moo)]?.GetValue() ?? "",
- Soi = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Soi)]?.GetValue() ?? "",
- Road = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Road)]?.GetValue() ?? "",
- District = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.District)]?.GetValue() ?? "",
- Amphur = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Amphur)]?.GetValue() ?? "",
- Province = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Province)]?.GetValue() ?? "",
- ZipCode = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.ZipCode)]?.GetValue() ?? "",
- Telephone = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Telephone)]?.GetValue() ?? "",
- Mobile = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Mobile)]?.GetValue() ?? "",
- Address1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Address1)]?.GetValue() ?? "",
- Moo1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Moo1)]?.GetValue() ?? "",
- Soi1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Soi1)]?.GetValue() ?? "",
- Road1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Road1)]?.GetValue() ?? "",
- District1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.District1)]?.GetValue() ?? "",
- Amphur1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Amphur1)]?.GetValue() ?? "",
- Province1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Province1)]?.GetValue() ?? "",
- ZipCode1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.ZipCode1)]?.GetValue() ?? "",
+ Address = GetCellValue(reader, cols, CandidateFileHeader.Address),
+ Moo = GetCellValue(reader, cols, CandidateFileHeader.Moo),
+ Soi = GetCellValue(reader, cols, CandidateFileHeader.Soi),
+ Road = GetCellValue(reader, cols, CandidateFileHeader.Road),
+ District = GetCellValue(reader, cols, CandidateFileHeader.District),
+ Amphur = GetCellValue(reader, cols, CandidateFileHeader.Amphur),
+ Province = GetCellValue(reader, cols, CandidateFileHeader.Province),
+ ZipCode = GetCellValue(reader, cols, CandidateFileHeader.ZipCode),
+ Telephone = GetCellValue(reader, cols, CandidateFileHeader.Telephone),
+ Mobile = GetCellValue(reader, cols, CandidateFileHeader.Mobile),
+ Address1 = GetCellValue(reader, cols, CandidateFileHeader.Address1),
+ Moo1 = GetCellValue(reader, cols, CandidateFileHeader.Moo1),
+ Soi1 = GetCellValue(reader, cols, CandidateFileHeader.Soi1),
+ Road1 = GetCellValue(reader, cols, CandidateFileHeader.Road1),
+ District1 = GetCellValue(reader, cols, CandidateFileHeader.District1),
+ Amphur1 = GetCellValue(reader, cols, CandidateFileHeader.Amphur1),
+ Province1 = GetCellValue(reader, cols, CandidateFileHeader.Province1),
+ ZipCode1 = GetCellValue(reader, cols, CandidateFileHeader.ZipCode1),
};
// payment
var payment = new RecruitPayment()
{
- PaymentId = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.PaymentID)]?.GetValue() ?? "",
- CompanyCode = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.CompanyCode)]?.GetValue() ?? "",
- TextFile = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.TextFile)]?.GetValue() ?? "",
- BankCode = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.BankCode)]?.GetValue() ?? "",
- AccountNumber = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.AccouontNumer)]?.GetValue() ?? "",
- TransDate = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.TransDate)]?.GetValue() ?? "",
- TransTime = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.TransTime)]?.GetValue() ?? "",
- CustomerName = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.CustomerName)]?.GetValue() ?? "",
- RefNo1 = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.RefNo1)]?.GetValue() ?? "",
- TermBranch = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.TermBranch)]?.GetValue() ?? "",
- TellerId = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.TellerID)]?.GetValue() ?? "",
- CreditDebit = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.CreditDebit)]?.GetValue() ?? "",
- PaymentType = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Type)]?.GetValue(),
- ChequeNo = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.ChequeNo)]?.GetValue() ?? "",
- Amount = (decimal)workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Amount)]?.GetValue(),
- ChqueBankCode = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.ChqBankCode)]?.GetValue() ?? ""
+ PaymentId = GetCellValue(reader, cols, CandidateFileHeader.PaymentID),
+ CompanyCode = GetCellValue(reader, cols, CandidateFileHeader.CompanyCode),
+ TextFile = GetCellValue(reader, cols, CandidateFileHeader.TextFile),
+ BankCode = GetCellValue(reader, cols, CandidateFileHeader.BankCode),
+ AccountNumber = GetCellValue(reader, cols, CandidateFileHeader.AccouontNumer),
+ TransDate = GetCellValue(reader, cols, CandidateFileHeader.TransDate),
+ TransTime = GetCellValue(reader, cols, CandidateFileHeader.TransTime),
+ CustomerName = GetCellValue(reader, cols, CandidateFileHeader.CustomerName),
+ RefNo1 = GetCellValue(reader, cols, CandidateFileHeader.RefNo1),
+ TermBranch = GetCellValue(reader, cols, CandidateFileHeader.TermBranch),
+ TellerId = GetCellValue(reader, cols, CandidateFileHeader.TellerID),
+ CreditDebit = GetCellValue(reader, cols, CandidateFileHeader.CreditDebit),
+ PaymentType = GetCellValue(reader, cols, CandidateFileHeader.Type),
+ ChequeNo = GetCellValue(reader, cols, CandidateFileHeader.ChequeNo),
+ Amount = GetCellDecimal(reader, cols, CandidateFileHeader.Amount),
+ ChqueBankCode = GetCellValue(reader, cols, CandidateFileHeader.ChqBankCode)
};
// occupation
var occupation = new RecruitOccupation()
{
- Occupation = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Occupation)]?.GetValue() ?? "",
- Position = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Position)]?.GetValue() ?? "",
- Workplace = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Workplace)]?.GetValue() ?? "",
- Telephone = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.WorkplaceTelephone)]?.GetValue() ?? "",
- WorkAge = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.WorkAge)]?.GetValue() ?? "",
+ Occupation = GetCellValue(reader, cols, CandidateFileHeader.Occupation),
+ Position = GetCellValue(reader, cols, CandidateFileHeader.Position),
+ Workplace = GetCellValue(reader, cols, CandidateFileHeader.Workplace),
+ Telephone = GetCellValue(reader, cols, CandidateFileHeader.WorkplaceTelephone),
+ WorkAge = GetCellValue(reader, cols, CandidateFileHeader.WorkAge),
};
// certificate
var certificate = new RecruitCertificate()
{
- CertificateNo = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.CertificateNo)]?.GetValue() ?? "",
- Description = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.CertificateDesc)]?.GetValue() ?? "",
- IssueDate = Convert.ToDateTime(workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.CertificateIssueDate)]?.GetValue().ToDateTime(DateTimeFormat.Ymd, "-")),
- ExpiredDate = Convert.ToDateTime(workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.CertificateExpireDate)]?.GetValue().ToDateTime(DateTimeFormat.Ymd, "-"))
+ CertificateNo = GetCellValue(reader, cols, CandidateFileHeader.CertificateNo),
+ Description = GetCellValue(reader, cols, CandidateFileHeader.CertificateDesc),
+ IssueDate = Convert.ToDateTime(GetCellValue(reader, cols, CandidateFileHeader.CertificateIssueDate).ToDateTime(DateTimeFormat.Ymd, "-")),
+ ExpiredDate = Convert.ToDateTime(GetCellValue(reader, cols, CandidateFileHeader.CertificateExpireDate).ToDateTime(DateTimeFormat.Ymd, "-"))
};
var education = new RecruitEducation()
{
- Degree = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Degree)]?.GetValue() ?? "",
- Major = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.Major)]?.GetValue() ?? "",
- MajorGroupId = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.MajorGroupID)]?.GetValue() ?? "",
- MajorGroupName = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.MajorGroupName)]?.GetValue() ?? "",
- University = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.University)]?.GetValue() ?? "",
- GPA = (double)workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.GPA)]?.GetValue(),
- Specialist = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.SpecialList)]?.GetValue() ?? "",
- HighDegree = workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.HighDegree)]?.GetValue() ?? "",
- BachelorDate = Convert.ToDateTime(workSheet?.Cells[row, GetColumnIndex(cols, CandidateFileHeader.BachelorDate)]?.GetValue().ToDateTime(DateTimeFormat.Ymd, "-"))
+ Degree = GetCellValue(reader, cols, CandidateFileHeader.Degree),
+ Major = GetCellValue(reader, cols, CandidateFileHeader.Major),
+ MajorGroupId = GetCellValue(reader, cols, CandidateFileHeader.MajorGroupID),
+ MajorGroupName = GetCellValue(reader, cols, CandidateFileHeader.MajorGroupName),
+ University = GetCellValue(reader, cols, CandidateFileHeader.University),
+ GPA = GetCellDouble(reader, cols, CandidateFileHeader.GPA),
+ Specialist = GetCellValue(reader, cols, CandidateFileHeader.SpecialList),
+ HighDegree = GetCellValue(reader, cols, CandidateFileHeader.HighDegree),
+ BachelorDate = Convert.ToDateTime(GetCellValue(reader, cols, CandidateFileHeader.BachelorDate).ToDateTime(DateTimeFormat.Ymd, "-"))
};
r.Addresses.Add(address);
@@ -242,7 +246,6 @@ public class ImportBackgroundService : BackgroundService
batchCertificates.Add(certificate);
batchEducations.Add(education);
- row++;
batchCount++;
totalProcessed++;
@@ -300,7 +303,7 @@ public class ImportBackgroundService : BackgroundService
_context.ChangeTracker.Clear();
}
- }
+ } while (reader.NextResult());
job.TotalCount = _tracker.GetJob(job.JobId)?.ProcessedCount ?? 0;
}
@@ -331,11 +334,14 @@ public class ImportBackgroundService : BackgroundService
var importId = imported.Id;
var importRef = _context.Attach(new RecruitImport { Id = importId }).Entity;
- using var c_package = new ExcelPackage(new FileInfo(job.ImportFile));
- for (int i = 0; i < c_package.Workbook.Worksheets.Count; i++)
+ System.Text.Encoding.RegisterProvider(System.Text.CodePagesEncodingProvider.Instance);
+ using var stream = System.IO.File.OpenRead(job.ImportFile);
+ using var reader = ExcelReaderFactory.CreateReader(stream);
+
+ do
{
- var workSheet = c_package.Workbook.Worksheets[i];
- var totalRows = workSheet.Dimension.Rows;
+ // Skip header row
+ if (!reader.Read()) continue;
int row = 2;
int batchCount = 0;
@@ -348,33 +354,33 @@ public class ImportBackgroundService : BackgroundService
var batchAddresses = new List();
var batchPayments = new List();
- while (row <= totalRows)
+ while (reader.Read())
{
- var cell1 = workSheet?.Cells[row, 1]?.GetValue();
- if (cell1 == "" || cell1 == null) break;
+ var cell1 = reader.GetValue(0)?.ToString();
+ if (string.IsNullOrEmpty(cell1)) break;
try
{
var r = new Models.Recruits.Recruit();
r.Id = Guid.NewGuid();
- r.ExamId = workSheet?.Cells[row, 1]?.GetValue() ?? "";
- r.PositionName = workSheet?.Cells[row, 3]?.GetValue() ?? "";
- r.HddPosition = workSheet?.Cells[row, 4]?.GetValue() ?? "";
- r.Prefix = workSheet?.Cells[row, 5]?.GetValue() == "อื่น ๆ" ? workSheet?.Cells[row, 6]?.GetValue() ?? "" : workSheet?.Cells[row, 5]?.GetValue() ?? "";
- r.FirstName = workSheet?.Cells[row, 7]?.GetValue() ?? "";
- r.LastName = workSheet?.Cells[row, 8]?.GetValue() ?? "";
- r.Gendor = workSheet?.Cells[row, 98]?.GetValue() ?? "";
- r.National = workSheet?.Cells[row, 9]?.GetValue() ?? "";
+ r.ExamId = reader.GetValue(0)?.ToString() ?? "";
+ r.PositionName = reader.GetValue(2)?.ToString() ?? "";
+ r.HddPosition = reader.GetValue(3)?.ToString() ?? "";
+ r.Prefix = reader.GetValue(4)?.ToString() == "อื่น ๆ" ? reader.GetValue(5)?.ToString() ?? "" : reader.GetValue(4)?.ToString() ?? "";
+ r.FirstName = reader.GetValue(6)?.ToString() ?? "";
+ r.LastName = reader.GetValue(7)?.ToString() ?? "";
+ r.Gendor = reader.GetValue(97)?.ToString() ?? "";
+ r.National = reader.GetValue(8)?.ToString() ?? "";
r.Race = "";
- r.Religion = workSheet?.Cells[row, 10]?.GetValue() ?? "";
- r.DateOfBirth = !string.IsNullOrWhiteSpace(workSheet?.Cells[row, 11]?.GetValue()) ? _recruitService.CheckDateTime(workSheet?.Cells[row, 11]?.GetValue() ?? "", "dd/MM/yyyy") : null;
- r.CitizenId = workSheet?.Cells[row, 12]?.GetValue() ?? "";
- r.typeTest = workSheet?.Cells[row, 13]?.GetValue() ?? "";
+ r.Religion = reader.GetValue(9)?.ToString() ?? "";
+ r.DateOfBirth = !string.IsNullOrWhiteSpace(reader.GetValue(10)?.ToString()) ? _recruitService.CheckDateTime(reader.GetValue(10)?.ToString() ?? "", "dd/MM/yyyy") : null;
+ r.CitizenId = reader.GetValue(11)?.ToString() ?? "";
+ r.typeTest = reader.GetValue(12)?.ToString() ?? "";
r.Marry = "";
r.Isspecial = "N";
r.CitizenCardExpireDate = null;
r.ModifiedDate = null;
- r.ApplyDate = !string.IsNullOrWhiteSpace(workSheet?.Cells[row, 87]?.GetValue()) ? _recruitService.CheckDateTime(workSheet?.Cells[row, 87]?.GetValue() ?? "", "dd/MM/yyyy") : null;
+ r.ApplyDate = !string.IsNullOrWhiteSpace(reader.GetValue(86)?.ToString()) ? _recruitService.CheckDateTime(reader.GetValue(86)?.ToString() ?? "", "dd/MM/yyyy") : null;
r.PositionType = "";
r.PositionLevel = "";
r.CreatedAt = DateTime.Now;
@@ -389,15 +395,15 @@ public class ImportBackgroundService : BackgroundService
var education = new RecruitEducation()
{
Id = Guid.NewGuid(),
- Degree = workSheet?.Cells[row, 18]?.GetValue() ?? "",
- Major = workSheet?.Cells[row, 19]?.GetValue() == "อื่น ๆ" ? workSheet?.Cells[row, 20]?.GetValue() ?? "" : workSheet?.Cells[row, 19]?.GetValue() ?? "",
+ Degree = reader.GetValue(17)?.ToString() ?? "",
+ Major = reader.GetValue(18)?.ToString() == "อื่น ๆ" ? reader.GetValue(19)?.ToString() ?? "" : reader.GetValue(18)?.ToString() ?? "",
MajorGroupId = "",
MajorGroupName = "",
- University = workSheet?.Cells[row, 21]?.GetValue() == "อื่น ๆ" ? workSheet?.Cells[row, 22]?.GetValue() ?? "" : workSheet?.Cells[row, 21]?.GetValue() ?? "",
- GPA = (double)workSheet?.Cells[row, 26]?.GetValue(),
+ University = reader.GetValue(20)?.ToString() == "อื่น ๆ" ? reader.GetValue(21)?.ToString() ?? "" : reader.GetValue(20)?.ToString() ?? "",
+ GPA = GetReaderDouble(reader, 25),
Specialist = "",
- HighDegree = workSheet?.Cells[row, 27]?.GetValue() ?? "",
- BachelorDate = !string.IsNullOrWhiteSpace(workSheet?.Cells[row, 25]?.GetValue()) ? _recruitService.CheckDateTime(workSheet?.Cells[row, 25]?.GetValue() ?? "", "dd/MM/yyyy") : null,
+ HighDegree = reader.GetValue(26)?.ToString() ?? "",
+ BachelorDate = !string.IsNullOrWhiteSpace(reader.GetValue(24)?.ToString()) ? _recruitService.CheckDateTime(reader.GetValue(24)?.ToString() ?? "", "dd/MM/yyyy") : null,
Recruit = r,
CreatedAt = DateTime.Now,
CreatedUserId = job.UserId ?? "",
@@ -410,11 +416,11 @@ public class ImportBackgroundService : BackgroundService
var occupation = new RecruitOccupation()
{
Id = Guid.NewGuid(),
- Occupation = workSheet?.Cells[row, 33]?.GetValue() == "อื่น ๆ" ? workSheet?.Cells[row, 34]?.GetValue() ?? "" : workSheet?.Cells[row, 33]?.GetValue() ?? "",
- Position = workSheet?.Cells[row, 37]?.GetValue() ?? "",
- Workplace = $"{(workSheet?.Cells[row, 36]?.GetValue() ?? "")} {(workSheet?.Cells[row, 35]?.GetValue() ?? "")}",
- Telephone = workSheet?.Cells[row, 9999]?.GetValue() ?? "",
- WorkAge = workSheet?.Cells[row, 9999]?.GetValue() ?? "",
+ Occupation = reader.GetValue(32)?.ToString() == "อื่น ๆ" ? reader.GetValue(33)?.ToString() ?? "" : reader.GetValue(32)?.ToString() ?? "",
+ Position = reader.GetValue(36)?.ToString() ?? "",
+ Workplace = $"{(reader.GetValue(35)?.ToString() ?? "")} {(reader.GetValue(34)?.ToString() ?? "")}",
+ Telephone = "",
+ WorkAge = "",
Recruit = r,
CreatedAt = DateTime.Now,
CreatedUserId = job.UserId ?? "",
@@ -427,24 +433,24 @@ public class ImportBackgroundService : BackgroundService
var address = new RecruitAddress()
{
Id = Guid.NewGuid(),
- Address = $"{(workSheet?.Cells[row, 49]?.GetValue() ?? "")} {(workSheet?.Cells[row, 50]?.GetValue() ?? "")}",
- Moo = workSheet?.Cells[row, 51]?.GetValue() ?? "",
- Soi = workSheet?.Cells[row, 52]?.GetValue() ?? "",
- Road = workSheet?.Cells[row, 53]?.GetValue() ?? "",
- District = workSheet?.Cells[row, 54]?.GetValue() ?? "",
- Amphur = workSheet?.Cells[row, 55]?.GetValue() ?? "",
- Province = workSheet?.Cells[row, 56]?.GetValue() ?? "",
- ZipCode = (workSheet?.Cells[row, 57]?.GetValue() ?? "").Trim(),
- Telephone = workSheet?.Cells[row, 58]?.GetValue() ?? "",
+ Address = $"{(reader.GetValue(48)?.ToString() ?? "")} {(reader.GetValue(49)?.ToString() ?? "")}",
+ Moo = reader.GetValue(50)?.ToString() ?? "",
+ Soi = reader.GetValue(51)?.ToString() ?? "",
+ Road = reader.GetValue(52)?.ToString() ?? "",
+ District = reader.GetValue(53)?.ToString() ?? "",
+ Amphur = reader.GetValue(54)?.ToString() ?? "",
+ Province = reader.GetValue(55)?.ToString() ?? "",
+ ZipCode = (reader.GetValue(56)?.ToString() ?? "").Trim(),
+ Telephone = reader.GetValue(57)?.ToString() ?? "",
Mobile = "",
- Address1 = $"{(workSheet?.Cells[row, 61]?.GetValue() ?? "")} {(workSheet?.Cells[row, 62]?.GetValue() ?? "")}",
- Moo1 = workSheet?.Cells[row, 63]?.GetValue() ?? "",
- Soi1 = workSheet?.Cells[row, 64]?.GetValue() ?? "",
- Road1 = workSheet?.Cells[row, 65]?.GetValue() ?? "",
- District1 = workSheet?.Cells[row, 66]?.GetValue() ?? "",
- Amphur1 = workSheet?.Cells[row, 67]?.GetValue() ?? "",
- Province1 = workSheet?.Cells[row, 68]?.GetValue() ?? "",
- ZipCode1 = (workSheet?.Cells[row, 69]?.GetValue() ?? "").Trim(),
+ Address1 = $"{(reader.GetValue(60)?.ToString() ?? "")} {(reader.GetValue(61)?.ToString() ?? "")}",
+ Moo1 = reader.GetValue(62)?.ToString() ?? "",
+ Soi1 = reader.GetValue(63)?.ToString() ?? "",
+ Road1 = reader.GetValue(64)?.ToString() ?? "",
+ District1 = reader.GetValue(65)?.ToString() ?? "",
+ Amphur1 = reader.GetValue(66)?.ToString() ?? "",
+ Province1 = reader.GetValue(67)?.ToString() ?? "",
+ ZipCode1 = (reader.GetValue(68)?.ToString() ?? "").Trim(),
Recruit = r,
CreatedAt = DateTime.Now,
CreatedUserId = job.UserId ?? "",
@@ -457,22 +463,22 @@ public class ImportBackgroundService : BackgroundService
var payment = new RecruitPayment()
{
Id = Guid.NewGuid(),
- PaymentId = workSheet?.Cells[row, 104]?.GetValue() ?? "",
- CompanyCode = workSheet?.Cells[row, 105]?.GetValue() ?? "",
- TextFile = workSheet?.Cells[row, 106]?.GetValue() ?? "",
- BankCode = workSheet?.Cells[row, 107]?.GetValue() ?? "",
- AccountNumber = workSheet?.Cells[row, 108]?.GetValue() ?? "",
- TransDate = workSheet?.Cells[row, 109]?.GetValue() ?? "",
- TransTime = workSheet?.Cells[row, 110]?.GetValue() ?? "",
- CustomerName = workSheet?.Cells[row, 111]?.GetValue() ?? "",
- RefNo1 = workSheet?.Cells[row, 112]?.GetValue() ?? "",
- TermBranch = workSheet?.Cells[row, 113]?.GetValue() ?? "",
- TellerId = workSheet?.Cells[row, 114]?.GetValue() ?? "",
- CreditDebit = workSheet?.Cells[row, 115]?.GetValue() ?? "",
- PaymentType = workSheet?.Cells[row, 116]?.GetValue() ?? "",
- ChequeNo = workSheet?.Cells[row, 117]?.GetValue() ?? "",
- Amount = (decimal)workSheet?.Cells[row, 118]?.GetValue(),
- ChqueBankCode = workSheet?.Cells[row, 119]?.GetValue() ?? "",
+ PaymentId = reader.GetValue(103)?.ToString() ?? "",
+ CompanyCode = reader.GetValue(104)?.ToString() ?? "",
+ TextFile = reader.GetValue(105)?.ToString() ?? "",
+ BankCode = reader.GetValue(106)?.ToString() ?? "",
+ AccountNumber = reader.GetValue(107)?.ToString() ?? "",
+ TransDate = reader.GetValue(108)?.ToString() ?? "",
+ TransTime = reader.GetValue(109)?.ToString() ?? "",
+ CustomerName = reader.GetValue(110)?.ToString() ?? "",
+ RefNo1 = reader.GetValue(111)?.ToString() ?? "",
+ TermBranch = reader.GetValue(112)?.ToString() ?? "",
+ TellerId = reader.GetValue(113)?.ToString() ?? "",
+ CreditDebit = reader.GetValue(114)?.ToString() ?? "",
+ PaymentType = reader.GetValue(115)?.ToString() ?? "",
+ ChequeNo = reader.GetValue(116)?.ToString() ?? "",
+ Amount = GetReaderDecimal(reader, 117),
+ ChqueBankCode = reader.GetValue(118)?.ToString() ?? "",
Recruit = r,
CreatedAt = DateTime.Now,
CreatedUserId = job.UserId ?? "",
@@ -545,7 +551,7 @@ public class ImportBackgroundService : BackgroundService
_context.ChangeTracker.Clear();
}
- }
+ } while (reader.NextResult());
job.TotalCount = _tracker.GetJob(job.JobId)?.ProcessedCount ?? 0;
}
@@ -600,69 +606,80 @@ public class ImportBackgroundService : BackgroundService
await _context.SaveChangesAsync();
_context.ChangeTracker.Clear();
- // preload recruits (lightweight - only ExamId)
+ // preload recruits using AsNoTracking to avoid EF tracking overhead
var recruitsDict = await _context.Recruits
+ .AsNoTracking()
.Where(x => x.RecruitImport.Id == rec_import.Id)
.GroupBy(x => x.ExamId)
.Where(g => g.Count() == 1)
.Select(g => g.First())
.ToDictionaryAsync(x => x.ExamId, x => x);
- using var c_package = new ExcelPackage(new FileInfo(job.ImportFile));
- for (int i = 0; i < c_package.Workbook.Worksheets.Count; i++)
+ System.Text.Encoding.RegisterProvider(System.Text.CodePagesEncodingProvider.Instance);
+ using var stream = System.IO.File.OpenRead(job.ImportFile);
+ using var reader = ExcelReaderFactory.CreateReader(stream);
+
+ do
{
- var workSheet = c_package.Workbook.Worksheets[i];
- var cols = workSheet.GetHeaderColumns();
- int row = 8;
+ // Read header rows (rows 1-7), then data starts at row 8
+ // Skip 7 rows: first 7 are header/metadata
+ for (int skip = 0; skip < 7; skip++)
+ {
+ if (!reader.Read()) break;
+ }
+
+ var cols = new string[reader.FieldCount];
+ // Use current row (row 7) as header reference — not actually used for ScoreFile column mapping
+ // ScoreFile uses hardcoded column indices
+
int batchCount = 0;
const int batchSize = 500;
int totalProcessed = 0;
- var endRow = workSheet.Dimension.End.Row;
var batchScores = new List();
- while (row <= endRow)
+ while (reader.Read())
{
- var cell1 = workSheet?.Cells[row, 1]?.GetValue();
- if (cell1 == "" || cell1 == null) break;
+ var cell1 = reader.GetValue(0)?.ToString();
+ if (string.IsNullOrEmpty(cell1)) break;
var r = new RecruitScore();
- r.ExamId = workSheet?.Cells[row, 2]?.GetValue();
+ r.ExamId = reader.GetValue(1)?.ToString();
if (!string.IsNullOrEmpty(r.ExamId) && recruitsDict.TryGetValue(r.ExamId, out var recruit))
{
- r.CitizenId = workSheet?.Cells[row, 3]?.GetValue()?.Trim();
+ r.CitizenId = reader.GetValue(2)?.ToString()?.Trim();
r.FullA = 200;
- r.SumA = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 5]?.GetValue()) ? 0.00 : Math.Round(workSheet.Cells[row, 5].GetValue(), 2);
- r.PercentageA = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 6]?.GetValue()) ? 0.00 : Math.Round(workSheet.Cells[row, 6].GetValue(), 2);
- r.AStatus = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 7]?.GetValue()) ? "" : workSheet?.Cells[row, 7]?.GetValue();
- r.SumAB = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 5]?.GetValue()) ? 0.00 : Math.Round(workSheet.Cells[row, 5].GetValue(), 2);
- r.ABStatus = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 7]?.GetValue()) ? "" : workSheet?.Cells[row, 7]?.GetValue();
+ r.SumA = string.IsNullOrWhiteSpace(reader.GetValue(4)?.ToString()) ? 0.00 : Math.Round(Convert.ToDouble(reader.GetValue(4)), 2);
+ r.PercentageA = string.IsNullOrWhiteSpace(reader.GetValue(5)?.ToString()) ? 0.00 : Math.Round(Convert.ToDouble(reader.GetValue(5)), 2);
+ r.AStatus = string.IsNullOrWhiteSpace(reader.GetValue(6)?.ToString()) ? "" : reader.GetValue(6)?.ToString();
+ r.SumAB = string.IsNullOrWhiteSpace(reader.GetValue(4)?.ToString()) ? 0.00 : Math.Round(Convert.ToDouble(reader.GetValue(4)), 2);
+ r.ABStatus = string.IsNullOrWhiteSpace(reader.GetValue(6)?.ToString()) ? "" : reader.GetValue(6)?.ToString();
r.FullC = 50;
- r.SumC = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 8]?.GetValue()) ? 0.00 : Math.Round(workSheet.Cells[row, 8].GetValue(), 2);
+ r.SumC = string.IsNullOrWhiteSpace(reader.GetValue(7)?.ToString()) ? 0.00 : Math.Round(Convert.ToDouble(reader.GetValue(7)), 2);
r.FullD = 50;
- r.SumD = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 9]?.GetValue()) ? 0.00 : Math.Round(workSheet.Cells[row, 9].GetValue(), 2);
- r.SumCD = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 10]?.GetValue()) ? 0.00 : Math.Round(workSheet.Cells[row, 10].GetValue(), 2);
- r.PercentageC = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 11]?.GetValue()) ? 0.00 : Math.Round(workSheet.Cells[row, 11].GetValue(), 2);
- r.CStatus = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 12]?.GetValue()) ? "" : workSheet?.Cells[row, 12]?.GetValue();
+ r.SumD = string.IsNullOrWhiteSpace(reader.GetValue(8)?.ToString()) ? 0.00 : Math.Round(Convert.ToDouble(reader.GetValue(8)), 2);
+ r.SumCD = string.IsNullOrWhiteSpace(reader.GetValue(9)?.ToString()) ? 0.00 : Math.Round(Convert.ToDouble(reader.GetValue(9)), 2);
+ r.PercentageC = string.IsNullOrWhiteSpace(reader.GetValue(10)?.ToString()) ? 0.00 : Math.Round(Convert.ToDouble(reader.GetValue(10)), 2);
+ r.CStatus = string.IsNullOrWhiteSpace(reader.GetValue(11)?.ToString()) ? "" : reader.GetValue(11)?.ToString();
r.FullScore = 300;
- r.TotalScore = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 13]?.GetValue()) ? 0.00 : Math.Round(workSheet.Cells[row, 13].GetValue(), 2);
+ r.TotalScore = string.IsNullOrWhiteSpace(reader.GetValue(12)?.ToString()) ? 0.00 : Math.Round(Convert.ToDouble(reader.GetValue(12)), 2);
- var examStatusCol7 = workSheet?.Cells[row, 7]?.GetValue()?.Trim();
- var examStatusCol14 = workSheet?.Cells[row, 14]?.GetValue()?.Trim();
+ var examStatusCol7 = reader.GetValue(6)?.ToString()?.Trim();
+ var examStatusCol14 = reader.GetValue(13)?.ToString()?.Trim();
r.ExamStatus =
examStatusCol7 == "ขาดสอบ" ? "ขส." :
examStatusCol14 == "ได้" ? "ผ่าน" :
examStatusCol14 == "ตก" ? "ไม่ผ่าน" : "-";
- r.RemarkScore = string.IsNullOrWhiteSpace(workSheet?.Cells[row, 15]?.GetValue()) ? string.Empty : workSheet?.Cells[row, 15]?.GetValue();
+ r.RemarkScore = string.IsNullOrWhiteSpace(reader.GetValue(14)?.ToString()) ? string.Empty : reader.GetValue(14)?.ToString();
- var examAttr = workSheet?.Cells[row, 16]?.GetValue()?.Trim();
+ var examAttr = reader.GetValue(15)?.ToString()?.Trim();
r.ExamAttribute =
examAttr == "ผ่าน" ? "มีคุณสมบัติ" :
examAttr == "ไม่ผ่าน" ? "ไม่มีคุณสมบัติ" : "";
- r.Major = workSheet.Name;
+ r.Major = reader.Name; // worksheet name
r.CreatedAt = DateTime.Now;
r.CreatedUserId = job.UserId ?? "";
r.CreatedFullName = job.FullName ?? "System Administrator";
@@ -674,7 +691,6 @@ public class ImportBackgroundService : BackgroundService
batchScores.Add(r);
}
- row++;
batchCount++;
totalProcessed++;
@@ -692,7 +708,7 @@ public class ImportBackgroundService : BackgroundService
{
await _context.BulkInsertAsync(batchScores);
}
- }
+ } while (reader.NextResult());
job.TotalCount = _tracker.GetJob(job.JobId)?.ProcessedCount ?? 0;
}
@@ -741,8 +757,9 @@ public class ImportBackgroundService : BackgroundService
await _context.SaveChangesAsync();
_context.ChangeTracker.Clear();
- // preload scores - re-query from DB to avoid tracking issues
+ // preload scores using AsNoTracking to avoid EF tracking overhead
var scoreList = await _context.RecruitScores
+ .AsNoTracking()
.Where(s => s.ScoreImport.RecruitImportId == rec_import.Id && !string.IsNullOrEmpty(s.ExamId))
.GroupBy(x => x.ExamId)
.Where(g => g.Count() == 1)
@@ -750,31 +767,34 @@ public class ImportBackgroundService : BackgroundService
.ToListAsync();
var score = scoreList.ToDictionary(s => s.ExamId!, s => s);
- // Read from saved file (ResultFile uses stream from Form, but we saved to disk)
+ System.Text.Encoding.RegisterProvider(System.Text.CodePagesEncodingProvider.Instance);
using var stream = System.IO.File.OpenRead(job.ImportFile);
- using var c_package = new ExcelPackage(stream);
+ using var reader = ExcelReaderFactory.CreateReader(stream);
- foreach (var workSheet in c_package.Workbook.Worksheets)
+ do
{
- int row = 7;
+ // Skip 6 header rows, data starts at row 7
+ for (int skip = 0; skip < 6; skip++)
+ {
+ if (!reader.Read()) break;
+ }
+
int batchCount = 0;
const int batchSize = 500;
- var endRow = workSheet.Dimension.End.Row;
var batchUpdates = new List();
- while (row <= endRow)
+ while (reader.Read())
{
- var examId = workSheet?.Cells[row, 2]?.GetValue();
+ var examId = reader.GetValue(1)?.ToString();
if (string.IsNullOrWhiteSpace(examId))
{
- row++;
continue;
}
if (score.TryGetValue(examId, out var existingScore))
{
- existingScore.Number = workSheet?.Cells[row, 1]?.GetValue();
- existingScore.RemarkExamOrder = workSheet?.Cells[row, 4]?.GetValue() ?? string.Empty;
+ existingScore.Number = reader.GetValue(0)?.ToString();
+ existingScore.RemarkExamOrder = reader.GetValue(3)?.ToString() ?? string.Empty;
existingScore.LastUpdatedAt = DateTime.Now;
existingScore.LastUpdateUserId = job.UserId ?? "";
existingScore.LastUpdateFullName = job.FullName ?? "System Administrator";
@@ -782,8 +802,6 @@ public class ImportBackgroundService : BackgroundService
batchCount++;
}
- row++;
-
if (batchCount >= batchSize)
{
await _context.BulkUpdateAsync(batchUpdates);
@@ -797,7 +815,7 @@ public class ImportBackgroundService : BackgroundService
{
await _context.BulkUpdateAsync(batchUpdates);
}
- }
+ } while (reader.NextResult());
}
#endregion
@@ -819,6 +837,81 @@ public class ImportBackgroundService : BackgroundService
}
}
+ ///
+ /// Get string value from ExcelDataReader by header column name
+ ///
+ private static string GetCellValue(IExcelDataReader reader, string[] cols, string headerName)
+ {
+ var idx = GetColumnIndex(cols, headerName);
+ if (idx <= 0 || idx > reader.FieldCount) return "";
+ return reader.GetValue(idx - 1)?.ToString() ?? "";
+ }
+
+ ///
+ /// Get DateTime value from ExcelDataReader by header column name
+ ///
+ private static DateTime? GetCellDateTime(IExcelDataReader reader, string[] cols, string headerName)
+ {
+ var idx = GetColumnIndex(cols, headerName);
+ if (idx <= 0 || idx > reader.FieldCount) return null;
+ var val = reader.GetValue(idx - 1);
+ if (val is DateTime dt) return dt;
+ if (val != null && DateTime.TryParse(val.ToString(), out var parsed)) return parsed;
+ return null;
+ }
+
+ ///
+ /// Get double value from ExcelDataReader by header column name
+ ///
+ private static double GetCellDouble(IExcelDataReader reader, string[] cols, string headerName)
+ {
+ var idx = GetColumnIndex(cols, headerName);
+ if (idx <= 0 || idx > reader.FieldCount) return 0.0;
+ var val = reader.GetValue(idx - 1);
+ if (val is double d) return d;
+ if (val != null && double.TryParse(val.ToString(), out var parsed)) return parsed;
+ return 0.0;
+ }
+
+ ///
+ /// Get decimal value from ExcelDataReader by header column name
+ ///
+ private static decimal GetCellDecimal(IExcelDataReader reader, string[] cols, string headerName)
+ {
+ var idx = GetColumnIndex(cols, headerName);
+ if (idx <= 0 || idx > reader.FieldCount) return 0m;
+ var val = reader.GetValue(idx - 1);
+ if (val is decimal dec) return dec;
+ if (val is double dbl) return (decimal)dbl;
+ if (val != null && decimal.TryParse(val.ToString(), out var parsed)) return parsed;
+ return 0m;
+ }
+
+ ///
+ /// Get double value from ExcelDataReader by 0-based column index
+ ///
+ private static double GetReaderDouble(IExcelDataReader reader, int index)
+ {
+ if (index < 0 || index >= reader.FieldCount) return 0.0;
+ var val = reader.GetValue(index);
+ if (val is double d) return d;
+ if (val != null && double.TryParse(val.ToString(), out var parsed)) return parsed;
+ return 0.0;
+ }
+
+ ///
+ /// Get decimal value from ExcelDataReader by 0-based column index
+ ///
+ private static decimal GetReaderDecimal(IExcelDataReader reader, int index)
+ {
+ if (index < 0 || index >= reader.FieldCount) return 0m;
+ var val = reader.GetValue(index);
+ if (val is decimal dec) return dec;
+ if (val is double dbl) return (decimal)dbl;
+ if (val != null && decimal.TryParse(val.ToString(), out var parsed)) return parsed;
+ return 0m;
+ }
+
#endregion
}
diff --git a/obj/project.assets.json b/obj/project.assets.json
index 818122a..82551d3 100644
--- a/obj/project.assets.json
+++ b/obj/project.assets.json
@@ -289,6 +289,35 @@
"lib/net7.0/EPPlus.System.Drawing.dll": {}
}
},
+ "ExcelDataReader/3.8.0": {
+ "type": "package",
+ "compile": {
+ "lib/netstandard2.1/ExcelDataReader.dll": {
+ "related": ".pdb;.xml"
+ }
+ },
+ "runtime": {
+ "lib/netstandard2.1/ExcelDataReader.dll": {
+ "related": ".pdb;.xml"
+ }
+ }
+ },
+ "ExcelDataReader.DataSet/3.8.0": {
+ "type": "package",
+ "dependencies": {
+ "ExcelDataReader": "3.8.0"
+ },
+ "compile": {
+ "lib/netstandard2.1/ExcelDataReader.DataSet.dll": {
+ "related": ".pdb;.xml"
+ }
+ },
+ "runtime": {
+ "lib/netstandard2.1/ExcelDataReader.DataSet.dll": {
+ "related": ".pdb;.xml"
+ }
+ }
+ },
"Google.Protobuf/3.19.4": {
"type": "package",
"compile": {
@@ -5670,6 +5699,56 @@
"readme.md"
]
},
+ "ExcelDataReader/3.8.0": {
+ "sha512": "kbUsldc5Fn9IKgzL2nr4VvN/mKqPqn8zGXUZpA7uL6svCA4psF+qMK519EhMvderpU4pAJoqk9DWpiSIkiZtXA==",
+ "type": "package",
+ "path": "exceldatareader/3.8.0",
+ "files": [
+ ".nupkg.metadata",
+ ".signature.p7s",
+ "ExcelDataReader.png",
+ "README.md",
+ "exceldatareader.3.8.0.nupkg.sha512",
+ "exceldatareader.nuspec",
+ "lib/net462/ExcelDataReader.dll",
+ "lib/net462/ExcelDataReader.pdb",
+ "lib/net462/ExcelDataReader.xml",
+ "lib/net8.0/ExcelDataReader.dll",
+ "lib/net8.0/ExcelDataReader.pdb",
+ "lib/net8.0/ExcelDataReader.xml",
+ "lib/netstandard2.0/ExcelDataReader.dll",
+ "lib/netstandard2.0/ExcelDataReader.pdb",
+ "lib/netstandard2.0/ExcelDataReader.xml",
+ "lib/netstandard2.1/ExcelDataReader.dll",
+ "lib/netstandard2.1/ExcelDataReader.pdb",
+ "lib/netstandard2.1/ExcelDataReader.xml"
+ ]
+ },
+ "ExcelDataReader.DataSet/3.8.0": {
+ "sha512": "+eQg5oinHir7ayE/rF4dedvy8J6FBDG8RDyKFQsS/VZG9ygrnNgW6U8JSlrfGfe3DxYgbVoVwYV9Hbk63JQJFQ==",
+ "type": "package",
+ "path": "exceldatareader.dataset/3.8.0",
+ "files": [
+ ".nupkg.metadata",
+ ".signature.p7s",
+ "ExcelDataReader.png",
+ "README.md",
+ "exceldatareader.dataset.3.8.0.nupkg.sha512",
+ "exceldatareader.dataset.nuspec",
+ "lib/net462/ExcelDataReader.DataSet.dll",
+ "lib/net462/ExcelDataReader.DataSet.pdb",
+ "lib/net462/ExcelDataReader.DataSet.xml",
+ "lib/net8.0/ExcelDataReader.DataSet.dll",
+ "lib/net8.0/ExcelDataReader.DataSet.pdb",
+ "lib/net8.0/ExcelDataReader.DataSet.xml",
+ "lib/netstandard2.0/ExcelDataReader.DataSet.dll",
+ "lib/netstandard2.0/ExcelDataReader.DataSet.pdb",
+ "lib/netstandard2.0/ExcelDataReader.DataSet.xml",
+ "lib/netstandard2.1/ExcelDataReader.DataSet.dll",
+ "lib/netstandard2.1/ExcelDataReader.DataSet.pdb",
+ "lib/netstandard2.1/ExcelDataReader.DataSet.xml"
+ ]
+ },
"Google.Protobuf/3.19.4": {
"sha512": "fd07/ykL4O4FhqrZIELm5lmiyOHfdPg9+o+hWr6tcfRdS7tHXnImg/2wtogLzlW2eEmr0J7j6ZrZvaWOLiJbxQ==",
"type": "package",
@@ -13996,6 +14075,8 @@
"CoreAdmin >= 2.7.0",
"EFCore.BulkExtensions.MySql >= 6.7.16",
"EPPlus >= 6.1.3",
+ "ExcelDataReader >= 3.8.0",
+ "ExcelDataReader.DataSet >= 3.8.0",
"Microsoft.AspNetCore.Authentication.JwtBearer >= 7.0.20",
"Microsoft.AspNetCore.Mvc.NewtonsoftJson >= 7.0.3",
"Microsoft.AspNetCore.Mvc.Versioning >= 5.0.0",
@@ -14089,6 +14170,14 @@
"target": "Package",
"version": "[6.1.3, )"
},
+ "ExcelDataReader": {
+ "target": "Package",
+ "version": "[3.8.0, )"
+ },
+ "ExcelDataReader.DataSet": {
+ "target": "Package",
+ "version": "[3.8.0, )"
+ },
"Microsoft.AspNetCore.Authentication.JwtBearer": {
"target": "Package",
"version": "[7.0.20, )"
diff --git a/obj/project.nuget.cache b/obj/project.nuget.cache
index e443564..4529909 100644
--- a/obj/project.nuget.cache
+++ b/obj/project.nuget.cache
@@ -1,6 +1,6 @@
{
"version": 2,
- "dgSpecHash": "XR3lYvVwNcQ=",
+ "dgSpecHash": "++tptrWNc3M=",
"success": true,
"projectFilePath": "/Users/suphonchaip/Develop/hrms/hrms-api-recruit/BMA.EHR.Recruit.csproj",
"expectedPackageFiles": [
@@ -21,6 +21,8 @@
"/Users/suphonchaip/.nuget/packages/epplus/6.1.3/epplus.6.1.3.nupkg.sha512",
"/Users/suphonchaip/.nuget/packages/epplus.interfaces/6.1.1/epplus.interfaces.6.1.1.nupkg.sha512",
"/Users/suphonchaip/.nuget/packages/epplus.system.drawing/6.1.1/epplus.system.drawing.6.1.1.nupkg.sha512",
+ "/Users/suphonchaip/.nuget/packages/exceldatareader/3.8.0/exceldatareader.3.8.0.nupkg.sha512",
+ "/Users/suphonchaip/.nuget/packages/exceldatareader.dataset/3.8.0/exceldatareader.dataset.3.8.0.nupkg.sha512",
"/Users/suphonchaip/.nuget/packages/google.protobuf/3.19.4/google.protobuf.3.19.4.nupkg.sha512",
"/Users/suphonchaip/.nuget/packages/humanizer.core/2.14.1/humanizer.core.2.14.1.nupkg.sha512",
"/Users/suphonchaip/.nuget/packages/k4os.compression.lz4/1.2.6/k4os.compression.lz4.1.2.6.nupkg.sha512",