1
0
mirror of https://github.com/bitwarden/server.git synced 2025-04-06 05:28:15 -05:00

fixes for databasecontext and json types

This commit is contained in:
Kyle Spearrin 2020-01-09 07:57:33 -05:00
parent 002efaafd4
commit 8026912eeb
9 changed files with 130 additions and 74 deletions

View File

@ -39,7 +39,7 @@
<PackageReference Include="Microsoft.Extensions.Configuration.EnvironmentVariables" Version="2.2.4" /> <PackageReference Include="Microsoft.Extensions.Configuration.EnvironmentVariables" Version="2.2.4" />
<PackageReference Include="Microsoft.Extensions.Configuration.UserSecrets" Version="2.2.0" /> <PackageReference Include="Microsoft.Extensions.Configuration.UserSecrets" Version="2.2.0" />
<PackageReference Include="Microsoft.Extensions.Identity.Stores" Version="2.2.0" /> <PackageReference Include="Microsoft.Extensions.Identity.Stores" Version="2.2.0" />
<PackageReference Include="Npgsql" Version="4.0.9" /> <PackageReference Include="Npgsql" Version="4.1.2" />
<PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" Version="2.2.4" /> <PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" Version="2.2.4" />
<PackageReference Include="Quartz" Version="3.0.7" /> <PackageReference Include="Quartz" Version="3.0.7" />
<PackageReference Include="Serilog.AspNetCore" Version="3.0.0" /> <PackageReference Include="Serilog.AspNetCore" Version="3.0.0" />

View File

@ -10,6 +10,7 @@ namespace Bit.Core.Models.EntityFramework
public User User { get; set; } public User User { get; set; }
public Organization Organization { get; set; } public Organization Organization { get; set; }
[IgnoreMap]
public JsonDocument DataJson public JsonDocument DataJson
{ {
get => _dataJson; get => _dataJson;
@ -19,6 +20,7 @@ namespace Bit.Core.Models.EntityFramework
_dataJson = value; _dataJson = value;
} }
} }
[IgnoreMap]
public JsonDocument AttachmentsJson public JsonDocument AttachmentsJson
{ {
get => _attachmentsJson; get => _attachmentsJson;

View File

@ -9,7 +9,8 @@ namespace Bit.Core.Models.EntityFramework
private JsonDocument _twoFactorProvidersJson; private JsonDocument _twoFactorProvidersJson;
public ICollection<Cipher> Ciphers { get; set; } public ICollection<Cipher> Ciphers { get; set; }
[IgnoreMap]
public JsonDocument TwoFactorProvidersJson public JsonDocument TwoFactorProvidersJson
{ {
get => _twoFactorProvidersJson; get => _twoFactorProvidersJson;

View File

@ -10,6 +10,7 @@ namespace Bit.Core.Models.EntityFramework
public ICollection<Cipher> Ciphers { get; set; } public ICollection<Cipher> Ciphers { get; set; }
[IgnoreMap]
public JsonDocument TwoFactorProvidersJson public JsonDocument TwoFactorProvidersJson
{ {
get => _twoFactorProvidersJson; get => _twoFactorProvidersJson;

View File

@ -1,16 +1,22 @@
using AutoMapper; using AutoMapper;
using Microsoft.Extensions.DependencyInjection;
namespace Bit.Core.Repositories.EntityFramework namespace Bit.Core.Repositories.EntityFramework
{ {
public abstract class BaseEntityFrameworkRepository public abstract class BaseEntityFrameworkRepository
{ {
public BaseEntityFrameworkRepository(DatabaseContext databaseContext, IMapper mapper) public BaseEntityFrameworkRepository(IServiceScopeFactory serviceScopeFactory, IMapper mapper)
{ {
DatabaseContext = databaseContext; ServiceScopeFactory = serviceScopeFactory;
Mapper = mapper; Mapper = mapper;
} }
protected DatabaseContext DatabaseContext { get; private set; } protected IServiceScopeFactory ServiceScopeFactory { get; private set; }
protected IMapper Mapper { get; private set; } protected IMapper Mapper { get; private set; }
public DatabaseContext GetDatabaseContext(IServiceScope serviceScope)
{
return serviceScope.ServiceProvider.GetRequiredService<DatabaseContext>();
}
} }
} }

View File

@ -6,34 +6,20 @@ namespace Bit.Core.Repositories.EntityFramework
{ {
public class DatabaseContext : DbContext public class DatabaseContext : DbContext
{ {
private readonly GlobalSettings _globalSettings; public DatabaseContext(DbContextOptions<DatabaseContext> options)
public DatabaseContext(
DbContextOptions<DatabaseContext> options,
GlobalSettings globalSettings)
: base(options) : base(options)
{ { }
_globalSettings = globalSettings;
}
public DbSet<User> Users { get; set; } public DbSet<User> Users { get; set; }
public DbSet<Cipher> Ciphers { get; set; } public DbSet<Cipher> Ciphers { get; set; }
public DbSet<Organization> Organizations { get; set; } public DbSet<Organization> Organizations { get; set; }
protected override void OnConfiguring(DbContextOptionsBuilder builder)
{
if(!string.IsNullOrWhiteSpace(_globalSettings.PostgreSql?.ConnectionString))
{
builder.UseNpgsql(_globalSettings.PostgreSql.ConnectionString);
}
}
protected override void OnModelCreating(ModelBuilder builder) protected override void OnModelCreating(ModelBuilder builder)
{ {
builder.Entity<Cipher>().Ignore(e => e.Data); builder.Entity<Cipher>().Ignore(e => e.Data);
builder.Entity<Cipher>().Property(e => e.Data).HasColumnName("Data"); builder.Entity<Cipher>().Property(e => e.DataJson).HasColumnName("Data");
builder.Entity<Cipher>().Ignore(e => e.Attachments); builder.Entity<Cipher>().Ignore(e => e.Attachments);
builder.Entity<Cipher>().Property(e => e.Attachments).HasColumnName("Attachments"); builder.Entity<Cipher>().Property(e => e.AttachmentsJson).HasColumnName("Attachments");
builder.Entity<User>().Ignore(e => e.TwoFactorProviders); builder.Entity<User>().Ignore(e => e.TwoFactorProviders);
builder.Entity<User>().Property(e => e.TwoFactorProvidersJson).HasColumnName("TwoFactorProviders"); builder.Entity<User>().Property(e => e.TwoFactorProvidersJson).HasColumnName("TwoFactorProviders");

View File

@ -7,19 +7,24 @@ using System.Linq;
using System.Collections.Generic; using System.Collections.Generic;
using AutoMapper; using AutoMapper;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.DependencyInjection;
namespace Bit.Core.Repositories.EntityFramework namespace Bit.Core.Repositories.EntityFramework
{ {
public class OrganizationRepository : Repository<TableModel.Organization, EFModel.Organization, Guid>, IOrganizationRepository public class OrganizationRepository : Repository<TableModel.Organization, EFModel.Organization, Guid>, IOrganizationRepository
{ {
public OrganizationRepository(DatabaseContext databaseContext, IMapper mapper) public OrganizationRepository(IServiceScopeFactory serviceScopeFactory, IMapper mapper)
: base(databaseContext, mapper, () => databaseContext.Organizations) : base(serviceScopeFactory, mapper, (DatabaseContext context) => context.Organizations)
{ } { }
public async Task<ICollection<TableModel.Organization>> GetManyByEnabledAsync() public async Task<ICollection<TableModel.Organization>> GetManyByEnabledAsync()
{ {
var organizations = await GetDbSet().Where(e => e.Enabled).ToListAsync(); using(var scope = ServiceScopeFactory.CreateScope())
return Mapper.Map<List<TableModel.Organization>>(organizations); {
var dbContext = GetDatabaseContext(scope);
var organizations = await GetDbSet(dbContext).Where(e => e.Enabled).ToListAsync();
return Mapper.Map<List<TableModel.Organization>>(organizations);
}
} }
public async Task<ICollection<TableModel.Organization>> GetManyByUserIdAsync(Guid userId) public async Task<ICollection<TableModel.Organization>> GetManyByUserIdAsync(Guid userId)
@ -31,13 +36,17 @@ namespace Bit.Core.Repositories.EntityFramework
public async Task<ICollection<TableModel.Organization>> SearchAsync(string name, string userEmail, bool? paid, public async Task<ICollection<TableModel.Organization>> SearchAsync(string name, string userEmail, bool? paid,
int skip, int take) int skip, int take)
{ {
// TODO: more filters using(var scope = ServiceScopeFactory.CreateScope())
var organizations = await GetDbSet() {
var dbContext = GetDatabaseContext(scope);
// TODO: more filters
var organizations = await GetDbSet(dbContext)
.Where(e => name == null || e.Name.StartsWith(name)) .Where(e => name == null || e.Name.StartsWith(name))
.OrderBy(e => e.Name) .OrderBy(e => e.Name)
.Skip(skip).Take(take) .Skip(skip).Take(take)
.ToListAsync(); .ToListAsync();
return Mapper.Map<List<TableModel.Organization>>(organizations); return Mapper.Map<List<TableModel.Organization>>(organizations);
}
} }
public async Task UpdateStorageAsync(Guid id) public async Task UpdateStorageAsync(Guid id)
@ -47,7 +56,10 @@ namespace Bit.Core.Repositories.EntityFramework
public async Task<ICollection<DataModel.OrganizationAbility>> GetManyAbilitiesAsync() public async Task<ICollection<DataModel.OrganizationAbility>> GetManyAbilitiesAsync()
{ {
return await GetDbSet() using(var scope = ServiceScopeFactory.CreateScope())
{
var dbContext = GetDatabaseContext(scope);
return await GetDbSet(dbContext)
.Select(e => new DataModel.OrganizationAbility .Select(e => new DataModel.OrganizationAbility
{ {
Enabled = e.Enabled, Enabled = e.Enabled,
@ -57,6 +69,7 @@ namespace Bit.Core.Repositories.EntityFramework
UsersGetPremium = e.UsersGetPremium, UsersGetPremium = e.UsersGetPremium,
Using2fa = e.Use2fa && e.TwoFactorProviders != null, Using2fa = e.Use2fa && e.TwoFactorProviders != null,
}).ToListAsync(); }).ToListAsync();
}
} }
} }
} }

View File

@ -3,6 +3,7 @@ using System.Threading.Tasks;
using Bit.Core.Models.Table; using Bit.Core.Models.Table;
using AutoMapper; using AutoMapper;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.DependencyInjection;
namespace Bit.Core.Repositories.EntityFramework namespace Bit.Core.Repositories.EntityFramework
{ {
@ -11,35 +12,47 @@ namespace Bit.Core.Repositories.EntityFramework
where T : class, ITableObject<TId> where T : class, ITableObject<TId>
where TEntity : class, ITableObject<TId> where TEntity : class, ITableObject<TId>
{ {
public Repository(DatabaseContext databaseContext, IMapper mapper, Func<DbSet<TEntity>> getDbSet) public Repository(IServiceScopeFactory serviceScopeFactory, IMapper mapper, Func<DatabaseContext, DbSet<TEntity>> getDbSet)
: base(databaseContext, mapper) : base(serviceScopeFactory, mapper)
{ {
GetDbSet = getDbSet; GetDbSet = getDbSet;
} }
protected Func<DbSet<TEntity>> GetDbSet { get; private set; } protected Func<DatabaseContext, DbSet<TEntity>> GetDbSet { get; private set; }
public virtual async Task<T> GetByIdAsync(TId id) public virtual async Task<T> GetByIdAsync(TId id)
{ {
var entity = await GetDbSet().FindAsync(id); using(var scope = ServiceScopeFactory.CreateScope())
return entity as T; {
var dbContext = GetDatabaseContext(scope);
var entity = await GetDbSet(dbContext).FindAsync(id);
return entity as T;
}
} }
public virtual async Task CreateAsync(T obj) public virtual async Task CreateAsync(T obj)
{ {
var entity = Mapper.Map<TEntity>(obj); using(var scope = ServiceScopeFactory.CreateScope())
DatabaseContext.Add(entity); {
await DatabaseContext.SaveChangesAsync(); var dbContext = GetDatabaseContext(scope);
var entity = Mapper.Map<TEntity>(obj);
dbContext.Add(entity);
await dbContext.SaveChangesAsync();
}
} }
public virtual async Task ReplaceAsync(T obj) public virtual async Task ReplaceAsync(T obj)
{ {
var entity = await GetDbSet().FindAsync(obj.Id); using(var scope = ServiceScopeFactory.CreateScope())
if(entity != null)
{ {
var mappedEntity = Mapper.Map<TEntity>(obj); var dbContext = GetDatabaseContext(scope);
DatabaseContext.Entry(entity).CurrentValues.SetValues(mappedEntity); var entity = await GetDbSet(dbContext).FindAsync(obj.Id);
await DatabaseContext.SaveChangesAsync(); if(entity != null)
{
var mappedEntity = Mapper.Map<TEntity>(obj);
dbContext.Entry(entity).CurrentValues.SetValues(mappedEntity);
await dbContext.SaveChangesAsync();
}
} }
} }
@ -57,9 +70,13 @@ namespace Bit.Core.Repositories.EntityFramework
public virtual async Task DeleteAsync(T obj) public virtual async Task DeleteAsync(T obj)
{ {
var entity = Mapper.Map<TEntity>(obj); using(var scope = ServiceScopeFactory.CreateScope())
DatabaseContext.Entry(entity).State = EntityState.Deleted; {
await DatabaseContext.SaveChangesAsync(); var dbContext = GetDatabaseContext(scope);
var entity = Mapper.Map<TEntity>(obj);
dbContext.Entry(entity).State = EntityState.Deleted;
await dbContext.SaveChangesAsync();
}
} }
} }
} }

View File

@ -7,54 +7,80 @@ using System.Threading.Tasks;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;
using Microsoft.Extensions.DependencyInjection;
namespace Bit.Core.Repositories.EntityFramework namespace Bit.Core.Repositories.EntityFramework
{ {
public class UserRepository : Repository<TableModel.User, EFModel.User, Guid>, IUserRepository public class UserRepository : Repository<TableModel.User, EFModel.User, Guid>, IUserRepository
{ {
public UserRepository(DatabaseContext databaseContext, IMapper mapper) public UserRepository(IServiceScopeFactory serviceScopeFactory, IMapper mapper)
: base(databaseContext, mapper, () => databaseContext.Users) : base(serviceScopeFactory, mapper, (DatabaseContext context) => context.Users)
{ } { }
public async Task<TableModel.User> GetByEmailAsync(string email) public async Task<TableModel.User> GetByEmailAsync(string email)
{ {
return await GetDbSet().FirstOrDefaultAsync(e => e.Email == email); using(var scope = ServiceScopeFactory.CreateScope())
{
var dbContext = GetDatabaseContext(scope);
return await GetDbSet(dbContext).FirstOrDefaultAsync(e => e.Email == email);
}
} }
public async Task<DataModel.UserKdfInformation> GetKdfInformationByEmailAsync(string email) public async Task<DataModel.UserKdfInformation> GetKdfInformationByEmailAsync(string email)
{ {
return await GetDbSet().Where(e => e.Email == email) using(var scope = ServiceScopeFactory.CreateScope())
.Select(e => new DataModel.UserKdfInformation {
{ var dbContext = GetDatabaseContext(scope);
Kdf = e.Kdf, return await GetDbSet(dbContext).Where(e => e.Email == email)
KdfIterations = e.KdfIterations .Select(e => new DataModel.UserKdfInformation
}).SingleOrDefaultAsync(); {
Kdf = e.Kdf,
KdfIterations = e.KdfIterations
}).SingleOrDefaultAsync();
}
} }
public async Task<ICollection<TableModel.User>> SearchAsync(string email, int skip, int take) public async Task<ICollection<TableModel.User>> SearchAsync(string email, int skip, int take)
{ {
var users = await GetDbSet() using(var scope = ServiceScopeFactory.CreateScope())
.Where(e => email == null || e.Email.StartsWith(email)) {
.OrderBy(e => e.Email) var dbContext = GetDatabaseContext(scope);
.Skip(skip).Take(take) var users = await GetDbSet(dbContext)
.ToListAsync(); .Where(e => email == null || e.Email.StartsWith(email))
return Mapper.Map<List<TableModel.User>>(users); .OrderBy(e => e.Email)
.Skip(skip).Take(take)
.ToListAsync();
return Mapper.Map<List<TableModel.User>>(users);
}
} }
public async Task<ICollection<TableModel.User>> GetManyByPremiumAsync(bool premium) public async Task<ICollection<TableModel.User>> GetManyByPremiumAsync(bool premium)
{ {
var users = await GetDbSet().Where(e => e.Premium == premium).ToListAsync(); using(var scope = ServiceScopeFactory.CreateScope())
return Mapper.Map<List<TableModel.User>>(users); {
var dbContext = GetDatabaseContext(scope);
var users = await GetDbSet(dbContext).Where(e => e.Premium == premium).ToListAsync();
return Mapper.Map<List<TableModel.User>>(users);
}
} }
public async Task<string> GetPublicKeyAsync(Guid id) public async Task<string> GetPublicKeyAsync(Guid id)
{ {
return await GetDbSet().Where(e => e.Id == id).Select(e => e.PublicKey).SingleOrDefaultAsync(); using(var scope = ServiceScopeFactory.CreateScope())
{
var dbContext = GetDatabaseContext(scope);
return await GetDbSet(dbContext).Where(e => e.Id == id).Select(e => e.PublicKey).SingleOrDefaultAsync();
}
} }
public async Task<DateTime> GetAccountRevisionDateAsync(Guid id) public async Task<DateTime> GetAccountRevisionDateAsync(Guid id)
{ {
return await GetDbSet().Where(e => e.Id == id).Select(e => e.AccountRevisionDate).SingleOrDefaultAsync(); using(var scope = ServiceScopeFactory.CreateScope())
{
var dbContext = GetDatabaseContext(scope);
return await GetDbSet(dbContext).Where(e => e.Id == id).Select(e => e.AccountRevisionDate)
.SingleOrDefaultAsync();
}
} }
public async Task UpdateStorageAsync(Guid id) public async Task UpdateStorageAsync(Guid id)
@ -64,15 +90,19 @@ namespace Bit.Core.Repositories.EntityFramework
public async Task UpdateRenewalReminderDateAsync(Guid id, DateTime renewalReminderDate) public async Task UpdateRenewalReminderDateAsync(Guid id, DateTime renewalReminderDate)
{ {
var user = new EFModel.User using(var scope = ServiceScopeFactory.CreateScope())
{ {
Id = id, var dbContext = GetDatabaseContext(scope);
RenewalReminderDate = renewalReminderDate var user = new EFModel.User
}; {
var set = GetDbSet(); Id = id,
set.Attach(user); RenewalReminderDate = renewalReminderDate
DatabaseContext.Entry(user).Property(e => e.RenewalReminderDate).IsModified = true; };
await DatabaseContext.SaveChangesAsync(); var set = GetDbSet(dbContext);
set.Attach(user);
dbContext.Entry(user).Property(e => e.RenewalReminderDate).IsModified = true;
await dbContext.SaveChangesAsync();
}
} }
} }
} }