using AsbCloudApp.Data; using AsbCloudApp.Services; using AsbCloudWebApi.SignalR; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.SignalR; using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Threading.Tasks; namespace AsbCloudWebApi.Controllers { /// /// Контроллер отчетов по буровым скважинам /// [Route("api/report")] [ApiController] public class ReportController : ControllerBase { private readonly IReportService reportService; private readonly IFileService fileService; private readonly IWellService wellService; private readonly IHubContext reportsHubContext; public ReportController(IReportService reportService, IWellService wellService, IFileService fileService, IHubContext reportsHubContext) { this.reportService = reportService; this.fileService = fileService; this.wellService = wellService; this.reportsHubContext = reportsHubContext; } private void HandleReportProgressAsync(float progress, string operation, int id) => Task.Run(() => { reportsHubContext.Clients.Group($"Report_{id}").SendAsync( nameof(IReportHubClient.GetReportProgress), new { Progress = progress, Operation = operation, ReportName = "" } ); }); private void HandleReportNameAsync(string reportName, int groupId) => Task.Run(() => { reportsHubContext.Clients.All.SendAsync( nameof(IReportHubClient.GetReportProgress), new { Progress = 100, Operation = "Отчет успешно создан", ReportName = reportName } ); }); /// /// Создает отчет по скважине с указанными параметрами /// /// id скважины /// id пользователя /// шаг интервала /// формат отчета (0-PDF, 1-LAS) /// дата начала интервала /// дата окончания интервала /// id фоновой задачи формирования отчета [HttpPost] [Route("{idWell}/report")] [ProducesResponseType(typeof(int), (int)System.Net.HttpStatusCode.OK)] public IActionResult CreateReport(int idWell, int idUser, int stepSeconds, int format, DateTime begin = default, DateTime end = default) { int? idCompany = User.GetCompanyId(); if (idCompany is null) return Forbid(); if (!wellService.IsCompanyInvolvedInWell((int)idCompany, idWell)) return Forbid(); var id = reportService.CreateReport(idWell, idUser, stepSeconds, format, begin, end, HandleReportProgressAsync, HandleReportNameAsync); return Ok(id); } /// /// Возвращает файл-отчет с диска на сервере /// /// id скважины /// имя запрашиваемого файла (отчета) /// файл с отчетом [HttpGet] [Route("{idWell}/{reportName}")] [ProducesResponseType(typeof(PhysicalFileResult), (int)System.Net.HttpStatusCode.OK)] public IActionResult GetReport([FromRoute] int idWell, string reportName) { try { int? idCompany = User.GetCompanyId(); if (idCompany is null) return Forbid(); if (!wellService.IsCompanyInvolvedInWell((int)idCompany, idWell)) return Forbid(); // TODO: словарь content typoв var relativePath = Path.Combine(fileService.RootPath, $"{idWell}", $"{reportService.ReportCategoryId}", reportName); return PhysicalFile(Path.GetFullPath(relativePath), "application/pdf", reportName); } catch (FileNotFoundException ex) { return NotFound($"Файл не найден. Текст ошибки: {ex.Message}"); } } /// /// Возвращает имена отчетов, хранящихся на диске, /// которые подходят под указанные параметры /// /// id скважины /// шаг интервала /// формат отчета (0-PDF, 1-LAS) /// дата начала интервала /// дата окончания интервала /// Список имен существующих отчетов (отчетов) [HttpGet] [Route("{idWell}/suitableReports")] [ProducesResponseType(typeof(IEnumerable), (int)System.Net.HttpStatusCode.OK)] public IActionResult GetSuitableReportsNames(int idWell, int stepSeconds, int format, DateTime begin = default, DateTime end = default) { var suitableReportsNames = reportService.GetSuitableReports(idWell, begin, end, stepSeconds, format); if (suitableReportsNames is null || !suitableReportsNames.Any()) return NoContent(); return Ok(suitableReportsNames); } /// /// Возвращает прогнозируемое количество страниц будущего отчета /// /// id скважины /// дата начала интервала /// дата окончания интервала /// шаг интервала /// формат отчета (0-PDF, 1-LAS) /// прогнозируемое кол-во страниц отчета [HttpGet] [Route("{idWell}/reportSize")] [ProducesResponseType(typeof(string), (int)System.Net.HttpStatusCode.OK)] public IActionResult GetReportSize(int idWell, int stepSeconds, int format, DateTime begin = default, DateTime end = default) { int? idCompany = User.GetCompanyId(); if (idCompany is null) return Forbid(); if (!wellService.IsCompanyInvolvedInWell((int)idCompany, idWell)) return Forbid(); int reportSize = reportService.GetReportPagesCount(idWell, begin, end, stepSeconds, format); return Ok(reportSize); } /// /// Возвращает даты самого старого и самого свежего отчетов в БД /// /// id скважины /// Даты самого старого и самого свежего отчетов в БД [HttpGet] [Route("{idWell}/reportsDatesRange")] [ProducesResponseType(typeof(DatesRangeDto), (int)System.Net.HttpStatusCode.OK)] public IActionResult GetReportsDateRange(int idWell) { int? idCompany = User.GetCompanyId(); if (idCompany is null) return Forbid(); if (!wellService.IsCompanyInvolvedInWell((int)idCompany, idWell)) return Forbid(); DatesRangeDto wellReportsDatesRange = reportService.GetReportsDatesRange(idWell); return Ok(wellReportsDatesRange); } } }