using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity.EntityFrameworkCore; using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore.ChangeTracking; using Microsoft.EntityFrameworkCore.Storage.ValueConversion; using Microsoft.Extensions.Logging; using Remotely.Shared.Models; using System; using System.Collections.Generic; using System.Linq; using System.Text.Json; namespace Remotely.Server.Data { public class AppDb : IdentityDbContext { private static ValueComparer _stringArrayComparer = new( (a, b) => a.SequenceEqual(b), c => c.Aggregate(0, (a, b) => HashCode.Combine(a, b.GetHashCode())), c => c.ToArray()); public AppDb(DbContextOptions context) : base(context) { } public DbSet Alerts { get; set; } public DbSet ApiTokens { get; set; } public DbSet DeviceGroups { get; set; } public DbSet Devices { get; set; } public DbSet EventLogs { get; set; } public DbSet InviteLinks { get; set; } public DbSet Organizations { get; set; } public DbSet ScriptRuns { get; set; } public DbSet SavedScripts { get; set; } public DbSet ScriptSchedules { get; set; } public DbSet ScriptResults { get; set; } public DbSet SharedFiles { get; set; } public new DbSet Users { 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.EventLogs) .WithOne(x => x.Organization); 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() .HasMany(x => x.ScriptRuns) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.ScriptSchedules) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.ScriptResults) .WithOne(x => x.Organization); builder.Entity() .HasMany(x => x.SavedScripts) .WithOne(x => x.Organization); builder.Entity() .HasOne(x => x.Organization) .WithMany(x => x.RemotelyUsers); builder.Entity() .HasMany(x => x.DeviceGroups) .WithMany(x => x.Users); 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() .HasMany(x => x.SavedScripts) .WithOne(x => x.Creator); builder.Entity() .HasMany(x => x.ScriptSchedules) .WithOne(x => x.Creator); 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() .HasMany(x => x.ScriptRuns) .WithMany(x => x.Devices); builder.Entity() .HasMany(x => x.ScriptRunsCompleted) .WithMany(x => x.DevicesCompleted); builder.Entity() .HasMany(x => x.ScriptSchedules) .WithMany(x => x.Devices); builder.Entity() .HasMany(x => x.Devices); builder.Entity() .HasMany(x => x.ScriptSchedules) .WithMany(x => x.DeviceGroups); builder.Entity() .HasMany(x => x.Devices) .WithMany(x => x.ScriptRuns); builder.Entity() .HasMany(x => x.DevicesCompleted) .WithMany(x => x.ScriptRunsCompleted); builder.Entity() .Property(x => x.ErrorOutput) .HasConversion( x => JsonSerializer.Serialize(x, null), x => JsonSerializer.Deserialize(x, null)) .Metadata .SetValueComparer(_stringArrayComparer); builder.Entity() .Property(x => x.StandardOutput) .HasConversion( x => JsonSerializer.Serialize(x, null), x => JsonSerializer.Deserialize(x, null)) .Metadata .SetValueComparer(_stringArrayComparer); 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()); } } } } } }