auto.bus_api/Server/Services/ReportService.cs
cuqmbr e9af067dfa feat: add imperative resource-based authorization
I decided not to make authorization requirements and handlers for each and every resource because the validation logic is pretty similar
2023-05-22 15:27:11 +03:00

886 lines
35 KiB
C#
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

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<Route>();
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<bool> DoesCompanyExist(int id)
{
return await _dbContext.Companies.AnyAsync(c => c.Id == id);
}
async Task<bool> DoesTicketGroupExist(int id)
{
return await _dbContext.TicketGroups.AnyAsync(tg => tg.Id == id);
}
}