forked from ddrilling/AsbCloudServer
Merge branch 'fix/#23919905-daily-diagram-create-report-from-signalr' of http://test.digitaldrilling.ru:8080/DDrilling/AsbCloudServer into fix/#23919905-daily-diagram-create-report-from-signalr
This commit is contained in:
commit
678b409ca6
@ -1,5 +1,5 @@
|
|||||||
using System.Threading;
|
using System.Collections.Generic;
|
||||||
using System.Threading.Tasks;
|
using AsbCloudApp.Data;
|
||||||
using AsbCloudApp.Data.WellOperationImport;
|
using AsbCloudApp.Data.WellOperationImport;
|
||||||
|
|
||||||
namespace AsbCloudApp.Services.WellOperationImport;
|
namespace AsbCloudApp.Services.WellOperationImport;
|
||||||
@ -16,7 +16,5 @@ public interface IWellOperationImportService
|
|||||||
/// <param name="idUser"></param>
|
/// <param name="idUser"></param>
|
||||||
/// <param name="idType"></param>
|
/// <param name="idType"></param>
|
||||||
/// <param name="sheet"></param>
|
/// <param name="sheet"></param>
|
||||||
/// <param name="deleteBeforeImport"></param>
|
IEnumerable<WellOperationDto> Import(int idWell, int idUser, int idType, SheetDto sheet);
|
||||||
/// <param name="cancellationToken"></param>
|
|
||||||
Task ImportAsync(int idWell, int idUser, int idType, SheetDto sheet, bool deleteBeforeImport, CancellationToken cancellationToken);
|
|
||||||
}
|
}
|
@ -113,17 +113,12 @@ sudo -u postgres psql
|
|||||||
SELECT * FROM pg_stat_replication;
|
SELECT * FROM pg_stat_replication;
|
||||||
```
|
```
|
||||||
|
|
||||||
7. Для включения синхронного режима необходимо выполнить следующую команду
|
7. Сделать рестарт primary-сервера.
|
||||||
```
|
|
||||||
ALTER SYSTEM SET synchronous_standby_names TO '*';
|
|
||||||
```
|
|
||||||
|
|
||||||
8. Сделать рестарт primary-сервера.
|
8. Внести запись в любую таблицу базы данных primary-сервера
|
||||||
|
9. Убедиться, что соответствующая запись появилась в таблице базы данных standby-сервера
|
||||||
9. Внести запись в любую таблицу базы данных primary-сервера
|
10. Попытаться внести запись в таблицу базы данных standby-сервера.
|
||||||
10. Убедиться, что соответствующая запись появилась в таблице базы данных standby-сервера
|
11. Убедиться, что операция завершилась с ошибкой
|
||||||
11. Попытаться внести запись в таблицу базы данных standby-сервера.
|
|
||||||
12. Убедиться, что операция завершилась с ошибкой
|
|
||||||
> cannot execute OPERATION in a read-only transaction
|
> cannot execute OPERATION in a read-only transaction
|
||||||
|
|
||||||
|
|
||||||
|
Binary file not shown.
@ -2,12 +2,9 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using AsbCloudApp.Data;
|
using AsbCloudApp.Data;
|
||||||
using AsbCloudApp.Data.WellOperationImport;
|
using AsbCloudApp.Data.WellOperationImport;
|
||||||
using AsbCloudApp.Repositories;
|
using AsbCloudApp.Repositories;
|
||||||
using AsbCloudApp.Requests;
|
|
||||||
using AsbCloudApp.Services.WellOperationImport;
|
using AsbCloudApp.Services.WellOperationImport;
|
||||||
|
|
||||||
namespace AsbCloudInfrastructure.Services.WellOperationImport;
|
namespace AsbCloudInfrastructure.Services.WellOperationImport;
|
||||||
@ -25,14 +22,14 @@ public class WellOperationImportService : IWellOperationImportService
|
|||||||
this.wellOperationRepository = wellOperationRepository;
|
this.wellOperationRepository = wellOperationRepository;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task ImportAsync(int idWell, int idUser, int idType, SheetDto sheet, bool deleteBeforeImport, CancellationToken cancellationToken)
|
public IEnumerable<WellOperationDto> Import(int idWell, int idUser, int idType, SheetDto sheet)
|
||||||
{
|
{
|
||||||
var validationErrors = new List<string>();
|
var validationErrors = new List<string>();
|
||||||
|
|
||||||
var sections = wellOperationRepository.GetSectionTypes();
|
var sections = wellOperationRepository.GetSectionTypes();
|
||||||
var categories = wellOperationRepository.GetCategories(false);
|
var categories = wellOperationRepository.GetCategories(false);
|
||||||
|
|
||||||
var operations = new List<WellOperationDto>();
|
var wellOperations = new List<WellOperationDto>();
|
||||||
|
|
||||||
foreach (var row in sheet.Rows)
|
foreach (var row in sheet.Rows)
|
||||||
{
|
{
|
||||||
@ -62,14 +59,14 @@ public class WellOperationImportService : IWellOperationImportService
|
|||||||
throw new FileFormatException(
|
throw new FileFormatException(
|
||||||
$"Лист '{sheet.Name}'. Строка '{row.Number}' неправильно получена дата начала операции");
|
$"Лист '{sheet.Name}'. Строка '{row.Number}' неправильно получена дата начала операции");
|
||||||
|
|
||||||
if (operations.LastOrDefault()?.DateStart > row.Date)
|
if (wellOperations.LastOrDefault()?.DateStart > row.Date)
|
||||||
throw new FileFormatException(
|
throw new FileFormatException(
|
||||||
$"Лист '{sheet.Name}' строка '{row.Number}' дата позднее даты предыдущей операции");
|
$"Лист '{sheet.Name}' строка '{row.Number}' дата позднее даты предыдущей операции");
|
||||||
|
|
||||||
if (row.Duration is not (>= 0d and <= 240d))
|
if (row.Duration is not (>= 0d and <= 240d))
|
||||||
throw new FileFormatException($"Лист '{sheet.Name}'. Строка '{row.Number}' некорректная длительность операции");
|
throw new FileFormatException($"Лист '{sheet.Name}'. Строка '{row.Number}' некорректная длительность операции");
|
||||||
|
|
||||||
operations.Add(new WellOperationDto
|
wellOperations.Add(new WellOperationDto
|
||||||
{
|
{
|
||||||
IdWell = idWell,
|
IdWell = idWell,
|
||||||
IdUser = idUser,
|
IdUser = idUser,
|
||||||
@ -89,26 +86,12 @@ public class WellOperationImportService : IWellOperationImportService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (operations.Any() && operations.Min(o => o.DateStart) - operations.Max(o => o.DateStart) > drillingDurationLimitMax)
|
if (wellOperations.Any() && wellOperations.Min(o => o.DateStart) - wellOperations.Max(o => o.DateStart) > drillingDurationLimitMax)
|
||||||
validationErrors.Add($"Лист {sheet.Name} содержит диапазон дат больше {drillingDurationLimitMax}");
|
validationErrors.Add($"Лист {sheet.Name} содержит диапазон дат больше {drillingDurationLimitMax}");
|
||||||
|
|
||||||
if (validationErrors.Any())
|
if (validationErrors.Any())
|
||||||
throw new FileFormatException(string.Join("\r\n", validationErrors));
|
throw new FileFormatException(string.Join("\r\n", validationErrors));
|
||||||
|
|
||||||
if (!operations.Any())
|
return wellOperations;
|
||||||
return;
|
|
||||||
|
|
||||||
if (deleteBeforeImport)
|
|
||||||
{
|
|
||||||
var existingOperations = await wellOperationRepository.GetAsync(new WellOperationRequest
|
|
||||||
{
|
|
||||||
IdWell = idWell,
|
|
||||||
OperationType = idType
|
|
||||||
}, cancellationToken);
|
|
||||||
|
|
||||||
await wellOperationRepository.DeleteAsync(existingOperations.Select(o => o.Id), cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
await wellOperationRepository.InsertRangeAsync(operations, cancellationToken);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -254,7 +254,7 @@ public class DailyReportServiceTest
|
|||||||
wellServiceMock.GetOrDefaultAsync(Arg.Any<int>(), Arg.Any<CancellationToken>())
|
wellServiceMock.GetOrDefaultAsync(Arg.Any<int>(), Arg.Any<CancellationToken>())
|
||||||
.ReturnsForAnyArgs(fakeWell);
|
.ReturnsForAnyArgs(fakeWell);
|
||||||
|
|
||||||
trajectoryFactNnbRepositoryMock.GetAsync(Arg.Any<int>(), Arg.Any<CancellationToken>())
|
trajectoryFactNnbRepositoryMock.GetByRequestAsync(Arg.Any<TrajectoryRequest>(), Arg.Any<CancellationToken>())
|
||||||
.ReturnsForAnyArgs(new[] { fakeLastFactTrajectory });
|
.ReturnsForAnyArgs(new[] { fakeLastFactTrajectory });
|
||||||
|
|
||||||
wellOperationRepositoryMock.GetAsync(Arg.Any<WellOperationRequest>(), Arg.Any<CancellationToken>())
|
wellOperationRepositoryMock.GetAsync(Arg.Any<WellOperationRequest>(), Arg.Any<CancellationToken>())
|
||||||
|
@ -1,72 +1,119 @@
|
|||||||
using Microsoft.AspNetCore.Mvc;
|
using AsbCloudApp.Exceptions;
|
||||||
|
using AsbCloudWebApi.SignalR;
|
||||||
|
using Microsoft.AspNetCore.Authorization;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using Microsoft.AspNetCore.SignalR;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.ComponentModel.DataAnnotations;
|
using System.ComponentModel.DataAnnotations;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
namespace AsbCloudWebApi.Controllers
|
namespace AsbCloudWebApi.Controllers;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Имитирует разные типы ответа сервера
|
||||||
|
/// </summary>
|
||||||
|
[Route("api/[controller]")]
|
||||||
|
[ApiController]
|
||||||
|
public class MockController : ControllerBase
|
||||||
{
|
{
|
||||||
/// <summary>
|
private readonly IServiceProvider provider;
|
||||||
/// Имитирует разные типы ответа сервера
|
|
||||||
/// </summary>
|
public MockController(IServiceProvider provider)
|
||||||
[Route("api/[controller]")]
|
|
||||||
[ApiController]
|
|
||||||
public class MockController : ControllerBase
|
|
||||||
{
|
{
|
||||||
/// <summary>
|
this.provider = provider;
|
||||||
/// имитирует http-400
|
}
|
||||||
/// </summary>
|
|
||||||
[HttpGet("400")]
|
/// <summary>
|
||||||
[ProducesResponseType(typeof(ValidationProblemDetails), (int)System.Net.HttpStatusCode.BadRequest)]
|
/// имитирует http-400
|
||||||
public IActionResult Get400([FromQuery, Required]IDictionary<string, string> args)
|
/// </summary>
|
||||||
|
[HttpGet("400")]
|
||||||
|
[ProducesResponseType(typeof(ValidationProblemDetails), (int)System.Net.HttpStatusCode.BadRequest)]
|
||||||
|
public IActionResult Get400([FromQuery, Required]IDictionary<string, string> args)
|
||||||
|
{
|
||||||
|
var errors = new Dictionary<string, string[]>();
|
||||||
|
|
||||||
|
foreach (var arg in args)
|
||||||
{
|
{
|
||||||
var errors = new Dictionary<string, string[]>();
|
var countOfErrors = ((arg.Key + arg.Value).Length % 3) + 1;
|
||||||
|
var errorsText = Enumerable.Range(0, countOfErrors)
|
||||||
foreach (var arg in args)
|
.Select(i => $"{arg.Value} не соответствует критериям проверки № {i}");
|
||||||
{
|
|
||||||
var countOfErrors = ((arg.Key + arg.Value).Length % 3) + 1;
|
|
||||||
var errorsText = Enumerable.Range(0, countOfErrors)
|
|
||||||
.Select(i => $"{arg.Value} не соответствует критериям проверки № {i}");
|
|
||||||
|
|
||||||
errors.Add(arg.Key, errorsText.ToArray());
|
errors.Add(arg.Key, errorsText.ToArray());
|
||||||
}
|
|
||||||
|
|
||||||
if (errors.Any())
|
|
||||||
{
|
|
||||||
var problem = new ValidationProblemDetails(errors);
|
|
||||||
return BadRequest(problem);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var problem = new ValidationProblemDetails { Detail = "at least one argument must be provided" };
|
|
||||||
return BadRequest(problem);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
if (errors.Any())
|
||||||
/// имитирует http-403
|
|
||||||
/// </summary>
|
|
||||||
[HttpGet("403")]
|
|
||||||
public IActionResult Get403()
|
|
||||||
{
|
{
|
||||||
return Forbid();
|
var problem = new ValidationProblemDetails(errors);
|
||||||
|
return BadRequest(problem);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
/// <summary>
|
|
||||||
/// имитирует http-401
|
|
||||||
/// </summary>
|
|
||||||
[HttpGet("401")]
|
|
||||||
public IActionResult Get401()
|
|
||||||
{
|
{
|
||||||
return Unauthorized();
|
var problem = new ValidationProblemDetails { Detail = "at least one argument must be provided" };
|
||||||
}
|
return BadRequest(problem);
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// имитирует http-500
|
|
||||||
/// </summary>
|
|
||||||
[HttpGet("500")]
|
|
||||||
public IActionResult Get500()
|
|
||||||
{
|
|
||||||
throw new System.Exception("Это тестовое исключение");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// имитирует http-403
|
||||||
|
/// </summary>
|
||||||
|
[HttpGet("403")]
|
||||||
|
public IActionResult Get403()
|
||||||
|
{
|
||||||
|
return Forbid();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// имитирует http-401
|
||||||
|
/// </summary>
|
||||||
|
[HttpGet("401")]
|
||||||
|
public IActionResult Get401()
|
||||||
|
{
|
||||||
|
return Unauthorized();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// имитирует http-500
|
||||||
|
/// </summary>
|
||||||
|
[HttpGet("500")]
|
||||||
|
public IActionResult Get500()
|
||||||
|
{
|
||||||
|
throw new System.Exception("Это тестовое исключение");
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// имитация отправки SignalR данных
|
||||||
|
/// </summary>
|
||||||
|
/// <example>
|
||||||
|
/// </example>
|
||||||
|
/// <param name="hubName">
|
||||||
|
/// Поддерживаемые hubЫ: wellInfo, notifications, telemetry, reports
|
||||||
|
/// </param>
|
||||||
|
/// <param name="methodName">Название вызываемого на клиенте метода. Прим.:"ReceiveDataSaub". Список методов см. в swagger definition signalr</param>
|
||||||
|
/// <param name="groupName">Группа пользователей. Прим.: "well_1". Если не задана - все пользователи. Шаблон формирования групп см. описание методов в swagger definition signalr</param>
|
||||||
|
/// <param name="body">передаваемая нагрузка. (json)</param>
|
||||||
|
/// <param name="token"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
[HttpPost("signalr/hubs/{hubName}/{methodName}/{groupName}")]
|
||||||
|
[Authorize]
|
||||||
|
public async Task<IActionResult> PostAsync(string hubName, string methodName, string? groupName, object body, CancellationToken token)
|
||||||
|
{
|
||||||
|
IHubClients clients = hubName.ToLower() switch {
|
||||||
|
"wellinfo" => provider.GetRequiredService<IHubContext<NotificationHub>>().Clients,
|
||||||
|
"notifications" => provider.GetRequiredService<IHubContext<NotificationHub>>().Clients,
|
||||||
|
"telemetry" => provider.GetRequiredService<IHubContext<TelemetryHub>>().Clients,
|
||||||
|
"reports" => provider.GetRequiredService<IHubContext<ReportsHub>>().Clients,
|
||||||
|
_ => throw new ArgumentInvalidException(nameof(hubName), "hubName does not listed"),
|
||||||
|
};
|
||||||
|
|
||||||
|
IClientProxy selectedClients = string.IsNullOrEmpty(groupName)
|
||||||
|
? clients.All
|
||||||
|
: clients.Group(groupName);
|
||||||
|
|
||||||
|
await selectedClients.SendAsync(methodName, body, token);
|
||||||
|
return Ok();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,8 +9,10 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.ComponentModel.DataAnnotations;
|
using System.ComponentModel.DataAnnotations;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using AsbCloudApp.Data.WellOperationImport;
|
||||||
using AsbCloudApp.Services.WellOperationImport;
|
using AsbCloudApp.Services.WellOperationImport;
|
||||||
using AsbCloudApp.Data.WellOperationImport.Options;
|
using AsbCloudApp.Data.WellOperationImport.Options;
|
||||||
using AsbCloudApp.Exceptions;
|
using AsbCloudApp.Exceptions;
|
||||||
@ -204,37 +206,87 @@ namespace AsbCloudWebApi.Controllers
|
|||||||
return Ok(result);
|
return Ok(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Добавляет новую операцию на скважину
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="idWell">Id скважины</param>
|
||||||
|
/// <param name="idType">Тип добавляемой операции</param>
|
||||||
|
/// <param name="wellOperation">Добавляемая операция</param>
|
||||||
|
/// <param name="cancellationToken"></param>
|
||||||
|
/// <returns>Количество добавленных в БД записей</returns>
|
||||||
|
[HttpPost("{idType:int}")]
|
||||||
|
[Permission]
|
||||||
|
[ProducesResponseType(typeof(int), StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(typeof(ValidationProblemDetails), StatusCodes.Status400BadRequest)]
|
||||||
|
public async Task<IActionResult> InsertAsync(
|
||||||
|
[Range(1, int.MaxValue, ErrorMessage = "Id скважины не может быть меньше 1")] int idWell,
|
||||||
|
[Range(0, 1, ErrorMessage = "Тип операции недопустим. Допустимые: 0, 1")] int idType,
|
||||||
|
WellOperationDto wellOperation,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
if (!await CanUserAccessToWellAsync(idWell, cancellationToken))
|
||||||
|
return Forbid();
|
||||||
|
|
||||||
|
if (!await CanUserEditWellOperationsAsync(idWell, cancellationToken))
|
||||||
|
return Forbid();
|
||||||
|
|
||||||
|
wellOperation.IdWell = idWell;
|
||||||
|
wellOperation.LastUpdateDate = DateTimeOffset.UtcNow;
|
||||||
|
wellOperation.IdUser = User.GetUserId();
|
||||||
|
wellOperation.IdType = idType;
|
||||||
|
|
||||||
|
var result = await operationRepository.InsertRangeAsync(new[] { wellOperation }, cancellationToken);
|
||||||
|
|
||||||
|
return Ok(result);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Добавляет новые операции на скважине
|
/// Добавляет новые операции на скважине
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="idWell">id скважины</param>
|
/// <param name="idWell">Id скважины</param>
|
||||||
/// <param name="values">Данные о добавляемых операциях</param>
|
/// <param name="wellOperations">Добавляемые операции</param>
|
||||||
/// <param name="token">Токен отмены задачи</param>
|
/// <param name="idType">Тип добавляемых операций</param>
|
||||||
/// <returns>Количество добавленных в БД строк</returns>
|
/// <param name="deleteBeforeInsert">Удалить операции перед сохранением</param>
|
||||||
[HttpPost]
|
/// <param name="cancellationToken"></param>
|
||||||
|
/// <returns>Количество добавленных в БД записей</returns>
|
||||||
|
[HttpPost("{idType:int}/{deleteBeforeInsert:bool}")]
|
||||||
[Permission]
|
[Permission]
|
||||||
[ProducesResponseType(typeof(IEnumerable<WellOperationDto>), (int)System.Net.HttpStatusCode.OK)]
|
[ProducesResponseType(typeof(int), StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(typeof(ValidationProblemDetails), StatusCodes.Status400BadRequest)]
|
||||||
public async Task<IActionResult> InsertRangeAsync(
|
public async Task<IActionResult> InsertRangeAsync(
|
||||||
[Range(1, int.MaxValue, ErrorMessage = "Id скважины не может быть меньше 1")] int idWell,
|
[Range(1, int.MaxValue, ErrorMessage = "Id скважины не может быть меньше 1")] int idWell,
|
||||||
[FromBody] IEnumerable<WellOperationDto> values,
|
[Range(0, 1, ErrorMessage = "Тип операции недопустим. Допустимые: 0, 1")] int idType,
|
||||||
CancellationToken token)
|
bool deleteBeforeInsert,
|
||||||
|
[FromBody] IEnumerable<WellOperationDto> wellOperations,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
if (!await CanUserAccessToWellAsync(idWell, token))
|
if (!await CanUserAccessToWellAsync(idWell, cancellationToken))
|
||||||
return Forbid();
|
|
||||||
|
|
||||||
if (!await CanUserEditWellOperationsAsync(idWell, token))
|
|
||||||
return Forbid();
|
return Forbid();
|
||||||
|
|
||||||
foreach (var value in values)
|
if (!await CanUserEditWellOperationsAsync(idWell, cancellationToken))
|
||||||
|
return Forbid();
|
||||||
|
|
||||||
|
if (deleteBeforeInsert && wellOperations.Any())
|
||||||
{
|
{
|
||||||
value.IdWell = idWell;
|
var existingOperations = await operationRepository.GetAsync(new WellOperationRequest
|
||||||
value.LastUpdateDate = DateTimeOffset.UtcNow;
|
{
|
||||||
value.IdUser = User.GetUserId();
|
IdWell = idWell,
|
||||||
|
OperationType = idType
|
||||||
|
}, cancellationToken);
|
||||||
|
|
||||||
|
await operationRepository.DeleteAsync(existingOperations.Select(o => o.Id), cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var wellOperation in wellOperations)
|
||||||
|
{
|
||||||
|
wellOperation.IdWell = idWell;
|
||||||
|
wellOperation.LastUpdateDate = DateTimeOffset.UtcNow;
|
||||||
|
wellOperation.IdUser = User.GetUserId();
|
||||||
|
wellOperation.IdType = idType;
|
||||||
}
|
}
|
||||||
|
|
||||||
var result = await operationRepository.InsertRangeAsync(values, token)
|
var result = await operationRepository.InsertRangeAsync(wellOperations, cancellationToken);
|
||||||
.ConfigureAwait(false);
|
|
||||||
|
|
||||||
return Ok(result);
|
return Ok(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -299,46 +351,19 @@ namespace AsbCloudWebApi.Controllers
|
|||||||
/// <param name="idWell">id скважины</param>
|
/// <param name="idWell">id скважины</param>
|
||||||
/// <param name="options">Параметры для парсинга файла</param>
|
/// <param name="options">Параметры для парсинга файла</param>
|
||||||
/// <param name="files">Коллекция из одного файла xlsx</param>
|
/// <param name="files">Коллекция из одного файла xlsx</param>
|
||||||
/// <param name="deleteBeforeImport">Удалить операции перед импортом = 1, если файл валидный</param>
|
/// <param name="cancellationToken"></param>
|
||||||
/// <param name="token"></param>
|
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
[HttpPost("import/default/{deleteBeforeImport}")]
|
[HttpPost("import/default")]
|
||||||
|
[ProducesResponseType(typeof(IEnumerable<WellOperationDto>), StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(typeof(ValidationProblemDetails), StatusCodes.Status400BadRequest)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status204NoContent)]
|
||||||
[Permission]
|
[Permission]
|
||||||
public async Task<IActionResult> ImportDefaultExcelFileAsync(int idWell,
|
public Task<IActionResult> ImportDefaultExcelFileAsync(int idWell,
|
||||||
[FromQuery] WellOperationImportDefaultOptionsDto options,
|
[FromQuery] WellOperationImportDefaultOptionsDto options,
|
||||||
[FromForm] IFormFileCollection files,
|
[FromForm] IFormFileCollection files,
|
||||||
[Range(0, 1, ErrorMessage = "Недопустимое значение. Допустимые: 0, 1")] int deleteBeforeImport,
|
CancellationToken cancellationToken) => ImportExcelFileAsync(idWell, files, options,
|
||||||
CancellationToken token)
|
(stream, _) => wellOperationDefaultExcelParser.Parse(stream, options),
|
||||||
{
|
cancellationToken);
|
||||||
var idUser = User.GetUserId();
|
|
||||||
|
|
||||||
if (!idUser.HasValue)
|
|
||||||
throw new ForbidException("Неизвестный пользователь");
|
|
||||||
|
|
||||||
await AssertUserHasAccessToImportWellOperationsAsync(idWell, token);
|
|
||||||
|
|
||||||
if (files.Count < 1)
|
|
||||||
return this.ValidationBadRequest(nameof(files), "Нет файла");
|
|
||||||
|
|
||||||
var file = files[0];
|
|
||||||
if (Path.GetExtension(file.FileName).ToLower() != ".xlsx")
|
|
||||||
return this.ValidationBadRequest(nameof(files), "Требуется xlsx файл.");
|
|
||||||
|
|
||||||
using Stream stream = file.OpenReadStream();
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var sheet = wellOperationDefaultExcelParser.Parse(stream, options);
|
|
||||||
|
|
||||||
await wellOperationImportService.ImportAsync(idWell, idUser.Value, options.IdType, sheet, (deleteBeforeImport & 1) > 0, token);
|
|
||||||
}
|
|
||||||
catch (FileFormatException ex)
|
|
||||||
{
|
|
||||||
return this.ValidationBadRequest(nameof(files), ex.Message);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ok();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Импорт операций из excel (xlsx) файла. ГПНХ (Хантос)
|
/// Импорт операций из excel (xlsx) файла. ГПНХ (Хантос)
|
||||||
@ -346,46 +371,19 @@ namespace AsbCloudWebApi.Controllers
|
|||||||
/// <param name="idWell">id скважины</param>
|
/// <param name="idWell">id скважины</param>
|
||||||
/// <param name="options">Параметры для парсинга файла</param>
|
/// <param name="options">Параметры для парсинга файла</param>
|
||||||
/// <param name="files">Коллекция из одного файла xlsx</param>
|
/// <param name="files">Коллекция из одного файла xlsx</param>
|
||||||
/// <param name="deleteBeforeImport">Удалить операции перед импортом = 1, если файл валидный</param>
|
/// <param name="cancellationToken"></param>
|
||||||
/// <param name="token"></param>
|
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
[HttpPost("import/gazpromKhantos/{deleteBeforeImport}")]
|
[HttpPost("import/gazpromKhantos")]
|
||||||
|
[ProducesResponseType(typeof(IEnumerable<WellOperationDto>), StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(typeof(ValidationProblemDetails), StatusCodes.Status400BadRequest)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status204NoContent)]
|
||||||
[Permission]
|
[Permission]
|
||||||
public async Task<IActionResult> ImportGazpromKhantosExcelFileAsync(int idWell,
|
public Task<IActionResult> ImportGazpromKhantosExcelFileAsync(int idWell,
|
||||||
[FromQuery] WellOperationImportGazpromKhantosOptionsDto options,
|
[FromQuery] WellOperationImportGazpromKhantosOptionsDto options,
|
||||||
[FromForm] IFormFileCollection files,
|
[FromForm] IFormFileCollection files,
|
||||||
[Range(0, 1, ErrorMessage = "Недопустимое значение. Допустимые: 0, 1")] int deleteBeforeImport,
|
CancellationToken cancellationToken) => ImportExcelFileAsync(idWell, files, options,
|
||||||
CancellationToken token)
|
(stream, _) => wellOperationGazpromKhantosExcelParser.Parse(stream, options),
|
||||||
{
|
cancellationToken);
|
||||||
var idUser = User.GetUserId();
|
|
||||||
|
|
||||||
if (!idUser.HasValue)
|
|
||||||
throw new ForbidException("Неизвестный пользователь");
|
|
||||||
|
|
||||||
await AssertUserHasAccessToImportWellOperationsAsync(idWell, token);
|
|
||||||
|
|
||||||
if (files.Count < 1)
|
|
||||||
return this.ValidationBadRequest(nameof(files), "Нет файла");
|
|
||||||
|
|
||||||
var file = files[0];
|
|
||||||
if (Path.GetExtension(file.FileName).ToLower() != ".xlsx")
|
|
||||||
return this.ValidationBadRequest(nameof(files), "Требуется xlsx файл.");
|
|
||||||
|
|
||||||
using Stream stream = file.OpenReadStream();
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var sheet = wellOperationGazpromKhantosExcelParser.Parse(stream, options);
|
|
||||||
|
|
||||||
await wellOperationImportService.ImportAsync(idWell, idUser.Value, options.IdType, sheet, (deleteBeforeImport & 1) > 0, token);
|
|
||||||
}
|
|
||||||
catch (FileFormatException ex)
|
|
||||||
{
|
|
||||||
return this.ValidationBadRequest(nameof(files), ex.Message);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ok();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Создает excel файл с операциями по скважине
|
/// Создает excel файл с операциями по скважине
|
||||||
@ -453,7 +451,11 @@ namespace AsbCloudWebApi.Controllers
|
|||||||
return File(stream, "application/octet-stream", fileName);
|
return File(stream, "application/octet-stream", fileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task AssertUserHasAccessToImportWellOperationsAsync(int idWell, CancellationToken token)
|
private async Task<IActionResult> ImportExcelFileAsync<TOptions>(int idWell, [FromForm] IFormFileCollection files,
|
||||||
|
TOptions options,
|
||||||
|
Func<Stream, TOptions, SheetDto> parseMethod,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
where TOptions : IWellOperationImportOptions
|
||||||
{
|
{
|
||||||
var idCompany = User.GetCompanyId();
|
var idCompany = User.GetCompanyId();
|
||||||
var idUser = User.GetUserId();
|
var idUser = User.GetUserId();
|
||||||
@ -461,16 +463,54 @@ namespace AsbCloudWebApi.Controllers
|
|||||||
if (!idCompany.HasValue || !idUser.HasValue)
|
if (!idCompany.HasValue || !idUser.HasValue)
|
||||||
throw new ForbidException("Неизвестный пользователь");
|
throw new ForbidException("Неизвестный пользователь");
|
||||||
|
|
||||||
if (!await CanUserAccessToWellAsync(idWell, token))
|
if (!await CanUserAccessToWellAsync(idWell, cancellationToken))
|
||||||
throw new ForbidException("Нет доступа к скважине");
|
throw new ForbidException("Нет доступа к скважине");
|
||||||
|
|
||||||
if (!await CanUserEditWellOperationsAsync(idWell, token))
|
if (!await CanUserEditWellOperationsAsync(idWell, cancellationToken))
|
||||||
throw new ForbidException("Недостаточно прав для редактирования ГГД на завершенной скважине");
|
throw new ForbidException("Недостаточно прав для редактирования ГГД на завершенной скважине");
|
||||||
|
|
||||||
if (!await wellService.IsCompanyInvolvedInWellAsync(idCompany.Value, idWell, token))
|
if (!await wellService.IsCompanyInvolvedInWellAsync(idCompany.Value, idWell, cancellationToken))
|
||||||
throw new ForbidException("Скважина недоступна для компании");
|
throw new ForbidException("Скважина недоступна для компании");
|
||||||
}
|
|
||||||
|
|
||||||
|
if (files.Count < 1)
|
||||||
|
return this.ValidationBadRequest(nameof(files), "Нет файла");
|
||||||
|
|
||||||
|
var file = files[0];
|
||||||
|
if (Path.GetExtension(file.FileName).ToLower() != ".xlsx")
|
||||||
|
return this.ValidationBadRequest(nameof(files), "Требуется xlsx файл.");
|
||||||
|
|
||||||
|
using Stream stream = file.OpenReadStream();
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var sheet = parseMethod(stream, options);
|
||||||
|
|
||||||
|
var wellOperations = wellOperationImportService.Import(idWell, idUser.Value, options.IdType, sheet)
|
||||||
|
.OrderBy(w => w.DateStart);
|
||||||
|
|
||||||
|
var dateStart = wellOperations.Min(w => w.DateStart);
|
||||||
|
|
||||||
|
foreach (var wellOperation in wellOperations)
|
||||||
|
wellOperation.Day = (wellOperation.DateStart - dateStart).TotalDays;
|
||||||
|
|
||||||
|
if (!wellOperations.Any())
|
||||||
|
return NoContent();
|
||||||
|
|
||||||
|
return Ok(wellOperations);
|
||||||
|
}
|
||||||
|
catch (FileFormatException ex)
|
||||||
|
{
|
||||||
|
return this.ValidationBadRequest(nameof(files), ex.Message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<bool> CanUserAccessToWellAsync(int idWell, CancellationToken token)
|
||||||
|
{
|
||||||
|
int? idCompany = User.GetCompanyId();
|
||||||
|
return idCompany is not null && await wellService.IsCompanyInvolvedInWellAsync((int)idCompany,
|
||||||
|
idWell, token).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
private async Task<bool> CanUserEditWellOperationsAsync(int idWell, CancellationToken token)
|
private async Task<bool> CanUserEditWellOperationsAsync(int idWell, CancellationToken token)
|
||||||
{
|
{
|
||||||
var idUser = User.GetUserId();
|
var idUser = User.GetUserId();
|
||||||
@ -485,13 +525,5 @@ namespace AsbCloudWebApi.Controllers
|
|||||||
|
|
||||||
return well.IdState != 2 || userRepository.HasPermission(idUser.Value, "WellOperation.editCompletedWell");
|
return well.IdState != 2 || userRepository.HasPermission(idUser.Value, "WellOperation.editCompletedWell");
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<bool> CanUserAccessToWellAsync(int idWell, CancellationToken token)
|
|
||||||
{
|
|
||||||
int? idCompany = User.GetCompanyId();
|
|
||||||
return idCompany is not null && await wellService.IsCompanyInvolvedInWellAsync((int)idCompany,
|
|
||||||
idWell, token).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user