using Microsoft.AspNetCore.Mvc; using Microsoft.EntityFrameworkCore; using MigraDocCore.DocumentObjectModel; using MigraDocCore.DocumentObjectModel.Tables; using MigraDocCore.Rendering; using PdfSharpCore; using PdfSharpCore.Drawing; using PdfSharpCore.Pdf; using Server.Data; using Server.Models; using SharedModels.Responses; using Utils; using Route = Server.Models.Route; namespace Server.Services; public class ReportService : IReportService { private readonly ApplicationDbContext _dbContext; private readonly ISessionUserService _sessionUserService; public ReportService(ApplicationDbContext dbContext, ISessionUserService sessionUserService) { _dbContext = dbContext; _sessionUserService = sessionUserService; } public async Task<(bool IsSucceed, IActionResult? actionResult, Stream ticketPdf)> GetTicket(int ticketGroupId) { if (!await DoesTicketGroupExist(ticketGroupId)) { return (false, new NotFoundResult(), null!); } if (_sessionUserService.GetAuthUserRole() != Identity.Roles.Administrator.ToString()) { if ((await _dbContext.TicketGroups.FirstAsync(tg => tg.Id == ticketGroupId)).UserId != _sessionUserService.GetAuthUserId()) { return (false, new UnauthorizedResult(), null!); } } var dbTicketGroup = await _dbContext.TicketGroups .Include(tg => tg.User) .Include(tg => tg.Tickets) .ThenInclude(t => t.VehicleEnrollment) .ThenInclude(ve => ve.Vehicle) .ThenInclude(v => v.Company) .Include(tg => tg.User) .Include(tg => tg.Tickets) .ThenInclude(t => t.VehicleEnrollment) .ThenInclude(ve => ve.Route) .ThenInclude(r => r.RouteAddresses) .ThenInclude(ra => ra.Address) .ThenInclude(a => a.City) .ThenInclude(c => c.State) .ThenInclude(s => s.Country) .Include(tg => tg.User) .Include(tg => tg.Tickets) .ThenInclude(t => t.VehicleEnrollment) .ThenInclude(ve => ve.RouteAddressDetails) .FirstAsync(tg => tg.Id == ticketGroupId); // Define document var document = new PdfDocument(); document.Info.Title = "ticket"; document.Info.Author = "auto.bus"; // Craft document var pdfPage = document.AddPage(); pdfPage.Size = PageSize.A4; var gfx = XGraphics.FromPdfPage(pdfPage); // HACK² gfx.MUH = PdfFontEncoding.Unicode; var doc = new Document(); doc.DefaultPageSetup.LeftMargin = Unit.FromCentimeter(1); doc.DefaultPageSetup.RightMargin = Unit.FromCentimeter(1); doc.DefaultPageSetup.MirrorMargins = true; DefineStyles(doc); CreatePage(doc); FillPage(doc, dbTicketGroup); var docRender = new DocumentRenderer(doc); docRender.PrepareDocument(); docRender.RenderPage(gfx, 1); // Save document var memoryStream = new MemoryStream(); document.Save(memoryStream); return (true, null, memoryStream); void DefineStyles(Document doc) { var styles = doc.Styles["Normal"]; styles.Font.Name = "Courier New Cyr"; styles = doc.Styles.AddStyle("Table", "Normal"); styles.Font.Size = 10; styles.ParagraphFormat.SpaceBefore = 2.5; styles.ParagraphFormat.SpaceAfter = 2.5; } void CreatePage(Document doc) { var section = doc.AddSection(); // Create header var paragraph = section.Headers.Primary.AddParagraph(); paragraph.AddText("auto.bus"); paragraph.Style = StyleNames.Header; paragraph.Format.Font.Size = 20; paragraph.Format.Font.Bold = true; paragraph = section.Headers.Primary.AddParagraph(); paragraph.Format.Alignment = ParagraphAlignment.Center; paragraph.Format.Font.Size = 0; paragraph.Format.Borders.Top = new Border { Width = 1, Color = Colors.Black }; paragraph.Format.Borders.Bottom = new Border { Color = Colors.Transparent }; // Add title paragraph = section.AddParagraph(); paragraph.AddText("Посадочний документ"); paragraph.Format.Font.Size = 20; paragraph.Format.Font.Bold = true; // Add break line before table paragraph = section.AddParagraph(); paragraph.Format.Alignment = ParagraphAlignment.Center; paragraph.Format.Font.Size = 0; paragraph.Format.Borders.Top = new Border { Width = 1, Color = Colors.Black }; paragraph.Format.Borders.Bottom = new Border { Color = Colors.Transparent }; paragraph.Format.Borders.Style = BorderStyle.DashLargeGap; paragraph.Format.SpaceBefore = 5; paragraph.Format.SpaceAfter = 5; // Add table and define columns var table = section.AddTable(); table.Style = "Table"; table.Borders.Color = Colors.Black; table.Borders.Width = 0.25; table.Borders.Left.Width = 0.5; table.Borders.Right.Width = 0.5; table.Rows.LeftIndent = 0; table.Rows.Height = Unit.FromPoint(15); table.Rows.VerticalAlignment = VerticalAlignment.Center; for (int i = 0; i < 12; i++) { var column = table.AddColumn(Unit.FromCentimeter(1.583)); column.Format.Alignment = ParagraphAlignment.Center; } // Add break line after table paragraph = section.AddParagraph(); paragraph.Format.Alignment = ParagraphAlignment.Center; paragraph.Format.Font.Size = 0; paragraph.Format.Borders.Top = new Border { Width = 1, Color = Colors.Black }; paragraph.Format.Borders.Bottom = new Border { Color = Colors.Transparent }; paragraph.Format.Borders.Style = BorderStyle.DashLargeGap; paragraph.Format.SpaceBefore = 5; paragraph.Format.SpaceAfter = 5; } void FillPage(Document doc, TicketGroup ticketGroup) { var table = doc.LastSection.LastTable; var row = table.AddRow(); table.AddRow(); row.Cells[0].MergeRight = 2; row.Cells[0].MergeDown = 1; row.Cells[0].AddParagraph("aut.bus – м. Харків, просп. Науки, 14"); row.Cells[0].Format.Alignment = ParagraphAlignment.Left; row.Cells[3].MergeRight = 2; row.Cells[3].MergeDown = 1; row.Cells[3].AddParagraph("ПОСАДОЧНИЙ ДОКУМЕНТ"); row.Cells[3].Format.Font.Bold = true; string ticketNums = ""; foreach (var ticket in ticketGroup.Tickets) { ticketNums += $"{ticket.Id}"; if (!ticketGroup.Tickets.Last().Equals(ticket)) { ticketNums += "; "; continue; } ticketNums += "."; } row.Cells[6].MergeRight = 2; row.Cells[6].MergeDown = 1; row.Cells[6].AddParagraph($"Група: {ticketGroup.Id}.\nКвитки: {ticketNums}"); row.Cells[6].Format.Alignment = ParagraphAlignment.Left; row.Cells[9].MergeRight = 2; row.Cells[9].MergeDown = 1; row.Cells[9].AddParagraph($"{ticketGroup.PurchaseDateTimeUtc:dd.MM.yyyy HH:mm:ss}"); row = table.AddRow(); row.Cells[0].MergeRight = 2; row.Cells[0].AddParagraph("Прізвище, Ім'я"); row.Cells[0].Format.Alignment = ParagraphAlignment.Left; row.Cells[3].MergeRight = 8; row.Cells[3].AddParagraph($"{ticketGroup.User.LastName} {ticketGroup.User.FirstName}"); row.Cells[3].Format.Alignment = ParagraphAlignment.Left; // Fill stations row = table.AddRow(); row.Cells[0].MergeRight = 11; row.Cells[0].AddParagraph("СТАНЦІЇ"); row.Cells[0].Format.Font.Bold = true; var isFilled = false; for (var i = 0; i < ticketGroup.Tickets.Count; i++) { var ticket = ticketGroup.Tickets[i]; var vehicle = ticket.VehicleEnrollment.Vehicle; row = table.AddRow(); table.AddRow(); row.Shading.Color = isFilled ? Color.FromRgbColor(25, Colors.Black) : Colors.White; isFilled = !isFilled; row.Cells[0].MergeRight = 1; row.Cells[0].MergeDown = 1; if (ticketGroup.Tickets.First().Equals(ticket)) { row.Cells[0].AddParagraph("Відправлення"); } else { row.Cells[0].AddParagraph("Пересадка на"); } row.Cells[2].MergeRight = 1; row.Cells[2].MergeDown = 1; row.Cells[2].AddParagraph($"{ticket.GetDepartureTime():dd.MM.yyyy HH:mm}"); row.Cells[4].MergeRight = 3; row.Cells[4].MergeDown = 1; row.Cells[4].Format.Font.Size = 8; row.Cells[4].AddParagraph($"{ticket.GetDepartureAddress().GetFullName()}"); row.Cells[8].MergeRight = 1; row.Cells[8].MergeDown = 1; row.Cells[8].AddParagraph("Тип, номер автобуса"); row.Cells[10].MergeRight = 1; row.Cells[10].MergeDown = 1; row.Cells[10].AddParagraph($"{vehicle.Type}, {vehicle.Number}"); row = table.AddRow(); table.AddRow(); row.Shading.Color = isFilled ? Color.FromRgbColor(25, Colors.Black) : Colors.White; isFilled = !isFilled; row.Cells[0].MergeRight = 1; row.Cells[0].MergeDown = 1; if (!ticketGroup.Tickets.Last().Equals(ticket)) { row.Cells[0].AddParagraph("Пересадка з"); } else { row.Cells[0].AddParagraph("Прибуття"); } row.Cells[2].MergeRight = 1; row.Cells[2].MergeDown = 1; row.Cells[2].AddParagraph($"{ticket.GetArrivalTime():dd.MM.yyyy HH:mm}"); row.Cells[4].MergeRight = 3; row.Cells[4].MergeDown = 1; row.Cells[4].Format.Font.Size = 8; row.Cells[4].AddParagraph($"{ticket.GetArrivalAddress().GetFullName()}"); row.Cells[8].MergeRight = 1; row.Cells[8].MergeDown = 1; row.Cells[8].AddParagraph("Тип, номер автобуса"); row.Cells[10].MergeRight = 1; row.Cells[10].MergeDown = 1; row.Cells[10].AddParagraph($"{vehicle.Type}, {vehicle.Number}"); if (!ticketGroup.Tickets.Last().Equals(ticket)) { var nextDepartureTime = ticketGroup.Tickets[i + 1].GetDepartureTime(); var freeTime = nextDepartureTime - ticket.GetArrivalTime(); row = table.AddRow(); table.AddRow(); row.Shading.Color = isFilled ? Color.FromRgbColor(25, Colors.Black) : Colors.White; isFilled = !isFilled; row.Cells[0].MergeRight = 2; row.Cells[0].MergeDown = 1; row.Cells[0].AddParagraph("Вільний час"); row.Cells[3].MergeRight = 5; row.Cells[3].MergeDown = 1; row.Cells[3].AddParagraph($"{ticket.GetArrivalTime():dd.MM.yyyy HH:mm} – {nextDepartureTime:dd.MM.yyyy HH:mm}"); row.Cells[9].MergeRight = 2; row.Cells[9].MergeDown = 1; row.Cells[9].AddParagraph($"{freeTime.ToString(@"dd\.hh\:mm\:ss")}"); } } // Fill value row = table.AddRow(); row.Cells[0].MergeRight = 11; row.Cells[0].AddParagraph("ВАРТІСТЬ"); row.Cells[0].Format.Font.Bold = true; isFilled = false; foreach (var ticket in ticketGroup.Tickets) { row = table.AddRow(); table.AddRow(); row.Shading.Color = isFilled ? Color.FromRgbColor(25, Colors.Black) : Colors.White; isFilled = !isFilled; row.Cells[0].MergeDown = 1; row.Cells[0].AddParagraph("Звідки"); row.Cells[1].MergeRight = 3; row.Cells[1].MergeDown = 1; row.Cells[1].Format.Font.Size = 8; row.Cells[1].AddParagraph($"{ticket.GetDepartureAddress().GetFullName()}"); row.Cells[5].MergeDown = 1; row.Cells[5].AddParagraph("Куди"); row.Cells[6].MergeRight = 3; row.Cells[6].MergeDown = 1; row.Cells[6].Format.Font.Size = 8; row.Cells[6].AddParagraph($"{ticket.GetArrivalAddress().GetFullName()}"); row.Cells[10].MergeDown = 1; row.Cells[10].AddParagraph("Ціна"); row.Cells[11].MergeDown = 1; row.Cells[11].AddParagraph($"{ticket.GetCost()}"); } row = table.AddRow(); row.Shading.Color = isFilled ? Color.FromRgbColor(25, Colors.Black) : Colors.White; isFilled = !isFilled; row.Cells[0].MergeRight = 9; row.Cells[0].AddParagraph("Загальна"); row.Cells[0].Format.Alignment = ParagraphAlignment.Left; row.Cells[10].Borders.Right.Color = Colors.Transparent; row.Cells[11].AddParagraph($"{ticketGroup.GetCost()}"); } } public async Task<(bool isSucceed, IActionResult? actionResult, Stream reportPdf)> GetCompanyReportPdf(int? companyId, DateTime fromDate, DateTime toDate) { if (_sessionUserService.GetAuthUserRole() == Identity.Roles.Administrator.ToString()) { if (companyId == null) { return (false, new BadRequestObjectResult("CompanyId must have a value"), null!); } } else { var result = await _sessionUserService.IsAuthUserCompanyOwner(); if (!result.isCompanyOwner) { return (false, new UnauthorizedResult(), null!); } companyId = result.companyId; } if (!await DoesCompanyExist((int)companyId)) { return (false, new NotFoundResult(), null!); } var dbCompany = await _dbContext.Companies .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Reviews) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Tickets) .ThenInclude(t => t.TicketGroup) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Route) .ThenInclude(r => r.RouteAddresses) .ThenInclude(ra => ra.Address) .ThenInclude(a => a.City) .ThenInclude(c => c.State) .ThenInclude(s => s.Country) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.RouteAddressDetails) .Select(c => new { Company = c, VehicleEnrollments = c.Vehicles .Select(v => new { Vehicle = v, VehicleEnrollments = v.VehicleEnrollments.Where(ve => ve.DepartureDateTimeUtc.Date >= fromDate && ve.DepartureDateTimeUtc <= toDate) }) }) .Select(o => o.Company) .FirstAsync(c => c.Id == companyId); var routesEnrolled = new List(); foreach (var vehicle in dbCompany.Vehicles) { foreach (var vehicleEnrollment in vehicle.VehicleEnrollments) { var route = vehicleEnrollment.Route; if (!routesEnrolled.Contains(route)) { routesEnrolled.Add(route); } } } var vehicleEnrolled = dbCompany.Vehicles; // Define document var document = new PdfDocument(); document.Info.Title = "ticket"; document.Info.Author = "auto.bus"; var doc = new Document(); doc.DefaultPageSetup.LeftMargin = Unit.FromCentimeter(1); doc.DefaultPageSetup.RightMargin = Unit.FromCentimeter(1); doc.DefaultPageSetup.MirrorMargins = true; DefineStyles(doc); CreatePage(doc); FillPage(doc); var docRender = new DocumentRenderer(doc); docRender.PrepareDocument(); var pageCount = docRender.FormattedDocument.PageCount; for (int i = 0; i < pageCount; i++) { var pdfPage = document.AddPage(); pdfPage.Size = PageSize.A4; using var gfx = XGraphics.FromPdfPage(pdfPage); // HACK² gfx.MUH = PdfFontEncoding.Unicode; docRender.RenderPage(gfx, i + 1); } // Save document var memoryStream = new MemoryStream(); document.Save(memoryStream); return (true, null, memoryStream); void DefineStyles(Document doc) { var styles = doc.Styles["Normal"]; styles.Font.Name = "Courier New Cyr"; styles.ParagraphFormat.LineSpacingRule = LineSpacingRule.OnePtFive; styles = doc.Styles.AddStyle("Table", "Normal"); styles.Font.Size = 10; styles.ParagraphFormat.SpaceBefore = 2.5; styles.ParagraphFormat.SpaceAfter = 2.5; styles.ParagraphFormat.LineSpacingRule = LineSpacingRule.Single; } void CreatePage(Document doc) { var section = doc.AddSection(); // Create footer // var paragraph = section.Footers.Primary.AddParagraph(); // paragraph.Format.Alignment = ParagraphAlignment.Center; // paragraph.AddPageField(); // section.Footers. // section.PageSetup.DifferentFirstPageHeaderFooter = true; var paragraph = section.AddParagraph("auto.bus"); paragraph.Format.Font.Size = 20; paragraph.Format.Font.Bold = true; paragraph.Format.LineSpacingRule = LineSpacingRule.OnePtFive; paragraph.Format.Alignment = ParagraphAlignment.Center; paragraph.Format.SpaceBefore = Unit.FromCentimeter(5); paragraph.Format.SpaceAfter = Unit.FromCentimeter(5); paragraph = section.AddParagraph("Фінансовий звіт"); paragraph.Format.Font.Size = 20; paragraph.Format.LineSpacingRule = LineSpacingRule.OnePtFive; paragraph.Format.Alignment = ParagraphAlignment.Center; paragraph = section.AddParagraph($"{dbCompany.Name}"); paragraph.Format.Font.Size = 20; paragraph.Format.LineSpacingRule = LineSpacingRule.OnePtFive; paragraph.Format.Alignment = ParagraphAlignment.Center; paragraph.Format.SpaceAfter = Unit.FromCentimeter(5); paragraph = section.AddParagraph($"{fromDate:dd.MM.yyyy} – {toDate:dd.MM.yyyy}"); paragraph.Format.Font.Size = 20; paragraph.Format.LineSpacingRule = LineSpacingRule.OnePtFive; paragraph.Format.Alignment = ParagraphAlignment.Center; Table table; // Section and table for each enrolled route for (int i = 0; i < routesEnrolled.Count; i++) { section = doc.AddSection(); // Add table and define columns table = section.AddTable(); table.Style = "Table"; table.Borders.Color = Colors.Black; table.Borders.Width = 0.25; table.Borders.Left.Width = 0.5; table.Borders.Right.Width = 0.5; table.Rows.LeftIndent = 0; table.Rows.Height = Unit.FromPoint(15); table.Rows.VerticalAlignment = VerticalAlignment.Center; for (int j = 0; j < 12; j++) { var column = table.AddColumn(Unit.FromCentimeter(1.583)); column.Format.Alignment = ParagraphAlignment.Center; } } // Section for total section = doc.AddSection(); } void FillPage(Document doc) { Section section; Paragraph paragraph; int i = 1; foreach (var route in routesEnrolled) { section = doc.Sections[i]; paragraph = section.Footers.Primary.AddParagraph(); paragraph.AddPageField(); paragraph.Format.Alignment = ParagraphAlignment.Center; var table = section.LastTable; var row = table.AddRow(); row.Cells[0].MergeRight = 11; row.Cells[0].AddParagraph($"МАРШРУТ №{route.Id}"); row.Cells[0].Format.Font.Bold = true; row = table.AddRow(); row.Cells[0].MergeRight = 1; row.Cells[0].AddParagraph("Відправлення"); row.Cells[2].MergeRight = 3; row.Cells[2].AddParagraph($"{route.RouteAddresses.First().Address.GetFullName()}"); row.Cells[6].MergeRight = 1; row.Cells[6].AddParagraph("Прибуття"); row.Cells[8].MergeRight = 3; row.Cells[8].AddParagraph($"{route.RouteAddresses.Last().Address.GetFullName()}"); row = table.AddRow(); row.Cells[0].MergeRight = 11; row.Cells[0].AddParagraph("КОРОТКО"); row.Cells[0].Format.Font.Bold = true; row = table.AddRow(); row.Cells[0].MergeRight = 1; row.Cells[0].AddParagraph("Поїздок проведено"); row.Cells[2].MergeRight = 1; row.Cells[2].AddParagraph("Поїздок скасовано"); row.Cells[4].MergeRight = 1; row.Cells[4].AddParagraph("Квитків продано"); row.Cells[6].MergeRight = 1; row.Cells[6].AddParagraph("Неповних квитків"); row.Cells[8].MergeRight = 1; row.Cells[8].AddParagraph("Грошей зароблено"); row.Cells[10].MergeRight = 1; row.Cells[10].AddParagraph("Середній рейтинг"); row = table.AddRow(); row.Shading.Color = Color.FromRgbColor(25, Colors.Black); row.Cells[0].MergeRight = 1; row.Cells[0].AddParagraph($"{route.GetCompanyEnrollmentCount(fromDate, toDate, (int) companyId)}"); row.Cells[2].MergeRight = 1; row.Cells[2].AddParagraph($"{route.GetCompanyCanceledEnrollmentCount(fromDate, toDate, (int) companyId)}"); row.Cells[4].MergeRight = 1; row.Cells[4].AddParagraph($"{route.GetCompanySoldTicketCount(fromDate, toDate, (int) companyId)}"); row.Cells[6].MergeRight = 1; row.Cells[6].AddParagraph($"{route.GetCompanyIndirectTicketCount(fromDate, toDate, (int) companyId)}"); row.Cells[8].MergeRight = 1; row.Cells[8].AddParagraph($"{route.GetCompanyTotalRevenue(fromDate, toDate, (int) companyId)}"); row.Cells[10].MergeRight = 1; var routeAverageRating = route.GetCompanyAverageRating(fromDate, toDate, (int) companyId); row.Cells[10].AddParagraph($"{(routeAverageRating == 0 ? "-" : routeAverageRating)}"); row = table.AddRow(); row.Cells[0].MergeRight = 11; row.Cells[0].AddParagraph("ДОКЛАДНО"); row.Cells[0].Format.Font.Bold = true; row = table.AddRow(); row.Cells[0].MergeRight = 2; row.Cells[0].AddParagraph("Ідентифікатор, тип та номер транспорту"); row.Cells[3].MergeRight = 1; row.Cells[3].AddParagraph("Поїздок заплановано, проведена та скасовано"); row.Cells[5].MergeRight = 2; row.Cells[5].AddParagraph("Квитків продано та повернено, з яких неповні"); row.Cells[8].MergeRight = 1; row.Cells[8].AddParagraph("Грошей зароблено"); row.Cells[10].MergeRight = 1; row.Cells[10].AddParagraph("Середній рейтинг"); var isFilled = true; foreach (var vehicle in vehicleEnrolled) { if (route.VehicleEnrollments.Count(ve => ve.VehicleId == vehicle.Id) == 0) { continue; } row = table.AddRow(); row.Shading.Color = isFilled ? Color.FromRgbColor(25, Colors.Black) : Colors.White; isFilled = !isFilled; row.Cells[0].MergeRight = 2; row.Cells[0].AddParagraph($"{vehicle.Id}, {vehicle.Type}, {vehicle.Number}"); var executedEnrollmentCount = vehicle.GetRouteEnrollmentCount(fromDate, toDate, route.Id); var canceledEnrollmentCount = vehicle.GetRouteCanceledEnrollmentCount(fromDate, toDate, route.Id); row.Cells[3].MergeRight = 1; row.Cells[3].AddParagraph($"{executedEnrollmentCount + canceledEnrollmentCount}, " + $"{executedEnrollmentCount}, {canceledEnrollmentCount}"); row.Cells[5].MergeRight = 2; row.Cells[5].AddParagraph($"{vehicle.GetRouteSoldTicketCount(fromDate, toDate, route.Id)}, " + $"{vehicle.GetRouteReturnedTicketCount(fromDate, toDate, route.Id)}; " + $"{vehicle.GetRouteIndirectTicketCount(fromDate, toDate, route.Id)}, " + $"{vehicle.GetRouteReturnedIndirectTicketCount(fromDate, toDate, route.Id)}"); row.Cells[8].MergeRight = 1; row.Cells[8].AddParagraph($"{vehicle.GetRouteTotalRevenue(fromDate, toDate, route.Id)}"); row.Cells[10].MergeRight = 1; var vehicleAverageRating = vehicle.GetRouteAverageRating(fromDate, toDate, route.Id); row.Cells[10].AddParagraph($"{(vehicleAverageRating == 0 ? "-" : vehicleAverageRating)}"); } i++; } section = doc.Sections[doc.Sections.Count - 1]; paragraph = section.AddParagraph("ПІДСУМОК"); paragraph.Format.Alignment = ParagraphAlignment.Center; paragraph.Format.Font.Size = 14; paragraph.Format.Font.Bold = true; section.AddParagraph(); paragraph = section.AddParagraph( $"У період з {fromDate:dd.MM.yyyy} по {toDate:dd.MM.yyyy} " + $"({(toDate - fromDate).Days} днів) компанією {dbCompany.Name} " + $"було заплановано {dbCompany.GetTotalEnrollmentCount(fromDate, toDate)} поїздки, " + $"з яких {dbCompany.GetTotalCanceledEnrollmentCount(fromDate, toDate)} було скасовано, " + $"продано {dbCompany.GetTotalSoldTicketCount(fromDate, toDate)} квитків, " + $"з яких {dbCompany.GetTotalReturnedTicketCount(fromDate, toDate)} було повернено. " + $"За цей час було зароблено {dbCompany.GetTotalRevenue(fromDate, toDate)} гривень. " + $"Середній рейтинг по всім поїздкам: {dbCompany.GetTotalAverageRating(fromDate, toDate)}"); paragraph.Format.Alignment = ParagraphAlignment.Justify; paragraph.Format.Font.Size = 14; } } public async Task<(bool isSucceed, IActionResult? actionResult, StatisticsResponse statistics)> GetCompanyReportRaw(int? companyId, DateTime fromDate, DateTime toDate) { if (_sessionUserService.GetAuthUserRole() == Identity.Roles.Administrator.ToString()) { if (companyId == null) { return (false, new BadRequestObjectResult("Query parameter CompanyId must have a value"), null!); } } else { var result = await _sessionUserService.IsAuthUserCompanyOwner(); if (!result.isCompanyOwner) { return (false, new UnauthorizedResult(), null!); } companyId = result.companyId; } if (!await DoesCompanyExist((int)companyId)) { return (false, new NotFoundResult(), null!); } var dbCompany = await _dbContext.Companies .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Reviews) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Tickets) .ThenInclude(t => t.TicketGroup) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Route) .ThenInclude(r => r.RouteAddresses) .ThenInclude(ra => ra.Address) .ThenInclude(a => a.City) .ThenInclude(c => c.State) .ThenInclude(s => s.Country) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.RouteAddressDetails) .Select(c => new { Company = c, VehicleEnrollments = c.Vehicles .Select(v => new { Vehicle = v, VehicleEnrollments = v.VehicleEnrollments.Where(ve => ve.DepartureDateTimeUtc.Date >= fromDate && ve.DepartureDateTimeUtc <= toDate) }) }) .Select(o => o.Company) .FirstAsync(c => c.Id == companyId); var statistics = new StatisticsResponse { EnrollmentsPlanned = dbCompany.GetTotalEnrollmentCount(fromDate, toDate), EnrollmentsCanceled = dbCompany.GetTotalCanceledEnrollmentCount(fromDate, toDate), TicketsSold = dbCompany.GetTotalSoldTicketCount(fromDate, toDate), TicketsReturned = dbCompany.GetTotalReturnedTicketCount(fromDate, toDate), MoneyEarned = dbCompany.GetTotalRevenue(fromDate, toDate), AverageRating = dbCompany.GetTotalAverageRating(fromDate, toDate) }; return (true, null, statistics); } public async Task<(bool isSucceed, IActionResult? actionResult, StatisticsResponse statistics)> GetAdminReportRaw(DateTime fromDate, DateTime toDate) { var dbCompanies = await _dbContext.Companies .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Reviews) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Tickets) .ThenInclude(t => t.TicketGroup) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.Route) .ThenInclude(r => r.RouteAddresses) .ThenInclude(ra => ra.Address) .ThenInclude(a => a.City) .ThenInclude(c => c.State) .ThenInclude(s => s.Country) .Include(c => c.Vehicles) .ThenInclude(v => v.VehicleEnrollments) .ThenInclude(ve => ve.RouteAddressDetails) .Select(c => new { Company = c, VehicleEnrollments = c.Vehicles .Select(v => new { Vehicle = v, VehicleEnrollments = v.VehicleEnrollments.Where(ve => ve.DepartureDateTimeUtc.Date >= fromDate && ve.DepartureDateTimeUtc <= toDate) }) }) .Select(o => o.Company) .ToArrayAsync(); var statistics = new StatisticsResponse(); foreach (var company in dbCompanies) { statistics.EnrollmentsPlanned += company.GetTotalEnrollmentCount(fromDate, toDate); statistics.EnrollmentsCanceled += company.GetTotalCanceledEnrollmentCount(fromDate, toDate); statistics.TicketsSold += company.GetTotalSoldTicketCount(fromDate, toDate); statistics.TicketsReturned += company.GetTotalReturnedTicketCount(fromDate, toDate); statistics.MoneyEarned += company.GetTotalRevenue(fromDate, toDate); statistics.AverageRating += company.GetTotalAverageRating(fromDate, toDate); } return (true, null, statistics); } async Task DoesCompanyExist(int id) { return await _dbContext.Companies.AnyAsync(c => c.Id == id); } async Task DoesTicketGroupExist(int id) { return await _dbContext.TicketGroups.AnyAsync(tg => tg.Id == id); } }