efcore在Saas系统下多租户零脚本分表分库读写分离解决方案

## 介绍
本文ShardinfCore版本x.6.0.20+
本期主角:
- [`ShardingCore`](https://github.com/dotnetcore/sharding-core) 一款ef-core下高性能、轻量级针对分表分库读写分离的解决方案,具有零依赖、零学习成本、零业务代码入侵适配

单dbcontext多数据库自动迁移

之前发过一篇文章 EFCore高级Saas系统下单DbContext如何支持不同数据库的迁移 这篇文章让efcore可以支持在单dbcontext下支持多个数据库的迁移来实现多租户下的不同数据库实现

前言

您是否有以下场景:

  • 多租户系统,数据库级别隔离
  • 大数据量,需要分表分库(动态添加),分库分表全自动维护处理
  • 租户之前可能需要使用不同的数据库模式,譬如有些租户要求用oracle,或者mmsql,或者mysql或者pgsql
  • 多租户系统在不同的数据库环境下需要维护的表结构复杂繁琐,需要维护许多脚本
  • 业务代码需要进行大范围的妥协来适应上述支持
  • 系统需要支持读写分离(动态添加)
  • 无需停机状态实时添加租户(租户线上签约)

当然我是一开始想先写这篇文章,但是写着写着发现有些时候这个问题就来了,譬如多数据库下efcore默认不支持迁移,经过不断地努力,大脑的思维宫殿我下意识就发现了解决方案,最终用一天时间解决了就是前面的一篇文章 EFCore高级Saas系统下单DbContext如何支持不同数据库的迁移 那么我们话不多说马上开始

接下来我们将实现A,B,C三个租户,其中A租户我们使用MSSQL的订单表使用按月分表,B租户我们使用MYSQL的订单表我们采用Id取模分表,C租户我们使用MSSQL也是使用订单按月分表但是起始时间和A不一样

管理租户数据

首先我们新建一个DbContext用来管理我们的租户信息

租户用户表

首先我们新建一张租户登录的用户表,每个用户就是我们对外的租户


public class SysUser {
public string Id { get; set; }
public string Name { get; set; }
public string Password { get; set; }
public DateTime CreationTime { get; set; }
public bool IsDeleted { get; set; }
}

租户配置表

然后我们新建一张租户的配置信息表用来后续初始化配置


public class SysUserTenantConfig
{
public string Id { get; set; }
public string UserId { get; set; }
/// <summary>
/// 添加ShardingTenantOptions的Json包
/// </summary>
public string ConfigJson { get; set; }
public DateTime CreationTime { get; set; }
public bool IsDeleted { get; set; }
}

定义租户配置


//为了满足上述需求我们需要对数据库和订单分片方式进行区分
public class ShardingTenantOptions
{
/// <summary>
/// 默认数据源名称
/// </summary>
public string DefaultDataSourceName { get; set;}
/// <summary>
/// 默认数据库地址
/// </summary>
public string DefaultConnectionString { get; set; }
/// <summary>
/// 数据库类型
/// </summary>
public DbTypeEnum DbType { get; set; }
/// <summary>
/// 分片模式 取模还是按月
/// </summary>
public OrderShardingTypeEnum OrderShardingType { get; set; }
/// <summary>
/// 按月分片其实时间
/// </summary>
public DateTime BeginTimeForSharding { get; set; }
/// <summary>
/// 分片迁移的命名空间
/// </summary>
public string MigrationNamespace { get; set; }
} public enum DbTypeEnum
{
MSSQL = 1,
MYSQL = 2
}   public enum OrderShardingTypeEnum
  {
  Mod=1,
  ByMonth=2
}

租户持久化DbContext

新建一个dbcontext用来存储我们的租户信息,当然你也可以使用文件或者redis之类的都行


public class IdentityDbContext:DbContext
{
public IdentityDbContext(DbContextOptions<IdentityDbContext> options):base(options)
{ } protected override void OnModelCreating(ModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.ApplyConfiguration(new SysUserMap());
modelBuilder.ApplyConfiguration(new SysUserTenantConfigMap());
}
}

这样我们就完成了租户信息的存储

租户管理者

  我们拥有了租户信息持久化的数据后需要对租户信息的使用进行配置

首先我们新建一个接口可以用来管理租户信息


public interface ITenantManager
{
/// <summary>
/// 获取所有的租户
/// </summary>
/// <returns></returns>
List<string> GetAll(); /// <summary>
/// 获取当前租户
/// </summary>
/// <returns></returns>
TenantContext GetCurrentTenantContext();
/// <summary>
/// 添加租户信息
/// </summary>
/// <param name="tenantId"></param>
/// <param name="shardingRuntimeContext"></param>
/// <returns></returns>
bool AddTenantSharding(string tenantId, IShardingRuntimeContext shardingRuntimeContext); /// <summary>
/// 创建租户环境
/// </summary>
/// <param name="tenantId"></param>
/// <returns></returns>
TenantScope CreateScope(string tenantId);
}
//租户的默认管理实现
public class DefaultTenantManager:ITenantManager
{
private readonly ITenantContextAccessor _tenantContextAccessor;
private readonly ConcurrentDictionary<string, IShardingRuntimeContext> _cache = new(); public DefaultTenantManager(ITenantContextAccessor tenantContextAccessor)
{
_tenantContextAccessor = tenantContextAccessor;
} public List<string> GetAll()
{
return _cache.Keys.ToList();
} public TenantContext GetCurrentTenantContext()
{
return _tenantContextAccessor.TenantContext;
} public bool AddTenantSharding(string tenantId, IShardingRuntimeContext shardingRuntimeContext)
{
return _cache.TryAdd(tenantId, shardingRuntimeContext);
} public TenantScope CreateScope(string tenantId)
{
if (!_cache.TryGetValue(tenantId, out var shardingRuntimeContext))
{
throw new InvalidOperationException("未找到对应租户的配置");
} _tenantContextAccessor.TenantContext = new TenantContext(shardingRuntimeContext);
return new TenantScope(_tenantContextAccessor);
}
}
//当前租户上下文访问者
public interface ITenantContextAccessor
{
TenantContext? TenantContext { get; set; }
}
//当前租户上下文访问者实现
public class TenantContextAccessor:ITenantContextAccessor
{
private static readonly AsyncLocal<TenantContext?> _tenantContext = new AsyncLocal<TenantContext?>();
public TenantContext? TenantContext
{
get => _tenantContext.Value;
set => _tenantContext.Value = value;
} }
//租户上下文
public class TenantContext
{
private readonly IShardingRuntimeContext _shardingRuntimeContext; public TenantContext(IShardingRuntimeContext shardingRuntimeContext)
{
_shardingRuntimeContext = shardingRuntimeContext;
}
public IShardingRuntimeContext GetShardingRuntimeContext()
{
return _shardingRuntimeContext;
}
}
//用来切换实现当前操作租户环境
public class TenantScope:IDisposable
{
public TenantScope(ITenantContextAccessor tenantContextAccessor)
{
TenantContextAccessor = tenantContextAccessor;
} public ITenantContextAccessor TenantContextAccessor { get; } public void Dispose()
{
}
}

构思ShardingCore如何不通过依赖注入使用

其实ShardingCore可以默认不在依赖注入中进行依赖注入,首先我们看下普通情况下ShardingCore如何实现非依赖注入获取分片上下文


var shardingRuntimeContext = new ShardingRuntimeBuilder<DefaultShardingDbContext>()
.UseRouteConfig(o =>
{
o.AddShardingTableRoute<SysUserTableRoute>();
}).UseConfig(o =>
{
o.ThrowIfQueryRouteNotMatch = false;
o.UseShardingQuery((conStr, builder) =>
{
builder.UseMySql(conStr, new MySqlServerVersion(new Version()))
.UseLoggerFactory(efLogger)
.UseQueryTrackingBehavior(QueryTrackingBehavior.NoTracking);
});
o.UseShardingTransaction((connection, builder) =>
{
builder
.UseMySql(connection, new MySqlServerVersion(new Version()))
.UseLoggerFactory(efLogger)
.UseQueryTrackingBehavior(QueryTrackingBehavior.NoTracking);
});
o.AddDefaultDataSource("ds0",
"server=127.0.0.1;port=3306;database=dbdbd0;userid=root;password=root;");
o.UseShardingMigrationConfigure(b =>
{
b.ReplaceService<IMigrationsSqlGenerator, ShardingMySqlMigrationsSqlGenerator>();
});
}).ReplaceService<ITableEnsureManager, MySqlTableEnsureManager>(ServiceLifetime.Singleton)
.Build();

这样我们就获得了IShardingRuntimeContext,将不同的IShardingRuntimeContext放到不同的数据库中我们就可以实现不同的租户了

订单表



    public class Order
{
public string Id { get; set; }
public string Name { get; set; }
public DateTime CreationTime { get; set; }
public bool IsDeleted { get; set; }
}

租户DbContext


public class TenantDbContext:AbstractShardingDbContext,IShardingTableDbContext
{
public TenantDbContext(DbContextOptions<TenantDbContext> options) : base(options)
{
} protected override void OnModelCreating(ModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.ApplyConfiguration(new OrderMap());
} public IRouteTail RouteTail { get; set; }
}

创建订单路由

订单按月分片路由

注意这边我们简单的通过采用一个静态字段来实现


public class OrderMonthTableRoute:AbstractSimpleShardingMonthKeyDateTimeVirtualTableRoute<Order>
{
private readonly ShardingTenantOptions _shardingTenantOptions; public OrderMonthTableRoute(ShardingTenantOptions shardingTenantOptions)
{
_shardingTenantOptions = shardingTenantOptions;
}
public override void Configure(EntityMetadataTableBuilder<Order> builder)
{
builder.ShardingProperty(o => o.CreationTime);
} public override bool AutoCreateTableByTime()
{
return true; } public override DateTime GetBeginTime()
{
return _shardingTenantOptions.BeginTimeForSharding;
}
}

订单取模分片路由

public class OrderModTableRoute:AbstractSimpleShardingModKeyStringVirtualTableRoute<Order>
{
private readonly ShardingTenantOptions _shardingTenantOptions; public OrderModTableRoute(ShardingTenantOptions shardingTenantOptions) : base(2, 5)
{
_shardingTenantOptions = shardingTenantOptions;
} public override void Configure(EntityMetadataTableBuilder<Order> builder)
{
builder.ShardingProperty(o => o.Id);
}
}

实现多数据库的code-first迁移

具体参考之前的博客EFCore高级Saas系统下单DbContext如何支持不同数据库的迁移

https://www.cnblogs.com/xuejiaming/p/16510482.html

分片创建者


public interface IShardingBuilder
{
IShardingRuntimeContext Build(ShardingTenantOptions tenantOptions);
} public class DefaultShardingBuilder:IShardingBuilder
{
public static readonly ILoggerFactory efLogger = LoggerFactory.Create(builder =>
{
builder.AddFilter((category, level) =>
category == DbLoggerCategory.Database.Command.Name && level == LogLevel.Information).AddConsole();
});
private readonly IServiceProvider _serviceProvider; public DefaultShardingBuilder(IServiceProvider serviceProvider)
{
_serviceProvider = serviceProvider;
}
public IShardingRuntimeContext Build(ShardingTenantOptions tenantOptions)
{
var shardingRuntimeBuilder = new ShardingRuntimeBuilder<TenantDbContext>()
.UseRouteConfig(o =>
{
if (tenantOptions.OrderShardingType == OrderShardingTypeEnum.Mod)
{
o.AddShardingTableRoute<OrderModTableRoute>();
}
if (tenantOptions.OrderShardingType == OrderShardingTypeEnum.ByMonth)
{
o.AddShardingTableRoute<OrderMonthTableRoute>();
}
}).UseConfig(o =>
{
o.ThrowIfQueryRouteNotMatch = false;
o.UseShardingQuery((conStr, builder) =>
{
if (tenantOptions.DbType == DbTypeEnum.MYSQL)
{
builder.UseMySql(conStr, new MySqlServerVersion(new Version()))
.UseMigrationNamespace(new MySqlMigrationNamespace());
}
if (tenantOptions.DbType == DbTypeEnum.MSSQL)
{
builder.UseSqlServer(conStr)
.UseMigrationNamespace(new SqlServerMigrationNamespace());
}
builder.UseLoggerFactory(efLogger)
.UseQueryTrackingBehavior(QueryTrackingBehavior.NoTracking)
.ReplaceService<IMigrationsAssembly,MultiDatabaseMigrationsAssembly>();
});
o.UseShardingTransaction((connection, builder) =>
{
if (tenantOptions.DbType == DbTypeEnum.MYSQL)
{
builder
.UseMySql(connection, new MySqlServerVersion(new Version()));
//.UseMigrationNamespace(new MySqlMigrationNamespace());//迁移只会用connection string创建所以可以不加
}
if (tenantOptions.DbType == DbTypeEnum.MSSQL)
{
builder.UseSqlServer(connection);
//.UseMigrationNamespace(new SqlServerMigrationNamespace());
}
builder.UseLoggerFactory(efLogger)
.UseQueryTrackingBehavior(QueryTrackingBehavior.NoTracking);
});
o.AddDefaultDataSource(tenantOptions.DefaultDataSourceName,tenantOptions.DefaultConnectionString);
//注意这个迁移必须要十分重要
//注意这个迁移必须要十分重要
//注意这个迁移必须要十分重要
//注意这个迁移必须要十分重要
o.UseShardingMigrationConfigure(b =>
{
if (tenantOptions.DbType == DbTypeEnum.MYSQL)
{
b.ReplaceService<IMigrationsSqlGenerator, ShardingMySqlMigrationsSqlGenerator>();
}
if (tenantOptions.DbType == DbTypeEnum.MSSQL)
{
b.ReplaceService<IMigrationsSqlGenerator, ShardingSqlServerMigrationsSqlGenerator>();
}
});
}).AddServiceConfigure(s =>
{
//IShardingRuntimeContext内部的依赖注入
s.AddSingleton(tenantOptions);
}); if (tenantOptions.DbType == DbTypeEnum.MYSQL)
{
shardingRuntimeBuilder.ReplaceService<ITableEnsureManager, MySqlTableEnsureManager>(ServiceLifetime
.Singleton);
}
if (tenantOptions.DbType == DbTypeEnum.MSSQL)
{
shardingRuntimeBuilder.ReplaceService<ITableEnsureManager, SqlServerTableEnsureManager>(ServiceLifetime
.Singleton);
}
return shardingRuntimeBuilder.Build(_serviceProvider);
}
}

到此为止基本上我们已经完成了多租户的大部分配置了,jwt部分就不在这边赘述了因为之前有实现过

Startup

主要关键的启动点我们应该怎么配置呢

启动初始化租户

首先我们需要针对程序启动后进行租户的初始化操作


public static class TenantExtension
{
public static void InitTenant(this IServiceProvider serviceProvider)
{
var tenantManager = serviceProvider.GetRequiredService<ITenantManager>();
var shardingBuilder = serviceProvider.GetRequiredService<IShardingBuilder>(); using (var scope = serviceProvider.CreateScope())
{
var identityDbContext = scope.ServiceProvider.GetRequiredService<IdentityDbContext>();
identityDbContext.Database.Migrate();
var sysUserTenantConfigs = identityDbContext.Set<SysUserTenantConfig>().ToList();
if (sysUserTenantConfigs.Any())
{
foreach (var sysUserTenantConfig in sysUserTenantConfigs)
{
var shardingTenantOptions = JsonConvert.DeserializeObject<ShardingTenantOptions>(sysUserTenantConfig.ConfigJson); var shardingRuntimeContext = shardingBuilder.Build(shardingTenantOptions); tenantManager.AddTenantSharding(sysUserTenantConfig.UserId, shardingRuntimeContext);
}
}
} var tenantIds = tenantManager.GetAll();
foreach (var tenantId in tenantIds)
{
using(tenantManager.CreateScope(tenantId))
using (var scope = serviceProvider.CreateScope())
{
var shardingRuntimeContext = tenantManager.GetCurrentTenantContext().GetShardingRuntimeContext();
//开启定时任务
shardingRuntimeContext.UseAutoShardingCreate();
var tenantDbContext = scope.ServiceProvider.GetService<TenantDbContext>();
//
tenantDbContext.Database.Migrate();
//补偿表
shardingRuntimeContext.UseAutoTryCompensateTable();
}
}
}
}

请求租户中间件

为了让我们的所有请求都可以使用指定对应的租户数据库

    public class TenantSelectMiddleware
{
private readonly RequestDelegate _next;
private readonly ITenantManager _tenantManager; public TenantSelectMiddleware(RequestDelegate next,ITenantManager tenantManager)
{
_next = next;
_tenantManager = tenantManager;
} /// <summary>
/// 1.中间件的方法必须叫Invoke,且为public,非static。
/// 2.Invoke方法第一个参数必须是HttpContext类型。
/// 3.Invoke方法必须返回Task。
/// 4.Invoke方法可以有多个参数,除HttpContext外其它参数会尝试从依赖注入容器中获取。
/// 5.Invoke方法不能有重载。
/// </summary>
/// Author : Napoleon
/// Created : 2020/1/30 21:30
public async Task Invoke(HttpContext context)
{ if (context.Request.Path.ToString().StartsWith("/api/tenant", StringComparison.CurrentCultureIgnoreCase))
{
if (!context.User.Identity.IsAuthenticated)
{
await _next(context);
return;
} var tenantId = context.User.Claims.FirstOrDefault((o) => o.Type == "uid")?.Value;
if (string.IsNullOrWhiteSpace(tenantId))
{
await DoUnAuthorized(context, "not found tenant id");
return;
} using (_tenantManager.CreateScope(tenantId))
{
await _next(context);
}
}
else
{
await _next(context);
}
} private async Task DoUnAuthorized(HttpContext context, string msg)
{
context.Response.StatusCode = 403;
await context.Response.WriteAsync(msg);
}
}

编写登录注册操作

startup处配置



    [Route("api/[controller]/[action]")]
[ApiController]
[AllowAnonymous]
public class PassportController : ControllerBase
{
private readonly IServiceProvider _serviceProvider;
private readonly IdentityDbContext _identityDbContext;
private readonly ITenantManager _tenantManager;
private readonly IShardingBuilder _shardingBuilder; public PassportController(IServiceProvider serviceProvider, IdentityDbContext identityDbContext,
ITenantManager tenantManager, IShardingBuilder shardingBuilder)
{
_serviceProvider = serviceProvider;
_identityDbContext = identityDbContext;
_tenantManager = tenantManager;
_shardingBuilder = shardingBuilder;
} [HttpPost]
public async Task<IActionResult> Register(RegisterRequest request)
{
if (await _identityDbContext.Set<SysUser>().AnyAsync(o => o.Name == request.Name))
return BadRequest("user not exists");
var sysUser = new SysUser()
{
Id = Guid.NewGuid().ToString("n"),
Name = request.Name,
Password = request.Password,
CreationTime = DateTime.Now
};
var shardingTenantOptions = new ShardingTenantOptions()
{
DbType = request.DbType,
OrderShardingType = request.OrderShardingType,
BeginTimeForSharding = request.BeginTimeForSharding.Value,
DefaultDataSourceName = "ds0",
DefaultConnectionString = GetDefaultString(request.DbType, sysUser.Id)
};
var sysUserTenantConfig = new SysUserTenantConfig()
{
Id = Guid.NewGuid().ToString("n"),
UserId = sysUser.Id,
CreationTime = DateTime.Now,
ConfigJson = JsonConvert.SerializeObject(shardingTenantOptions)
};
await _identityDbContext.AddAsync(sysUser);
await _identityDbContext.AddAsync(sysUserTenantConfig);
await _identityDbContext.SaveChangesAsync();
var shardingRuntimeContext = _shardingBuilder.Build(shardingTenantOptions);
_tenantManager.AddTenantSharding(sysUser.Id, shardingRuntimeContext);
using (_tenantManager.CreateScope(sysUser.Id))
using (var scope = _serviceProvider.CreateScope())
{
var runtimeContext = _tenantManager.GetCurrentTenantContext().GetShardingRuntimeContext();
runtimeContext.UseAutoShardingCreate(); //启动定时任务
var tenantDbContext = scope.ServiceProvider.GetService<TenantDbContext>();
tenantDbContext.Database.Migrate();
runtimeContext.UseAutoTryCompensateTable();
} return Ok();
} [HttpPost]
public async Task<IActionResult> Login(LoginRequest request)
{
var sysUser = await _identityDbContext.Set<SysUser>()
.FirstOrDefaultAsync(o => o.Name == request.Name && o.Password == request.Password);
if (sysUser == null)
return BadRequest("name or password error"); //秘钥,就是标头,这里用Hmacsha256算法,需要256bit的密钥
var securityKey =
new SigningCredentials(new SymmetricSecurityKey(Encoding.ASCII.GetBytes("123123!@#!@#123123")),
SecurityAlgorithms.HmacSha256);
//Claim,JwtRegisteredClaimNames中预定义了好多种默认的参数名,也可以像下面的Guid一样自己定义键名.
//ClaimTypes也预定义了好多类型如role、email、name。Role用于赋予权限,不同的角色可以访问不同的接口
//相当于有效载荷
var claims = new Claim[]
{
new Claim(JwtRegisteredClaimNames.Iss, "https://localhost:5000"),
new Claim(JwtRegisteredClaimNames.Aud, "api"),
new Claim("id", Guid.NewGuid().ToString("n")),
new Claim("uid", sysUser.Id),
};
SecurityToken securityToken = new JwtSecurityToken(
signingCredentials: securityKey,
expires: DateTime.Now.AddHours(2), //过期时间
claims: claims
);
var token = new JwtSecurityTokenHandler().WriteToken(securityToken);
return Ok(token);
} private string GetDefaultString(DbTypeEnum dbType, string userId)
{
switch (dbType)
{
case DbTypeEnum.MSSQL:
return $"Data Source=localhost;Initial Catalog=DB{userId};Integrated Security=True;";
case DbTypeEnum.MYSQL:
return $"server=127.0.0.1;port=3306;database=DB{userId};userid=root;password=L6yBtV6qNENrwBy7;";
default: throw new NotImplementedException();
}
}
} public class RegisterRequest
{
public string Name { get; set; }
public string Password { get; set; }
public DbTypeEnum DbType { get; set; }
public OrderShardingTypeEnum OrderShardingType { get; set; }
public DateTime? BeginTimeForSharding { get; set; }
} public class LoginRequest
{
public string Name { get; set; }
public string Password { get; set; }
}

启动配置


var builder = WebApplication.CreateBuilder(args); // Add services to the container. builder.Services.AddControllers();
builder.Services.AddAuthentication(); #region 用户系统配置 builder.Services.AddDbContext<IdentityDbContext>(o =>
o.UseSqlServer("Data Source=localhost;Initial Catalog=IdDb;Integrated Security=True;"));
//生成密钥
var keyByteArray = Encoding.ASCII.GetBytes("123123!@#!@#123123");
var signingKey = new SymmetricSecurityKey(keyByteArray);
//认证参数
builder.Services.AddAuthentication("Bearer")
.AddJwtBearer(o =>
{
o.TokenValidationParameters = new TokenValidationParameters
{
ValidateIssuerSigningKey = true,
IssuerSigningKey = signingKey,
ValidateIssuer = true,
ValidIssuer = "https://localhost:5000",
ValidateAudience = true,
ValidAudience = "api",
ValidateLifetime = true,
ClockSkew = TimeSpan.Zero,
RequireExpirationTime = true,
};
}); #endregion builder.Services.AddSingleton<ITenantManager, DefaultTenantManager>();
builder.Services.AddSingleton<ITenantContextAccessor, TenantContextAccessor>();
builder.Services.AddSingleton<IShardingBuilder, DefaultShardingBuilder>(); #region 配置ShardingCore var provider = builder.Configuration.GetValue("Provider", "UnKnown");
//Add-Migration InitialCreate -Context TenantDbContext -OutputDir Migrations\SqlServer -Args "--provider SqlServer"
//Add-Migration InitialCreate -Context TenantDbContext -OutputDir Migrations\MySql -Args "--provider MySql"
builder.Services.AddDbContext<TenantDbContext>((sp, b) =>
{
var tenantManager = sp.GetRequiredService<ITenantManager>();
var currentTenantContext = tenantManager.GetCurrentTenantContext();
//如果有上下文那么创建租户dbcontext否则就是启动命令Add-Migration
if (currentTenantContext != null)
{
var shardingRuntimeContext = currentTenantContext.GetShardingRuntimeContext();
b.UseDefaultSharding<TenantDbContext>(shardingRuntimeContext);
} if (args.IsNotEmpty())
{
//命令启动时为了保证Add-Migration正常运行
if (provider == "MySql")
{
b.UseMySql("server=127.0.0.1;port=3306;database=TenantDb;userid=root;password=L6yBtV6qNENrwBy7;",
new MySqlServerVersion(new Version()))
.UseMigrationNamespace(new MySqlMigrationNamespace())
.ReplaceService<IMigrationsAssembly, MultiDatabaseMigrationsAssembly>();
return;
} if (provider == "SqlServer")
{
b.UseSqlServer("Data Source=localhost;Initial Catalog=TenantDb;Integrated Security=True;")
.UseMigrationNamespace(new SqlServerMigrationNamespace())
.ReplaceService<IMigrationsAssembly, MultiDatabaseMigrationsAssembly>();
return;
}
}
}); #endregion var app = builder.Build(); //初始化启动配置租户信息
app.Services.InitTenant();
app.UseAuthorization(); //在认证后启用租户选择中间件
app.UseMiddleware<TenantSelectMiddleware>(); app.MapControllers(); app.Run();

添加迁移脚本

持久化identity迁移



多租户SqlServer版本



多租户MySql版本

启动程序

启动程序我们发现IdentityDbContext已经创建好了,并且支持了自动迁移

创建A租户

{
"Name":"A",
"Password":"A",
"DbType":1,
"OrderShardingType":2,
"BeginTimeForSharding":"2022-01-01",
"MigrationNamespace":"ShardingCoreMultiTenantSys.Migrations.SqlServer"
}

注意:MigrationNamespace应该自动生成,这边只是为了演示方便没写

完成

创建B租户

{
"Name":"B",
"Password":"B",
"DbType":2,
"OrderShardingType":1,
"BeginTimeForSharding":"2022-01-01",
"MigrationNamespace":"ShardingCoreMultiTenantSys.Migrations.Myql"
}

完美创建

创建C租户

{
"Name":"C",
"Password":"C",
"DbType":1,
"OrderShardingType":2,
"BeginTimeForSharding":"2022-06-01",
"MigrationNamespace":"ShardingCoreMultiTenantSys.Migrations.SqlServer"
}

C租户完美创建并且和A租户采用一样的分片规则不一样的分片起始时间

分别对abc进行crud

首先获取token,然后插入

A租户



B租户



C租户



最后完成

最后的最后

附上demo:ShardingCoreMultiTenantSys https://github.com/xuejmnet/ShardingCoreMultiTenantSys

您都看到这边了确定不点个star或者赞吗,一款.Net不得不学的分库分表解决方案,简单理解为sharding-jdbc在.net中的实现并且支持更多特性和更优秀的数据聚合,拥有原生性能的97%,并且无业务侵入性,支持未分片的所有efcore原生查询

efcore在Saas系统下多租户零脚本分表分库读写分离解决方案的更多相关文章

  1. EFCore高级Saas系统下一个DbContext如何支持多数据库迁移

    EFCore高级玩法单DbContext支持多数据库迁移 前言 随着系统的不断开发和迭代默认的efcore功能十分强大,但是随着Saas系统的引进efcore基于表字段的多租户模式已经非常完美了,但是 ...

  2. .Net下你不得不看的分表分库解决方案-多字段分片

    .Net下你不得不看的分表分库解决方案-多字段分片 介绍 本期主角:ShardingCore 一款ef-core下高性能.轻量级针对分表分库读写分离的解决方案,具有零依赖.零学习成本.零业务代码入侵 ...

  3. efcore使用ShardingCore实现分表分库下的多租户

    efcore使用ShardingCore实现分表分库下的多租户 介绍 本期主角:ShardingCore 一款ef-core下高性能.轻量级针对分表分库读写分离的解决方案,具有零依赖.零学习成本.零业 ...

  4. .Net 下高性能分表分库组件-连接模式原理

    ShardingCore ShardingCore 一款ef-core下高性能.轻量级针对分表分库读写分离的解决方案,具有零依赖.零学习成本.零业务代码入侵. Github Source Code 助 ...

  5. .Net下极限生产力之efcore分表分库全自动化迁移CodeFirst

    .Net下极限生产力之分表分库全自动化Migrations Code-First ## 介绍 本文ShardinfCore版本x.6.x.x+ 本期主角: - [`ShardingCore`](htt ...

  6. efcore分表分库原理解析

    ShardingCore ShardingCore 易用.简单.高性能.普适性,是一款扩展针对efcore生态下的分表分库的扩展解决方案,支持efcore2+的所有版本,支持efcore2+的所有数据 ...

  7. 总结下Mysql分表分库的策略及应用

    上月前面试某公司,对于mysql分表的思路,当时简要的说了下hash算法分表,以及discuz分表的思路,但是对于新增数据自增id存放的设计思想回答的不是很好(笔试+面试整个过程算是OK过了,因与个人 ...

  8. .NETCore 下支持分表分库、读写分离的通用 Repository

    首先声明这篇文章不是标题党,我说的这个类库是 FreeSql.Repository,它作为扩展库现实了通用仓储层功能,接口规范参考 abp vnext 定义,实现了基础的仓储层(CURD). 安装 d ...

  9. Linux下运行windows 系统下编辑的Python脚本显示“: 没有那个文件或目录”的过程及解决方案

    今天在 linux 系统下执行一windows下编辑的python脚本,提示(:没有那个文件或目录)英文提示:(:No such file of directory)如下: 查看文件的权限发现并没有问 ...

随机推荐

  1. Python图像处理:如何获取图像属性、兴趣ROI区域及通道处理

    摘要:本篇文章主要讲解Python调用OpenCV获取图像属性,截取感兴趣ROI区域,处理图像通道. 本文分享自华为云社区<[Python图像处理] 三.获取图像属性.兴趣ROI区域及通道处理 ...

  2. Java学习笔记-基础语法Ⅹ-进程线程

    学习快一个月了,现在学到了黑马Java教程的300集 打印流的特点: 只负责输出数据,不负责读取数据 有自己的特有方法 字节打印流:PrintStream,使用指定的文件名创建新的打印流 import ...

  3. Tutorial 3_软件工作量估计和编码规范

    软件过程与管理实验 实验3:编码规范 本次实验内容是个人软件过程部分,通过本次实验,学生将掌握以下内容: 1.建立自己的编码规范和代码审查表. 2.会用COCOMO II模型对软件工作量进行估计. [ ...

  4. API 工程化分享

    概要 本文是学习B站毛剑老师的<API 工程化分享>的学习笔记,分享了 gRPC 中的 Proto 管理方式,Proto 分仓源码方式,Proto 独立同步方式,Proto git sub ...

  5. 评估海外pop点网络质量,批量探测到整个国家运营商ip地址段时延

    1 查询当地供应商所有AS号和IP地址段,如下 可以手动复制也可以爬下来,此次测试地址不多,手动复制下来再做下格式话 61.99.128.0/17 61.99.0.0/16 61.98.96.0/20 ...

  6. 如何生成一个java文档

    如何生成一个java文档 众所周知,一个程序给别人看可能可以看懂,几万行程序就不一定了.在更多的时候,我们并不需要让别人知道我们的程序是怎么写的,只需要告诉他们怎么用的.那么,api文档就发挥了它的作 ...

  7. 56. Merge Intervals - LeetCode

    Question 56. Merge Intervals Solution 题目大意: 一个坐标轴,给你n个范围,把重叠的范围合并,返回合并后的坐标对 思路: 先排序,再遍历判断下一个开始是否在上一个 ...

  8. MyBatisPlus详解

    1.MyBatisPlus概述 需要的基础:MyBatis.Spring.SpringMVC 为什么要学习?MyBatisPlus可以节省我们大量工作时间,所有的CRUD代码它都可以自动化完成! 简介 ...

  9. Pandas:添加修改、高级过滤

    1.添加修改数据 Pandas 的数据修改是进行赋值,先把要修改的数据筛选出来,然后将同结构或者可解包的数据赋值给它: 修改数值 df.Q1 = [1, 3, 5, 7, 9] * 20 # 就会把值 ...

  10. 博弈论(nim游戏,SG函数)

    说到自己,就是个笑话.思考问题从不清晰,sg函数的问题证明方法就在眼前可却要弃掉.不过自己理解的也并不透彻,做题也不太行.耳边时不时会想起alf的:"行不行!" 基本的小概念 这里 ...