using System; using System.Collections.Generic; using Remotely.Shared.Models; using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity.EntityFrameworkCore; using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore.Storage.ValueConversion; using System.Linq; using System.Text.Json; using Microsoft.EntityFrameworkCore.ChangeTracking; namespace Remotely.Server.Data { public class ApplicationDbContext : IdentityDbContext { public ApplicationDbContext(DbContextOptions context) : base(context) { } public DbSet Alerts { get; set; } public DbSet ApiTokens { get; set; } public DbSet CommandResults { get; set; } public DbSet Devices { get; set; } public DbSet Organizations { get; set; } public new DbSet Users { get; set; } public DbSet EventLogs { get; set; } public DbSet SharedFiles { get; set; } public DbSet InviteLinks { get; set; } public DbSet DeviceGroups { get; set; } public DbSet PermissionLinks { get; set; } protected override void OnModelCreating(ModelBuilder builder) { base.OnModelCreating(builder); builder.Entity().ToTable("RemotelyUsers"); builder.Entity() .HasMany(x => x.Devices) .WithOne(x=>x.Organization); builder.Entity() .HasMany(x => x.RemotelyUsers) .WithOne(x=> x.Organization); builder.Entity() .HasMany(x => x.CommandResults) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.EventLogs) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.Devices) .WithOne(x => x.DeviceGroup); builder.Entity() .HasMany(x => x.PermissionLinks); builder.Entity() .HasMany(x => x.DeviceGroups) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.InviteLinks) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.SharedFiles) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.ApiTokens) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.Alerts) .WithOne(x => x.Organization); builder.Entity() .Property(x=>x.TargetDeviceIDs) .HasConversion( x => JsonSerializer.Serialize(x, null), x => JsonSerializer.Deserialize(x, null)); builder.Entity() .Property(x => x.TargetDeviceIDs) .Metadata.SetValueComparer(new ValueComparer(true)); builder.Entity() .Property(x => x.PSCoreResults) .HasConversion( x => JsonSerializer.Serialize(x, null), x => JsonSerializer.Deserialize>(x, null)); builder.Entity() .Property(x => x.PSCoreResults) .Metadata.SetValueComparer(new ValueComparer>(true)); builder.Entity() .Property(x => x.CommandResults) .HasConversion( x => JsonSerializer.Serialize(x, null), x => JsonSerializer.Deserialize>(x, null)); builder.Entity() .Property(x => x.CommandResults) .Metadata.SetValueComparer(new ValueComparer>(true)); //builder.Entity() // .HasNoKey(); //builder.Entity() // .HasNoKey(); builder.Entity() .HasOne(x => x.Organization) .WithMany(x => x.RemotelyUsers); builder.Entity() .HasMany(x => x.PermissionLinks); builder.Entity() .HasMany(x => x.Alerts) .WithOne(x => x.User); builder.Entity() .Property(x => x.UserOptions) .HasConversion( x => JsonSerializer.Serialize(x, null), x => JsonSerializer.Deserialize(x, null)); builder.Entity() .HasIndex(x => x.UserName); builder.Entity() .Property(x => x.Drives) .HasConversion( x => JsonSerializer.Serialize(x, null), x => JsonSerializer.Deserialize>(x, null)); builder.Entity() .Property(x => x.Drives) .Metadata.SetValueComparer(new ValueComparer>(true)); builder.Entity() .HasIndex(x => x.DeviceName); builder.Entity() .HasMany(x => x.Alerts) .WithOne(x => x.Device); builder.Entity() .HasIndex(x => x.Token); builder.Entity() .HasOne(x => x.User) .WithMany(x => x.Alerts); if (Database.ProviderName == "Microsoft.EntityFrameworkCore.Sqlite") { // SQLite does not have proper support for DateTimeOffset via Entity Framework Core, see the limitations // here: https://docs.microsoft.com/en-us/ef/core/providers/sqlite/limitations#query-limitations // To work around this, when the SQLite database provider is used, all model properties of type DateTimeOffset // use the DateTimeOffsetToBinaryConverter // Based on: https://github.com/aspnet/EntityFrameworkCore/issues/10784#issuecomment-415769754 // This only supports millisecond precision, but should be sufficient for most use cases. foreach (var entityType in builder.Model.GetEntityTypes()) { if (entityType.IsKeyless) { continue; } var properties = entityType.ClrType.GetProperties().Where(p => p.PropertyType == typeof(DateTimeOffset) || p.PropertyType == typeof(DateTimeOffset?)); foreach (var property in properties) { builder .Entity(entityType.Name) .Property(property.Name) .HasConversion(new DateTimeOffsetToStringConverter()); } } } } } }