Files
aliasvault/src/Databases/AliasServerDb/AliasServerDbContext.cs
2024-08-27 23:05:14 +02:00

249 lines
8.3 KiB
C#

//-----------------------------------------------------------------------
// <copyright file="AliasServerDbContext.cs" company="lanedirt">
// Copyright (c) lanedirt. All rights reserved.
// Licensed under the MIT license. See LICENSE.md file in the project root for full license information.
// </copyright>
//-----------------------------------------------------------------------
namespace AliasServerDb;
using AliasVault.WorkerStatus.Database;
using Microsoft.AspNetCore.DataProtection.EntityFrameworkCore;
using Microsoft.AspNetCore.Identity;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Configuration;
/// <summary>
/// The AliasServerDbContext class. Note: we are using DbContext instead of IdentityDbContext because
/// we have two separate user objects, one for the admin panel and one for the vault. We manually
/// define the Identity tables in the OnModelCreating method.
/// </summary>
public class AliasServerDbContext : WorkerStatusDbContext, IDataProtectionKeyContext
{
/// <summary>
/// Initializes a new instance of the <see cref="AliasServerDbContext"/> class.
/// </summary>
public AliasServerDbContext()
{
}
/// <summary>
/// Initializes a new instance of the <see cref="AliasServerDbContext"/> class.
/// </summary>
/// <param name="options">DbContextOptions.</param>
public AliasServerDbContext(DbContextOptions<AliasServerDbContext> options)
: base(options)
{
}
/// <summary>
/// Gets or sets the DataProtectionKeys DbSet.
/// </summary>
public DbSet<DataProtectionKey> DataProtectionKeys { get; set; }
/// <summary>
/// Gets or sets the AliasVaultUser DbSet.
/// </summary>
public DbSet<AliasVaultUser> AliasVaultUsers { get; set; }
/// <summary>
/// Gets or sets the AliasVaultRoles DbSet.
/// </summary>
public DbSet<AliasVaultRole> AliasVaultRoles { get; set; }
/// <summary>
/// Gets or sets the UserRoles DbSet.
/// </summary>
public DbSet<IdentityUserRole<string>> UserRoles { get; set; }
/// <summary>
/// Gets or sets the UserClaims DbSet.
/// </summary>
public DbSet<IdentityUserClaim<string>> UserClaims { get; set; }
/// <summary>
/// Gets or sets the UserLogin DbSet.
/// </summary>
public DbSet<IdentityUserLogin<string>> UserLogin { get; set; }
/// <summary>
/// Gets or sets the RoleClaims DbSet.
/// </summary>
public DbSet<IdentityRoleClaim<string>> RoleClaims { get; set; }
/// <summary>
/// Gets or sets the UserTokens DbSet.
/// </summary>
public DbSet<IdentityUserToken<string>> UserTokens { get; set; }
/// <summary>
/// Gets or sets the UserRefreshTokens DbSet.
/// </summary>
public DbSet<AliasVaultUserRefreshToken> AliasVaultUserRefreshTokens { get; set; }
/// <summary>
/// Gets or sets the AdminUser DbSet.
/// </summary>
public DbSet<AdminUser> AdminUsers { get; set; }
/// <summary>
/// Gets or sets the AdminRoles DbSet.
/// </summary>
public DbSet<AdminRole> AdminRoles { get; set; }
/// <summary>
/// Gets or sets the Vaults DbSet.
/// </summary>
public DbSet<Vault> Vaults { get; set; }
/// <summary>
/// Gets or sets the Emails DbSet.
/// </summary>
public DbSet<Email> Emails { get; set; }
/// <summary>
/// Gets or sets the EmailAttachments DbSet.
/// </summary>
public DbSet<EmailAttachment> EmailAttachments { get; set; }
/// <summary>
/// Gets or sets the UserEmailClaims DbSet.
/// </summary>
public DbSet<UserEmailClaim> UserEmailClaims { get; set; }
/// <summary>
/// Gets or sets the UserEncryptionKeys DbSet.
/// </summary>
public DbSet<UserEncryptionKey> UserEncryptionKeys { get; set; }
/// <summary>
/// Gets or sets the Logs DbSet.
/// </summary>
public DbSet<Log> Logs { get; set; }
/// <summary>
/// Gets or sets the AuthLogs DbSet.
/// </summary>
public DbSet<AuthLog> AuthLogs { get; set; }
/// <summary>
/// The OnModelCreating method.
/// </summary>
/// <param name="modelBuilder">ModelBuilder instance.</param>
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
foreach (var entity in modelBuilder.Model.GetEntityTypes())
{
foreach (var property in entity.GetProperties())
{
// NOTE: This is a workaround for SQLite. Add conditional check if SQLite is used.
// NOTE: SQL server doesn't need this override.
// SQLite does not support varchar(max) so we use TEXT.
if (property.ClrType == typeof(string) && property.GetMaxLength() == null)
{
property.SetColumnType("TEXT");
}
}
}
// Configure AspNetIdentity tables manually.
modelBuilder.Entity<IdentityUserRole<string>>(entity =>
{
entity.HasKey(r => new { r.UserId, r.RoleId });
entity.ToTable("UserRoles");
});
modelBuilder.Entity<IdentityUserClaim<string>>(entity =>
{
entity.HasKey(c => c.Id);
entity.ToTable("UserClaims");
});
modelBuilder.Entity<IdentityUserLogin<string>>(entity =>
{
entity.HasKey(l => new { l.LoginProvider, l.ProviderKey });
entity.ToTable("UserLogins");
});
modelBuilder.Entity<IdentityRoleClaim<string>>(entity =>
{
entity.HasKey(rc => rc.Id);
entity.ToTable("RoleClaims");
});
modelBuilder.Entity<IdentityUserToken<string>>(entity =>
{
entity.HasKey(t => new { t.UserId, t.LoginProvider, t.Name });
entity.ToTable("UserTokens");
});
// Configure Log entity
modelBuilder.Entity<Log>(builder =>
{
builder.ToTable("Logs");
builder.Property(e => e.Application).HasMaxLength(50).IsRequired();
builder.Property(e => e.Message);
builder.Property(e => e.MessageTemplate);
builder.Property(e => e.Level).HasMaxLength(128);
builder.Property(e => e.TimeStamp);
builder.Property(e => e.Exception);
builder.Property(e => e.Properties);
builder.Property(e => e.LogEvent);
// Indexes for faster querying
builder.HasIndex(e => e.TimeStamp);
builder.HasIndex(e => e.Application);
});
// Configure Vault - AliasVaultUser relationship
modelBuilder.Entity<Vault>()
.HasOne(l => l.User)
.WithMany(c => c.Vaults)
.HasForeignKey(l => l.UserId)
.OnDelete(DeleteBehavior.Cascade);
// Configure UserEmailClaim - AliasVaultUser relationship
modelBuilder.Entity<UserEmailClaim>()
.HasOne(l => l.User)
.WithMany(c => c.EmailClaims)
.HasForeignKey(l => l.UserId)
.OnDelete(DeleteBehavior.Cascade);
// Configure Email - UserEncryptionKey relationship
modelBuilder.Entity<Email>()
.HasOne(l => l.EncryptionKey)
.WithMany(c => c.Emails)
.HasForeignKey(l => l.UserEncryptionKeyId)
.OnDelete(DeleteBehavior.NoAction);
// Configure UserEncryptionKey - AliasVaultUser relationship
modelBuilder.Entity<UserEncryptionKey>()
.HasOne(l => l.User)
.WithMany(c => c.EncryptionKeys)
.HasForeignKey(l => l.UserId)
.OnDelete(DeleteBehavior.Cascade);
}
/// <summary>
/// Sets up the connection string if it is not already configured.
/// </summary>
/// <param name="optionsBuilder">DbContextOptionsBuilder instance.</param>
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
// If the options are not already configured, use the appsettings.json file.
if (!optionsBuilder.IsConfigured)
{
var configuration = new ConfigurationBuilder()
.SetBasePath(Directory.GetCurrentDirectory())
.AddJsonFile("appsettings.json")
.Build();
optionsBuilder
.UseSqlite(configuration.GetConnectionString("AliasServerDbContext"))
.UseLazyLoadingProxies();
}
}
}