2024-02-12 17:25:18 +05:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.ComponentModel.DataAnnotations;
|
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
2024-02-13 16:05:16 +05:00
|
|
|
using System.Reflection;
|
2024-02-12 17:25:18 +05:00
|
|
|
using AsbCloudApp.Data;
|
|
|
|
using AsbCloudApp.Requests.ParserOptions;
|
|
|
|
using AsbCloudApp.Services;
|
|
|
|
using ClosedXML.Excel;
|
2024-02-13 16:05:16 +05:00
|
|
|
using Mapster;
|
2024-02-12 17:25:18 +05:00
|
|
|
|
2024-02-13 16:05:16 +05:00
|
|
|
namespace AsbCloudInfrastructure.Services.Parser;
|
2024-02-12 17:25:18 +05:00
|
|
|
|
2024-02-16 13:53:10 +05:00
|
|
|
public abstract class ParserExcelService<TDto> : IParserService<TDto>
|
2024-02-12 17:25:18 +05:00
|
|
|
where TDto : class, IValidatableObject, IId
|
|
|
|
{
|
|
|
|
protected abstract string SheetName { get; }
|
2024-02-13 16:05:16 +05:00
|
|
|
|
|
|
|
protected virtual int HeaderRowsCount => 0;
|
2024-02-12 17:25:18 +05:00
|
|
|
|
2024-02-13 16:05:16 +05:00
|
|
|
protected abstract string TemplateFileName { get; }
|
|
|
|
|
|
|
|
protected abstract IDictionary<string, Cell> Cells { get; }
|
|
|
|
|
2024-02-16 13:53:10 +05:00
|
|
|
public virtual ParserResultDto<TDto> Parse<TOptions>(Stream file, TOptions options)
|
|
|
|
where TOptions : IParserOptionsRequest
|
2024-02-13 16:05:16 +05:00
|
|
|
{
|
|
|
|
using var workbook = new XLWorkbook(file);
|
|
|
|
var sheet = workbook.GetWorksheet(SheetName);
|
|
|
|
var dtos = ParseExcelSheet(sheet);
|
|
|
|
return dtos;
|
|
|
|
}
|
|
|
|
|
|
|
|
public virtual Stream GetTemplateFile() =>
|
|
|
|
Assembly.GetExecutingAssembly().GetTemplateCopyStream(TemplateFileName)
|
|
|
|
?? throw new ArgumentNullException($"Файл '{TemplateFileName}' не найден");
|
|
|
|
|
|
|
|
|
2024-02-14 14:13:43 +05:00
|
|
|
protected virtual IDictionary<string, object?> ParseRow(IXLRow xlRow)
|
2024-02-13 16:05:16 +05:00
|
|
|
{
|
|
|
|
var cells = Cells.ToDictionary(x => x.Key, x =>
|
|
|
|
{
|
|
|
|
var columnNumber = x.Value.ColumnNumber;
|
2024-02-14 14:13:43 +05:00
|
|
|
var xlCell = xlRow.Cell(columnNumber);
|
|
|
|
var cellValue = x.Value.GetValueFromCell(xlCell);
|
|
|
|
return cellValue;
|
2024-02-13 16:05:16 +05:00
|
|
|
});
|
2024-02-12 17:25:18 +05:00
|
|
|
|
2024-02-14 14:13:43 +05:00
|
|
|
return cells;
|
2024-02-13 16:05:16 +05:00
|
|
|
}
|
2024-02-12 17:25:18 +05:00
|
|
|
|
2024-02-14 14:13:43 +05:00
|
|
|
protected virtual TDto BuildDto(IDictionary<string, object?> row, int rowNumber)
|
2024-02-13 16:05:16 +05:00
|
|
|
{
|
2024-02-14 14:13:43 +05:00
|
|
|
var dto = row.Adapt<TDto>();
|
|
|
|
return dto;
|
2024-02-13 16:05:16 +05:00
|
|
|
}
|
2024-02-12 17:25:18 +05:00
|
|
|
|
|
|
|
private ValidationResultDto<TDto> Validate(TDto dto, int rowNumber)
|
|
|
|
{
|
|
|
|
var validationResults = new List<ValidationResult>();
|
2024-02-13 16:05:16 +05:00
|
|
|
|
2024-02-12 17:25:18 +05:00
|
|
|
var isValid = dto.Validate(validationResults);
|
|
|
|
|
|
|
|
if (isValid)
|
|
|
|
{
|
2024-02-13 16:05:16 +05:00
|
|
|
var validDto = new ValidationResultDto<TDto>
|
2024-02-12 17:25:18 +05:00
|
|
|
{
|
|
|
|
Item = dto
|
|
|
|
};
|
|
|
|
|
|
|
|
return validDto;
|
|
|
|
}
|
2024-02-13 16:05:16 +05:00
|
|
|
|
|
|
|
var columnsDict = Cells.ToDictionary(x => x.Key, x => x.Value.ColumnNumber);
|
|
|
|
|
2024-02-12 17:25:18 +05:00
|
|
|
var invalidDto = new ValidationResultDto<TDto>
|
|
|
|
{
|
|
|
|
Item = dto,
|
|
|
|
Warnings = validationResults
|
|
|
|
.SelectMany(v => v.MemberNames
|
2024-02-13 16:05:16 +05:00
|
|
|
.Where(columnsDict.ContainsKey)
|
2024-02-12 17:25:18 +05:00
|
|
|
.Select(m =>
|
|
|
|
{
|
2024-02-13 16:05:16 +05:00
|
|
|
var columnNumber = columnsDict[m];
|
2024-02-12 17:25:18 +05:00
|
|
|
var errorMessage = v.ErrorMessage;
|
2024-02-13 16:05:16 +05:00
|
|
|
var warningMessage = string.Format(XLExtentions.ProblemDetailsTemplate, SheetName, rowNumber, columnNumber,
|
|
|
|
errorMessage);
|
2024-02-12 17:25:18 +05:00
|
|
|
var warning = new ValidationResult(warningMessage, new[] { m });
|
|
|
|
return warning;
|
|
|
|
}))
|
|
|
|
};
|
|
|
|
|
|
|
|
return invalidDto;
|
|
|
|
}
|
|
|
|
|
2024-02-13 16:05:16 +05:00
|
|
|
protected virtual ParserResultDto<TDto> ParseExcelSheet(IXLWorksheet sheet)
|
2024-02-12 17:25:18 +05:00
|
|
|
{
|
2024-02-13 16:05:16 +05:00
|
|
|
var count = sheet.RowsUsed().Count() - HeaderRowsCount;
|
2024-02-12 17:25:18 +05:00
|
|
|
if (count <= 0)
|
|
|
|
return new ParserResultDto<TDto>();
|
|
|
|
|
|
|
|
var valiationResults = new List<ValidationResultDto<TDto>>(count);
|
|
|
|
var warnings = new List<ValidationResult>();
|
|
|
|
|
|
|
|
for (var i = 0; i < count; i++)
|
|
|
|
{
|
2024-02-13 16:05:16 +05:00
|
|
|
var xlRow = sheet.Row(1 + i + HeaderRowsCount);
|
2024-02-14 14:13:43 +05:00
|
|
|
var rowNumber = xlRow.RowNumber();
|
2024-02-12 17:25:18 +05:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2024-02-13 16:05:16 +05:00
|
|
|
var row = ParseRow(xlRow);
|
2024-02-14 14:13:43 +05:00
|
|
|
var dto = BuildDto(row, rowNumber);
|
|
|
|
var validationResult = Validate(dto, rowNumber);
|
2024-02-12 17:25:18 +05:00
|
|
|
valiationResults.Add(validationResult);
|
|
|
|
}
|
|
|
|
catch (FileFormatException ex)
|
|
|
|
{
|
|
|
|
var warning = new ValidationResult(ex.Message);
|
|
|
|
warnings.Add(warning);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var parserResult = new ParserResultDto<TDto>
|
|
|
|
{
|
|
|
|
Item = valiationResults
|
|
|
|
};
|
|
|
|
|
|
|
|
if (warnings.Any())
|
|
|
|
parserResult.Warnings = warnings;
|
|
|
|
|
|
|
|
return parserResult;
|
|
|
|
}
|
|
|
|
}
|