2021-04-07 18:01:56 +05:00
|
|
|
|
using AsbCloudApp.Data;
|
2022-04-11 18:00:34 +05:00
|
|
|
|
using AsbCloudApp.Data.SAUB;
|
2021-04-07 18:01:56 +05:00
|
|
|
|
using AsbCloudApp.Services;
|
2022-04-11 18:00:34 +05:00
|
|
|
|
using AsbCloudDb;
|
2021-04-07 18:01:56 +05:00
|
|
|
|
using AsbCloudDb.Model;
|
2021-07-28 09:46:58 +05:00
|
|
|
|
using Mapster;
|
2022-04-11 18:00:34 +05:00
|
|
|
|
using Microsoft.EntityFrameworkCore;
|
2023-02-21 18:01:03 +05:00
|
|
|
|
using Microsoft.Extensions.Caching.Memory;
|
2022-04-11 18:00:34 +05:00
|
|
|
|
using System;
|
2021-09-29 10:12:54 +05:00
|
|
|
|
using System.Collections.Generic;
|
2021-09-27 16:47:16 +05:00
|
|
|
|
using System.Linq;
|
2021-11-16 13:14:31 +05:00
|
|
|
|
using System.Threading;
|
2022-04-11 18:00:34 +05:00
|
|
|
|
using System.Threading.Tasks;
|
2021-04-07 18:01:56 +05:00
|
|
|
|
|
2022-04-11 18:00:34 +05:00
|
|
|
|
namespace AsbCloudInfrastructure.Services.SAUB
|
2021-04-07 18:01:56 +05:00
|
|
|
|
{
|
|
|
|
|
public class TelemetryService : ITelemetryService
|
|
|
|
|
{
|
2021-09-27 16:47:16 +05:00
|
|
|
|
private readonly IAsbCloudDbContext db;
|
2023-02-21 18:01:03 +05:00
|
|
|
|
private readonly IMemoryCache memoryCache;
|
2023-05-19 17:57:07 +05:00
|
|
|
|
private readonly TelemetryDataCache<TelemetryDataSaubDto> dataSaubCache;
|
2022-01-05 17:50:45 +05:00
|
|
|
|
private readonly ITimezoneService timezoneService;
|
2021-04-07 18:01:56 +05:00
|
|
|
|
|
2022-01-05 17:50:45 +05:00
|
|
|
|
public ITimezoneService TimeZoneService => timezoneService;
|
2021-11-22 11:30:08 +05:00
|
|
|
|
|
|
|
|
|
public TelemetryService(
|
2022-04-11 18:00:34 +05:00
|
|
|
|
IAsbCloudDbContext db,
|
2023-02-21 18:01:03 +05:00
|
|
|
|
IMemoryCache memoryCache,
|
2023-05-19 17:57:07 +05:00
|
|
|
|
TelemetryDataCache<TelemetryDataSaubDto> dataSaubCache,
|
2022-10-31 15:21:53 +05:00
|
|
|
|
ITimezoneService timezoneService)
|
2021-04-07 18:01:56 +05:00
|
|
|
|
{
|
2021-09-27 16:47:16 +05:00
|
|
|
|
this.db = db;
|
2023-02-21 18:01:03 +05:00
|
|
|
|
this.memoryCache = memoryCache;
|
2023-05-19 17:57:07 +05:00
|
|
|
|
this.dataSaubCache = dataSaubCache;
|
2022-01-05 17:50:45 +05:00
|
|
|
|
this.timezoneService = timezoneService;
|
2021-04-07 18:01:56 +05:00
|
|
|
|
}
|
2022-04-11 18:00:34 +05:00
|
|
|
|
|
2022-11-16 12:49:11 +05:00
|
|
|
|
private IEnumerable<Telemetry> GetTelemetryCache()
|
2023-02-22 09:40:02 +05:00
|
|
|
|
=> memoryCache.GetOrCreateBasic(
|
|
|
|
|
db.Set<Telemetry>()
|
|
|
|
|
.Include(t => t.Well));
|
|
|
|
|
|
2022-10-31 15:21:53 +05:00
|
|
|
|
private void DropTelemetryCache()
|
|
|
|
|
{
|
2023-02-21 18:01:03 +05:00
|
|
|
|
memoryCache.DropBasic<Telemetry>();
|
2022-10-31 15:21:53 +05:00
|
|
|
|
}
|
2021-04-07 18:01:56 +05:00
|
|
|
|
|
2022-01-05 17:50:45 +05:00
|
|
|
|
public DatesRangeDto GetDatesRange(int idTelemetry)
|
2021-12-27 11:48:23 +05:00
|
|
|
|
{
|
2023-05-19 17:57:07 +05:00
|
|
|
|
var cacheDataRange = dataSaubCache.GetOrDefaultDataDateRange(idTelemetry)
|
|
|
|
|
?? new ();
|
|
|
|
|
return cacheDataRange;
|
2021-12-27 11:48:23 +05:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-07 15:55:11 +05:00
|
|
|
|
public int? GetIdWellByTelemetryUid(string uid)
|
2021-04-23 10:21:25 +05:00
|
|
|
|
=> GetWellByTelemetryUid(uid)?.Id;
|
2021-04-30 17:35:35 +05:00
|
|
|
|
|
2021-11-16 13:14:31 +05:00
|
|
|
|
public async Task UpdateInfoAsync(string uid, TelemetryInfoDto info,
|
|
|
|
|
CancellationToken token)
|
2021-04-09 17:59:07 +05:00
|
|
|
|
{
|
2023-06-20 14:41:19 +05:00
|
|
|
|
var telemetry = GetOrCreateTelemetry(uid);
|
2021-07-28 09:46:58 +05:00
|
|
|
|
telemetry.Info = info.Adapt<TelemetryInfo>();
|
2021-11-16 13:14:31 +05:00
|
|
|
|
|
2021-11-22 11:30:08 +05:00
|
|
|
|
if (!string.IsNullOrEmpty(info.TimeZoneId) &&
|
2021-12-30 17:05:44 +05:00
|
|
|
|
telemetry.TimeZone?.IsOverride != true)
|
2022-01-05 17:50:45 +05:00
|
|
|
|
telemetry.TimeZone = new SimpleTimezone()
|
2021-11-16 13:14:31 +05:00
|
|
|
|
{
|
2021-11-16 17:14:36 +05:00
|
|
|
|
Hours = info.TimeZoneOffsetTotalHours,
|
2022-06-06 17:00:53 +05:00
|
|
|
|
TimezoneId = info.TimeZoneId
|
2021-11-16 13:14:31 +05:00
|
|
|
|
};
|
2022-04-11 18:00:34 +05:00
|
|
|
|
|
2022-10-31 15:21:53 +05:00
|
|
|
|
db.Telemetries.Upsert(telemetry);
|
|
|
|
|
await db.SaveChangesAsync(token);
|
|
|
|
|
DropTelemetryCache();
|
2021-11-16 17:14:36 +05:00
|
|
|
|
}
|
2021-11-17 15:54:01 +05:00
|
|
|
|
|
2023-06-20 14:41:19 +05:00
|
|
|
|
[Obsolete("This method will be private. Use TelemetryDto.TimeZone prop.")]
|
2022-01-05 17:50:45 +05:00
|
|
|
|
public SimpleTimezoneDto GetTimezone(int idTelemetry)
|
2021-11-18 14:25:11 +05:00
|
|
|
|
{
|
2022-11-16 12:49:11 +05:00
|
|
|
|
var telemetry = GetTelemetryCache().FirstOrDefault(t => t.Id == idTelemetry);
|
2022-01-05 17:50:45 +05:00
|
|
|
|
|
|
|
|
|
if (telemetry is null)
|
|
|
|
|
throw new Exception($"Telemetry id: {idTelemetry} does not exist.");
|
2021-11-18 14:25:11 +05:00
|
|
|
|
|
2023-06-20 14:41:19 +05:00
|
|
|
|
if (telemetry.Well?.Timezone is not null)
|
2023-06-20 15:41:36 +05:00
|
|
|
|
return telemetry.Well.Timezone.Adapt<SimpleTimezoneDto>();
|
2023-06-20 14:41:19 +05:00
|
|
|
|
|
|
|
|
|
if (telemetry.TimeZone is not null)
|
2022-01-05 17:50:45 +05:00
|
|
|
|
return telemetry.TimeZone.Adapt<SimpleTimezoneDto>();
|
2023-06-20 14:41:19 +05:00
|
|
|
|
|
2022-01-05 17:50:45 +05:00
|
|
|
|
throw new Exception($"Telemetry id: {idTelemetry} can't find timezone.");
|
2021-11-18 12:03:59 +05:00
|
|
|
|
}
|
|
|
|
|
|
2023-03-07 09:12:49 +05:00
|
|
|
|
public TelemetryBaseDto? GetOrDefaultTelemetryByIdWell(int idWell)
|
2021-11-16 13:14:31 +05:00
|
|
|
|
{
|
2023-06-20 14:41:19 +05:00
|
|
|
|
var entity = GetTelemetryCache()
|
2022-11-16 12:49:11 +05:00
|
|
|
|
.FirstOrDefault(t => t.Well?.Id == idWell);
|
2023-06-30 15:49:30 +05:00
|
|
|
|
|
|
|
|
|
if (entity?.Well?.Timezone is not null && entity.TimeZone.Hours != entity.Well.Timezone.Hours)
|
2023-06-20 14:41:19 +05:00
|
|
|
|
{
|
|
|
|
|
entity.TimeZone = entity.Well.Timezone;
|
|
|
|
|
//TODO: выдаем предупреждение!
|
|
|
|
|
}
|
|
|
|
|
return entity?.Adapt<TelemetryBaseDto>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public TelemetryDto GetOrCreateTelemetryByUid(string uid)
|
|
|
|
|
{
|
|
|
|
|
var entity = GetOrCreateTelemetry(uid);
|
|
|
|
|
if(entity.Well?.Timezone is not null && entity.TimeZone.Hours != entity.Well.Timezone.Hours)
|
|
|
|
|
{
|
|
|
|
|
entity.TimeZone = entity.Well.Timezone;
|
|
|
|
|
//TODO: выдаем предупреждение!
|
|
|
|
|
}
|
|
|
|
|
var dto = entity.Adapt<TelemetryDto>();
|
|
|
|
|
return dto;
|
2021-04-07 18:01:56 +05:00
|
|
|
|
}
|
|
|
|
|
|
2022-10-31 15:21:53 +05:00
|
|
|
|
private Well? GetWellByTelemetryUid(string uid)
|
2021-05-17 12:53:30 +05:00
|
|
|
|
{
|
2022-10-31 15:21:53 +05:00
|
|
|
|
var telemetry = GetOrDefaultTelemetryByUid(uid);
|
|
|
|
|
return telemetry?.Well;
|
2021-11-17 15:54:01 +05:00
|
|
|
|
}
|
|
|
|
|
|
2022-10-31 15:21:53 +05:00
|
|
|
|
private Telemetry? GetOrDefaultTelemetryByUid(string uid)
|
2021-04-07 18:01:56 +05:00
|
|
|
|
{
|
2022-11-16 12:49:11 +05:00
|
|
|
|
var telemetry = GetTelemetryCache().FirstOrDefault(t => t.RemoteUid == uid);
|
2022-10-31 15:21:53 +05:00
|
|
|
|
return telemetry;
|
2021-04-07 18:01:56 +05:00
|
|
|
|
}
|
2021-04-23 10:21:25 +05:00
|
|
|
|
|
2023-06-20 14:41:19 +05:00
|
|
|
|
private Telemetry GetOrCreateTelemetry(string uid)
|
2022-10-31 15:21:53 +05:00
|
|
|
|
{
|
|
|
|
|
var telemetry = GetOrDefaultTelemetryByUid(uid);
|
|
|
|
|
if (telemetry is null)
|
2022-01-10 18:12:31 +05:00
|
|
|
|
{
|
2022-10-31 15:21:53 +05:00
|
|
|
|
var newTelemetry = new Telemetry
|
2022-04-11 18:00:34 +05:00
|
|
|
|
{
|
2022-10-31 15:21:53 +05:00
|
|
|
|
RemoteUid = uid,
|
|
|
|
|
TimeZone = new SimpleTimezone
|
|
|
|
|
{
|
|
|
|
|
Hours = 5,
|
|
|
|
|
IsOverride = false,
|
|
|
|
|
TimezoneId = "default",
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
var entry = db.Telemetries.Add(newTelemetry);
|
|
|
|
|
db.SaveChanges();
|
|
|
|
|
DropTelemetryCache();
|
|
|
|
|
return entry.Entity;
|
|
|
|
|
}
|
|
|
|
|
return telemetry;
|
|
|
|
|
}
|
2021-09-29 10:12:54 +05:00
|
|
|
|
|
2021-12-17 12:48:58 +05:00
|
|
|
|
public async Task<int> MergeAsync(int from, int to, CancellationToken token)
|
2021-09-29 10:12:54 +05:00
|
|
|
|
{
|
2021-12-17 12:48:58 +05:00
|
|
|
|
if (from == to)
|
|
|
|
|
return -2;
|
|
|
|
|
|
|
|
|
|
var stopwath = new System.Diagnostics.Stopwatch();
|
|
|
|
|
stopwath.Start();
|
|
|
|
|
|
2022-04-11 18:00:34 +05:00
|
|
|
|
var transaction = await db.Database.BeginTransactionAsync(token).ConfigureAwait(false);
|
2021-09-29 10:12:54 +05:00
|
|
|
|
try
|
|
|
|
|
{
|
2021-12-17 12:48:58 +05:00
|
|
|
|
var affected = 0;
|
2022-04-11 18:00:34 +05:00
|
|
|
|
|
2021-12-17 12:48:58 +05:00
|
|
|
|
var wellFrom = await db.Wells.FirstOrDefaultAsync(w => w.IdTelemetry == from, token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
var wellTo = await db.Wells.FirstOrDefaultAsync(w => w.IdTelemetry == to, token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
if (wellTo is not null && wellFrom is not null)
|
|
|
|
|
return -2;
|
2021-09-29 10:12:54 +05:00
|
|
|
|
|
2021-12-17 12:48:58 +05:00
|
|
|
|
if (wellTo is null && wellFrom is not null)
|
2021-09-29 10:12:54 +05:00
|
|
|
|
{
|
2021-12-17 12:48:58 +05:00
|
|
|
|
wellFrom.IdTelemetry = to;
|
|
|
|
|
affected += await db.SaveChangesAsync(token);
|
2021-09-29 10:12:54 +05:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-17 12:48:58 +05:00
|
|
|
|
affected += await MergeEventsAndMessagesAndUsersAsync(from, to, token);
|
|
|
|
|
affected += await MergeDataAsync<TelemetryDataSaub>(from, to, token);
|
|
|
|
|
affected += await MergeDataAsync<TelemetryDataSpin>(from, to, token);
|
2022-04-11 18:00:34 +05:00
|
|
|
|
|
2021-12-17 12:48:58 +05:00
|
|
|
|
affected += await db.Database.ExecuteSqlRawAsync($"DELETE FROM t_telemetry_analysis WHERE id_telemetry = {from} OR id_telemetry = {to};", token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
affected += await db.Database.ExecuteSqlRawAsync($"DELETE FROM t_telemetry WHERE id = {from};", token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
await transaction.CommitAsync(token).ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
stopwath.Stop();
|
2021-12-30 17:05:44 +05:00
|
|
|
|
Console.WriteLine($"Successfully committed in {1d * stopwath.ElapsedMilliseconds / 1000d: #0.00} sec. Affected {affected} rows.");
|
2022-10-31 15:21:53 +05:00
|
|
|
|
DropTelemetryCache();
|
2021-12-17 12:48:58 +05:00
|
|
|
|
return affected;
|
2021-09-29 10:12:54 +05:00
|
|
|
|
}
|
2022-04-11 18:00:34 +05:00
|
|
|
|
catch (Exception ex)
|
2021-09-29 10:12:54 +05:00
|
|
|
|
{
|
2021-12-17 12:48:58 +05:00
|
|
|
|
System.Diagnostics.Trace.WriteLine($"Merge() Fail. Rollback. Reason is:{ex.Message}");
|
2023-02-21 18:01:03 +05:00
|
|
|
|
await transaction.RollbackAsync(CancellationToken.None);
|
2021-12-17 12:48:58 +05:00
|
|
|
|
return -1;
|
2021-09-29 10:12:54 +05:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-11-22 11:30:08 +05:00
|
|
|
|
|
2021-12-17 12:48:58 +05:00
|
|
|
|
private async Task<int> MergeEventsAndMessagesAndUsersAsync(int from, int to, CancellationToken token)
|
|
|
|
|
{
|
|
|
|
|
var messagesFrom = await db.TelemetryMessages
|
|
|
|
|
.Where(d => d.IdTelemetry == from)
|
|
|
|
|
.ToListAsync(token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
var usersFromQuery = db.TelemetryUsers
|
|
|
|
|
.Where(d => d.IdTelemetry == from);
|
|
|
|
|
var usersFrom = await usersFromQuery
|
|
|
|
|
.ToListAsync(token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
var usersTo = await db.TelemetryUsers
|
|
|
|
|
.Where(d => d.IdTelemetry == to)
|
|
|
|
|
.AsNoTracking()
|
|
|
|
|
.ToListAsync(token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
var usersToNextId = usersTo.Max(u => u.IdUser) + 100;
|
|
|
|
|
|
|
|
|
|
messagesFrom.ForEach(m => m.IdTelemetry = to);
|
|
|
|
|
|
|
|
|
|
foreach (var userFrom in usersFrom)
|
|
|
|
|
{
|
|
|
|
|
var userTo = usersTo
|
2022-04-11 18:00:34 +05:00
|
|
|
|
.FirstOrDefault(u => u.IdUser == userFrom.IdUser);
|
2021-12-17 12:48:58 +05:00
|
|
|
|
|
|
|
|
|
if (userTo is null ||
|
|
|
|
|
userTo.Name != userFrom.Name ||
|
|
|
|
|
userTo.Surname != userFrom.Surname ||
|
|
|
|
|
userTo.Patronymic != userFrom.Patronymic)
|
|
|
|
|
{
|
|
|
|
|
messagesFrom
|
|
|
|
|
.Where(m => m.IdTelemetryUser == userFrom.IdUser)
|
|
|
|
|
.ToList()
|
|
|
|
|
.ForEach(m => m.IdTelemetryUser = usersToNextId);
|
|
|
|
|
userFrom.IdUser = usersToNextId;
|
|
|
|
|
userFrom.IdTelemetry = to;
|
|
|
|
|
usersToNextId++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var eventsFromQuery = db.TelemetryEvents
|
|
|
|
|
.Where(d => d.IdTelemetry == from);
|
|
|
|
|
var eventsFrom = await eventsFromQuery
|
|
|
|
|
.ToListAsync(token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
var eventsTo = await db.TelemetryEvents
|
|
|
|
|
.Where(d => d.IdTelemetry == to)
|
|
|
|
|
.AsNoTracking()
|
|
|
|
|
.ToListAsync(token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
var eventsToNextId = eventsTo.Max(e => e.IdEvent) + 1;
|
|
|
|
|
|
|
|
|
|
foreach (var eventFrom in eventsFrom)
|
|
|
|
|
{
|
|
|
|
|
var eventTo = eventsTo
|
|
|
|
|
.FirstOrDefault(e => e.IdEvent == eventFrom.IdEvent);
|
|
|
|
|
|
|
|
|
|
if (eventTo is null ||
|
|
|
|
|
eventTo.IdCategory != eventFrom.IdCategory ||
|
|
|
|
|
eventTo.MessageTemplate != eventFrom.MessageTemplate)
|
|
|
|
|
{
|
|
|
|
|
messagesFrom
|
|
|
|
|
.Where(m => m.IdEvent == eventFrom.IdEvent)
|
|
|
|
|
.ToList()
|
|
|
|
|
.ForEach(m => m.IdEvent = eventsToNextId);
|
|
|
|
|
eventFrom.IdEvent = eventsToNextId;
|
|
|
|
|
eventFrom.IdTelemetry = to;
|
|
|
|
|
eventsToNextId++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
await db.Database.ExecuteSqlRawAsync($"ALTER TABLE t_telemetry_user DISABLE TRIGGER ALL;", token).ConfigureAwait(false);
|
|
|
|
|
await db.Database.ExecuteSqlRawAsync($"ALTER TABLE t_telemetry_event DISABLE TRIGGER ALL;", token).ConfigureAwait(false);
|
|
|
|
|
await db.Database.ExecuteSqlRawAsync($"ALTER TABLE t_telemetry_message DISABLE TRIGGER ALL;", token).ConfigureAwait(false);
|
|
|
|
|
var affected = await db.SaveChangesAsync(token).ConfigureAwait(false);
|
|
|
|
|
await db.Database.ExecuteSqlRawAsync($"ALTER TABLE t_telemetry_user ENABLE TRIGGER ALL;", token).ConfigureAwait(false);
|
|
|
|
|
await db.Database.ExecuteSqlRawAsync($"ALTER TABLE t_telemetry_event ENABLE TRIGGER ALL;", token).ConfigureAwait(false);
|
|
|
|
|
await db.Database.ExecuteSqlRawAsync($"ALTER TABLE t_telemetry_message ENABLE TRIGGER ALL;", token).ConfigureAwait(false);
|
|
|
|
|
db.TelemetryUsers.RemoveRange(usersFromQuery);
|
|
|
|
|
db.TelemetryEvents.RemoveRange(eventsFromQuery);
|
|
|
|
|
affected += await db.SaveChangesAsync(token).ConfigureAwait(false);
|
|
|
|
|
return affected;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task<int> MergeDataAsync<TEntity>(int from, int to, CancellationToken token)
|
2022-04-11 18:00:34 +05:00
|
|
|
|
where TEntity : class, AsbCloudDb.Model.ITelemetryData
|
2021-12-17 12:48:58 +05:00
|
|
|
|
{
|
|
|
|
|
const string IdTelemetryColumnName = "\"id_telemetry\"";
|
|
|
|
|
var dbSet = db.Set<TEntity>();
|
|
|
|
|
var tableName = dbSet.GetTableName();
|
|
|
|
|
var columns = dbSet.GetColumnsNames().ToList();
|
|
|
|
|
var index = columns.FindIndex(c => c.ToLower() == IdTelemetryColumnName.ToLower());
|
|
|
|
|
if (index < 0)
|
|
|
|
|
return -5;
|
|
|
|
|
columns[index] = $"{to} as {IdTelemetryColumnName}";
|
|
|
|
|
var columnsString = string.Join(',', columns);
|
|
|
|
|
var sql = $"INSERT INTO {tableName} " +
|
|
|
|
|
$"select {columnsString} " +
|
|
|
|
|
$"from {tableName} " +
|
|
|
|
|
$"where {IdTelemetryColumnName} = {from};" +
|
|
|
|
|
$"delete from {tableName} where {IdTelemetryColumnName} = {from};";
|
|
|
|
|
|
|
|
|
|
var affected = await db.Database.ExecuteSqlRawAsync(sql, token)
|
|
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
return affected;
|
|
|
|
|
}
|
2021-04-07 18:01:56 +05:00
|
|
|
|
}
|
|
|
|
|
}
|