using Microsoft.EntityFrameworkCore; using System; using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; #nullable disable namespace AsbCloudDb.Model { //Scaffold-DbContext "Host=localhost;Database=postgres;Username=postgres;Password=q;Persist Security Info=True" Npgsql.EntityFrameworkCore.PostgreSQL -OutputDir Model -DataAnnotations public partial class AsbCloudDbContext : DbContext, IAsbCloudDbContext { public virtual DbSet Clusters { get; set; } public virtual DbSet Companies { get; set; } public virtual DbSet Deposits { get; set; } public virtual DbSet FileCategories { get; set; } public virtual DbSet Files { get; set; } public virtual DbSet FileMarks { get; set; } public virtual DbSet Measures { get; set; } public virtual DbSet MeasureCategories { get; set; } public virtual DbSet ReportProperties { get; set; } public virtual DbSet SetpointsRequests { get; set; } public virtual DbSet Telemetries { get; set; } public virtual DbSet TelemetryDataSaub { get; set; } public virtual DbSet TelemetryDataSpin { get; set; } public virtual DbSet TelemetryAnalysis { get; set; } public virtual DbSet TelemetryEvents { get; set; } public virtual DbSet TelemetryMessages { get; set; } public virtual DbSet TelemetryUsers { get; set; } public virtual DbSet Users { get; set; } public virtual DbSet UserRoles { get; set; } public virtual DbSet Wells { get; set; } public virtual DbSet WellComposites { get; set; } public virtual DbSet WellOperations { get; set; } public virtual DbSet WellOperationCategories { get; set; } public virtual DbSet WellSectionTypes { get; set; } public virtual DbSet WellTypes { get; set; } public virtual DbSet DrillParams { get; set; } public virtual DbSet DrillFlowChart { get; set; } public virtual DbSet RelationUserUserRoles { get; set; } public virtual DbSet Permissions { get; set; } public virtual DbSet RelationUserRolePermissions { get; set; } //var options = new DbContextOptionsBuilder() // .UseNpgsql("Host=localhost;Database=postgres;Username=postgres;Password=q;Persist Security Info=True") // .Options; //var context = new AsbCloudDbContext(options); public AsbCloudDbContext() : base() { Database.Migrate(); } public AsbCloudDbContext(DbContextOptions options) : base(options) { //Database.SetCommandTimeout(60 * 60 * 2); Database.Migrate(); } protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder) { if (!optionsBuilder.IsConfigured) optionsBuilder.UseNpgsql("Host=localhost;Database=postgres;Username=postgres;Password=q;Persist Security Info=True"); } protected override void OnModelCreating(ModelBuilder modelBuilder) { modelBuilder.HasPostgresExtension("adminpack") .HasAnnotation("Relational:Collation", "Russian_Russia.1251"); modelBuilder.Entity(entity => { entity.HasOne(d => d.Deposit) .WithMany(p => p.Clusters) .HasForeignKey(d => d.IdDeposit) .HasConstraintName("t_cluster_t_deposit_id_fk"); }); modelBuilder.Entity(entity => { entity.HasKey( nameof(WellComposite.IdWell), nameof(WellComposite.IdWellSrc), nameof(WellComposite.IdWellSectionType)); entity.HasOne(d => d.Well) .WithMany(p => p.WellComposites) .HasForeignKey(d => d.IdWell) .HasConstraintName("t_well_сomposite_t_well_id_fk"); entity.HasOne(d => d.WellSrc) .WithMany(p => p.WellCompositeSrcs) .HasForeignKey(d => d.IdWellSrc) .HasConstraintName("t_well_сomposite_src_t_well_id_fk"); entity.HasOne(d => d.WellSectionType) .WithMany(p => p.WellComposites) .HasForeignKey(d => d.IdWellSectionType) .HasConstraintName("t_well_сomposite_t_well_section_type_id_fk"); }); modelBuilder.Entity(entity => { entity.HasOne(d => d.Telemetry) .WithMany(p => p.DataSaub) .HasForeignKey(d => d.IdTelemetry) .OnDelete(DeleteBehavior.Cascade) .HasConstraintName("t_telemetry_data_saub_t_telemetry_id_fk"); entity.HasKey(nameof(ITelemetryData.IdTelemetry), nameof(ITelemetryData.Date)); }); modelBuilder.Entity(entity => { entity.HasOne(d => d.Telemetry) .WithMany(p => p.DataSpin) .HasForeignKey(d => d.IdTelemetry) .OnDelete(DeleteBehavior.Cascade) .HasConstraintName("t_telemetry_data_spin_t_telemetry_id_fk"); entity.HasKey(nameof(ITelemetryData.IdTelemetry), nameof(ITelemetryData.Date)); }); modelBuilder.Entity(entity => { entity.HasOne(d => d.Telemetry) .WithMany(p => p.Messages) .HasForeignKey(d => d.IdTelemetry) .OnDelete(DeleteBehavior.Cascade) .HasConstraintName("t_messages_t_telemetry_id_fk"); }); modelBuilder.Entity(entity => { entity.HasKey(nameof(TelemetryUser.IdTelemetry), nameof(TelemetryUser.IdUser)); entity.HasOne(d => d.Telemetry) .WithMany(p => p.Users) .HasForeignKey(d => d.IdTelemetry) .OnDelete(DeleteBehavior.Cascade) .HasConstraintName("t_telemetry_user_t_telemetry_id_fk"); }); modelBuilder.Entity(entity => { entity.HasOne(d => d.Telemetry) .WithMany(p => p.Analysis) .HasForeignKey(d => d.IdTelemetry) .OnDelete(DeleteBehavior.Cascade) .HasConstraintName("t_analysis_t_telemetry_id_fk"); entity.HasOne(d => d.Operation) .WithMany(p => p.Analysis) .HasForeignKey(d => d.IdOperation) .OnDelete(DeleteBehavior.SetNull) .HasConstraintName("t_analysis_t_operation_id_fk"); }); modelBuilder.Entity(entity => { entity.HasKey(nameof(TelemetryEvent.IdTelemetry), nameof(TelemetryEvent.IdEvent)); entity.HasOne(d => d.Telemetry) .WithMany(p => p.Events) .HasForeignKey(d => d.IdTelemetry) .OnDelete(DeleteBehavior.Cascade) .HasConstraintName("t_event_t_telemetry_id_fk"); }); modelBuilder.Entity(entity => { entity.HasOne(d => d.Company) .WithMany(p => p.Users) .HasForeignKey(d => d.IdCompany) .OnDelete(DeleteBehavior.SetNull) .HasConstraintName("t_user_t_company_id_fk"); entity.HasIndex(d => d.Login) .IsUnique(); }); modelBuilder.Entity(entity => { entity.HasOne(d => d.Cluster) .WithMany(p => p.Wells) .HasForeignKey(d => d.IdCluster) .HasConstraintName("t_well_t_cluster_id_fk"); entity.HasOne(d => d.Telemetry) .WithOne(p => p.Well) .HasForeignKey(d => d.IdTelemetry) .OnDelete(DeleteBehavior.SetNull) .HasConstraintName("t_well_t_telemetry_id_fk"); }); modelBuilder.Entity(entity => { entity.HasKey(nameof(RelationCompanyWell.IdCompany), nameof(RelationCompanyWell.IdWell)); entity.HasOne(r => r.Well) .WithMany(w => w.RelationCompaniesWells) .HasForeignKey(r => r.IdWell) .HasConstraintName("t_relation_company_well_t_well_id_fk"); entity.HasOne(r => r.Company) .WithMany(w => w.RelationCompaniesWells) .HasForeignKey(r => r.IdCompany) .HasConstraintName("t_relation_company_well_t_company_id_fk"); }); modelBuilder.Entity(entity => { entity.HasIndex(d => d.DepthEnd); entity.HasIndex(d => d.DateStart); }); modelBuilder.Entity(entity => { entity.HasOne(r => r.WellSectionType) .WithMany(w => w.DrillParamsCollection) .HasForeignKey(r => r.IdWellSectionType) .HasConstraintName("t_drill_params_t_well_section_type_id_fk"); }); modelBuilder.Entity(entity => { entity.HasIndex(d => d.IdWellOperationCategory); }); modelBuilder.Entity(entity => { entity.HasOne(d => d.User) .WithMany(p => p.FileMarks) .HasForeignKey(d => d.IdUser) .OnDelete(DeleteBehavior.Cascade) .HasConstraintName("t_user_t_file_mark_fk"); entity.HasOne(d => d.FileInfo) .WithMany(p => p.FileMarks) .HasForeignKey(d => d.IdFile) .OnDelete(DeleteBehavior.Cascade) .HasConstraintName("t_file_mark_t_file_info_fk"); }); modelBuilder.Entity(entity => { entity.HasKey(e => new { e.IdUser, e.IdUserRole }); }); modelBuilder.Entity(entity => { entity.HasKey(e => new { e.IdUserRole, e.IdPermission }); entity.HasOne(r => r.Permission) .WithMany(p => p.RelationUserRolePermissions) .IsRequired(); entity.HasOne(r => r.UserRole) .WithMany(r => r.RelationUserRolePermissions) .IsRequired(); }); FillData(modelBuilder); } private static void FillData(ModelBuilder modelBuilder) { modelBuilder.Entity(entity => { entity.HasData(new List{ new UserRole{ Id = 1, Caption = "Администратор", }, new UserRole{ Id = 2, Caption = "Пользователь", }, }); }); modelBuilder.Entity(entity => { entity.HasData(new List{ new User{ Id = 1, IdCompany = 1, Login = "dev", PasswordHash = "Vlcj|4fa529103dde7ff72cfe76185f344d4aa87931f8e1b2044e8a7739947c3d18923464eaad93843e4f809c5e126d013072", // dev Name = "Разработчик", }, }); }); modelBuilder.Entity(entity => { entity.HasData(new List{ new RelationUserUserRole { IdUser = 1, IdUserRole = 2, }, }); }); modelBuilder.Entity(entity => { entity.HasData(new List{ new Company{ Id = 1, Caption = "ООО \"АСБ\"", IdCompanyType = 3}, }); }); modelBuilder.Entity(entity => { entity.HasData(new List { // Автоматически опеределяемые операции new WellOperationCategory {Id = 1, Name = "Невозможно определить операцию", Code = 0}, new WellOperationCategory {Id = 2, Name = "Роторное бурение", Code = 0 }, new WellOperationCategory {Id = 3, Name = "Слайдирование", Code = 0 }, new WellOperationCategory {Id = 4, Name = "Подъем с проработкой", Code = 0 }, new WellOperationCategory {Id = 5, Name = "Спуск с проработкой", Code = 0 }, new WellOperationCategory {Id = 6, Name = "Подъем с промывкой", Code = 0 }, new WellOperationCategory {Id = 7, Name = "Спуск с промывкой", Code = 0 }, new WellOperationCategory {Id = 8, Name = "Спуск в скважину", Code = 0 }, new WellOperationCategory {Id = 9, Name = "Спуск с вращением", Code = 0 }, new WellOperationCategory {Id = 10, Name = "Подъем из скважины", Code = 0 }, new WellOperationCategory {Id = 11, Name = "Подъем с вращением", Code = 0 }, new WellOperationCategory {Id = 12, Name = "Промывка в покое", Code = 0 }, new WellOperationCategory {Id = 13, Name = "Промывка с вращением", Code = 0 }, new WellOperationCategory {Id = 14, Name = "Удержание в клиньях", Code = 0 }, new WellOperationCategory {Id = 15, Name = "Неподвижное состояние", Code = 0 }, new WellOperationCategory {Id = 16, Name = "Вращение без циркуляции", Code = 0 }, new WellOperationCategory {Id = 17, Name = "На поверхности", Code = 0 }, // Операции ручного ввода new WellOperationCategory {Id = 1001, Name = "Бурение", Code = 0 }, new WellOperationCategory {Id = 1002, Name = "ГИС", Code = 0 }, new WellOperationCategory {Id = 1003, Name = "ГФР", Code = 0 }, new WellOperationCategory {Id = 1004, Name = "Монтаж ПВО", Code = 0 }, new WellOperationCategory {Id = 1005, Name = "Демонтаж ПВО", Code = 0 }, new WellOperationCategory {Id = 1006, Name = "Установка ФА", Code = 0 }, new WellOperationCategory {Id = 1007, Name = "Оборудование устья", Code = 0 }, new WellOperationCategory {Id = 1008, Name = "ОЗЦ", Code = 0 }, new WellOperationCategory {Id = 1011, Name = "Начало цикла строительства скважины", Code = 0 }, new WellOperationCategory {Id = 1012, Name = "Окончание цикла строительства скважины", Code = 0 }, new WellOperationCategory {Id = 1013, Name = "Опрессовка ПВО", Code = 0 }, new WellOperationCategory {Id = 1014, Name = "Опресовка Ц.К.", Code = 0 }, new WellOperationCategory {Id = 1015, Name = "Опрессовка ВЗД", Code = 0 }, new WellOperationCategory {Id = 1016, Name = "Перевод скв на другой тип промывочной жидкости", Code = 0 }, new WellOperationCategory {Id = 1017, Name = "Перезапись каротажа", Code = 0 }, new WellOperationCategory {Id = 1018, Name = "Перетяжка талевого каната", Code = 0 }, new WellOperationCategory {Id = 1019, Name = "Наращивание, промывка", Code = 0 }, new WellOperationCategory {Id = 1020, Name = "Подъем инструмента", Code = 0 }, new WellOperationCategory {Id = 1021, Name = "Подъем инструмента с промывкой", Code = 0 }, new WellOperationCategory {Id = 1022, Name = "Обратная проработка", Code = 0 }, new WellOperationCategory {Id = 1023, Name = "Сборка инструмента с мостков", Code = 0 }, new WellOperationCategory {Id = 1024, Name = "Подготовительные работы", Code = 0 }, new WellOperationCategory {Id = 1025, Name = "Сборка КНБК", Code = 0 }, new WellOperationCategory {Id = 1026, Name = "Разборка КНБК", Code = 0 }, new WellOperationCategory {Id = 1027, Name = "Промывка", Code = 0 }, new WellOperationCategory {Id = 1028, Name = "Промежуточная промывка", Code = 0 }, new WellOperationCategory {Id = 1029, Name = "Прокачка пачек", Code = 0 }, new WellOperationCategory {Id = 1030, Name = "Разбуривание тех.оснастки", Code = 0 }, new WellOperationCategory {Id = 1031, Name = "Ремонт", Code = 0 }, new WellOperationCategory {Id = 1032, Name = "Спуск инструмента", Code = 0 }, new WellOperationCategory {Id = 1033, Name = "Спуск инструмента с промывкой", Code = 0 }, new WellOperationCategory {Id = 1034, Name = "Прямая проработка", Code = 0 }, new WellOperationCategory {Id = 1035, Name = "Принудительная проработка", Code = 0 }, new WellOperationCategory {Id = 1037, Name = "Тех СПО-подъем", Code = 0 }, new WellOperationCategory {Id = 1038, Name = "Тех СПО-спуск", Code = 0 }, new WellOperationCategory {Id = 1039, Name = "Техническое обслуживание", Code = 0 }, new WellOperationCategory {Id = 1040, Name = "Цементаж", Code = 0 }, new WellOperationCategory {Id = 1041, Name = "Шаблонировка ствола", Code = 0 }, new WellOperationCategory {Id = 1042, Name = "Геологическое осложнение", Code = 0 }, new WellOperationCategory {Id = 1043, Name = "НПВ", Code = 0 }, new WellOperationCategory {Id = 1044, Name = "ВМР", Code = 0 }, new WellOperationCategory {Id = 1045, Name = "Прочее", Code = 0 }, new WellOperationCategory {Id = 1046, Name = "Спуск КНБК", Code = 0 }, new WellOperationCategory {Id = 1047, Name = "Подъем КНБК", Code = 0 }, new WellOperationCategory {Id = 1048, Name = "Спуск ОК", Code = 0 }, new WellOperationCategory {Id = 1050, Name = "Промывка при спуске ОК", Code = 0 }, new WellOperationCategory {Id = 1051, Name = "Замер ТС", Code = 0 }, new WellOperationCategory {Id = 1052, Name = "Тех. отстой", Code = 0 }, new WellOperationCategory {Id = 1053, Name = "Циркуляция и Обработка БР", Code = 0 }, new WellOperationCategory {Id = 1054, Name = "Срезка ствола", Code = 0 }, new WellOperationCategory {Id = 1055, Name = "Вспомогательные работы", Code = 0 }, }); }); modelBuilder.Entity(entity => { entity.HasData(new List { new FileCategory {Id = 1, Name = "Растворный сервис", ShortName = "fluidService"}, new FileCategory {Id = 2, Name = "Цементирование", ShortName = "cement"}, new FileCategory {Id = 3, Name = "ННБ", ShortName = "nnb"}, new FileCategory {Id = 4, Name = "ГТИ", ShortName = "gti"}, new FileCategory {Id = 5, Name = "Документы по скважине", ShortName = "wellDocuments"}, new FileCategory {Id = 6, Name = "Супервайзер", ShortName = "supervisor"}, new FileCategory {Id = 7, Name = "Мастер", ShortName = "master"}, new FileCategory {Id = 8, Name = "Долотный сервис", ShortName = "toolService"}, new FileCategory {Id = 9, Name = "Буровой подрядчик", ShortName = "drillService"}, new FileCategory {Id = 10, Name = "Сервис по заканчиванию скважины", ShortName = "closingService"}, //new FileCategory {Id = 11, Name = "", ShortName = ""}, new FileCategory {Id = 12, Name = "Рапорт", ShortName = "report"}, new FileCategory {Id = 13, Name = "Программа бурения, части", ShortName = "ПБч"}, new FileCategory {Id = 14, Name = "Программа бурения", ShortName = "ПБ"}, }); }); modelBuilder.Entity(entity => { entity.HasData(new List{ new WellSectionType{ Id = 1, Caption = "Пилотный ствол 1"}, new WellSectionType{ Id = 2, Caption = "Направление 1"}, new WellSectionType{ Id = 3, Caption = "Кондуктор 1"}, new WellSectionType{ Id = 4, Caption = "Эксплуатационная колонна 1"}, new WellSectionType{ Id = 5, Caption = "Транспортный ствол 1"}, new WellSectionType{ Id = 6, Caption = "Хвостовик 1"}, new WellSectionType{ Id = 7, Caption = "Пилотный ствол 2"}, new WellSectionType{ Id = 8, Caption = "Направление 2"}, new WellSectionType{ Id = 9, Caption = "Кондуктор 2"}, new WellSectionType{ Id = 10, Caption = "Эксплуатационная колонна 2"}, new WellSectionType{ Id = 11, Caption = "Транспортный ствол 2"}, new WellSectionType{ Id = 12, Caption = "Хвостовик 2"}, new WellSectionType{ Id = 13, Caption = "Пилотный ствол 3"}, new WellSectionType{ Id = 14, Caption = "Направление 3"}, new WellSectionType{ Id = 15, Caption = "Кондуктор 3"}, new WellSectionType{ Id = 16, Caption = "Эксплуатационная колонна 3"}, new WellSectionType{ Id = 17, Caption = "Транспортный ствол 3"}, new WellSectionType{ Id = 18, Caption = "Хвостовик 3"}, new WellSectionType{ Id = 19, Caption = "Пилотный ствол 4"}, new WellSectionType{ Id = 20, Caption = "Направление 4"}, new WellSectionType{ Id = 21, Caption = "Кондуктор 4"}, new WellSectionType{ Id = 22, Caption = "Эксплуатационная колонна 4"}, new WellSectionType{ Id = 23, Caption = "Транспортный ствол 4"}, new WellSectionType{ Id = 24, Caption = "Хвостовик 4"}, new WellSectionType{ Id = 25, Caption = "Пилотный ствол 5"}, new WellSectionType{ Id = 26, Caption = "Направление 5"}, new WellSectionType{ Id = 27, Caption = "Кондуктор 5"}, new WellSectionType{ Id = 28, Caption = "Эксплуатационная колонна 5"}, new WellSectionType{ Id = 29, Caption = "Транспортный ствол 5"}, new WellSectionType{ Id = 30, Caption = "Хвостовик 5"}, }); }); modelBuilder.Entity(entity => { entity.HasData(new List { new WellType{ Id = 1, Caption = "Наклонно-направленная"}, new WellType{ Id = 2, Caption = "Горизонтальная"}, }); }); modelBuilder.Entity(entity => { entity.HasData(new List { new MeasureCategory{ Id = 1, Name = "Показатели бурового раствора", ShortName = "Раствор"}, new MeasureCategory{ Id = 2, Name = "Шламограмма", ShortName = "Шламограмма"}, new MeasureCategory{ Id = 3, Name = "ННБ", ShortName = "ННБ"}, }); }); modelBuilder.Entity(entity => { entity.HasData(new List{ new CompanyType{ Id = 1, Caption = "Недрапользователь", }, new CompanyType{ Id = 2, Caption = "Буровой подрядчик", }, new CompanyType{ Id = 3, Caption = "Сервис автоматизации бурения", }, }); }); } public async Task<(DateTime From, DateTime To)> GetDatesRangeAsync(int idTelemetry, CancellationToken token = default) where TEntity : class, ITelemetryData { var dbSet = Set(); var datesRange = await (from m in dbSet where m.IdTelemetry == idTelemetry group m by m.IdTelemetry into g select new { From = g.Min(d => d.Date), To = g.Max(d => d.Date) }).AsNoTracking().FirstOrDefaultAsync(token); if (datesRange is null) return (DateTime.MinValue, DateTime.MaxValue); return (datesRange.From, datesRange.To); } public async Task> GetDepthToIntervalAsync(int telemetryId, int intervalHoursTimestamp, int workStartTimestamp, double timezoneOffset, CancellationToken token) { //TODO: Сменить на LINQ группирование using var command = Database.GetDbConnection().CreateCommand(); command.CommandText = $@"SELECT Min(t.bit_depth) AS MinDepth, Max(t.bit_depth) AS MaxDepth, Min(t.Date) AS dateStart FROM t_telemetry_data_saub AS t WHERE id_telemetry = {telemetryId} AND t.Id % 10 = 0 GROUP BY floor((extract(epoch from t.date) - {workStartTimestamp} + {timezoneOffset}) / {intervalHoursTimestamp});"; Database.OpenConnection(); using var reader = await command.ExecuteReaderAsync(token); var result = new List<(double? MinDepth, double? MaxDepth, DateTime BeginPeriodDate)>(); if (reader.HasRows) { while (reader.Read()) { result.Add( ( (double?)reader.GetValue(0), (double?)reader.GetValue(1), (DateTime)reader.GetValue(2) )); } } return result; } } }