using Microsoft.Data.Sqlite; using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore.Storage.ValueConversion; using Observatory.Framework.Files.Journal; using Observatory.Framework.Files.Journal.Odyssey; using Observatory.Framework.Files.Journal.Startup; using Observatory.Framework.Files.Journal.StationServices; using Observatory.Framework.Files.Journal.Travel; using Pulsar.Features.ShipLocker; /// /// An in-memory database context for Pulsar. /// public class PulsarContext : DbContext { public SqliteConnection Connection { get; private set; } public DbSet Commander { get; set; } public DbSet Materials { get; set; } public DbSet Rank { get; set; } public DbSet Progress { get; set; } public DbSet Reputation { get; set; } public DbSet EngineerProgress { get; set; } public DbSet LoadGames { get; set; } public DbSet Statistics { get; set; } public DbSet Locations { get; set; } public DbSet PowerPlay { get; set; } public DbSet ShipLocker { get; set; } public DbSet Missions { get; set; } public DbSet Loadout { get; set; } public DbSet Cargo { get; set; } protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder) { Connection = new SqliteConnection("Data Source=Journals.sqlite"); optionsBuilder.UseSqlite(Connection); } protected override void OnModelCreating(ModelBuilder modelBuilder) { modelBuilder.ApplyConfigurationsFromAssembly(typeof(PulsarContext).Assembly); base.OnModelCreating(modelBuilder); if (Database.ProviderName != "Microsoft.EntityFrameworkCore.Sqlite") return; // 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 modelBuilder.Model.GetEntityTypes()) { var properties = entityType.ClrType.GetProperties().Where(p => p.PropertyType == typeof(DateTimeOffset) || p.PropertyType == typeof(DateTimeOffset?)); foreach (var property in properties) { modelBuilder .Entity(entityType.Name) .Property(property.Name) .HasConversion(new DateTimeOffsetToBinaryConverter()); } } } public override void Dispose() { Connection.Dispose(); base.Dispose(); } }