FreeSql.Repository 实现了过滤器,它不只是查询时过滤,连删除/修改/插入时都会进行验证,避免数据安全问题。git
目前过滤器依附在仓储层实现,每一个仓储实例都有 IDataFilter 属性,可利用其完成过滤器管理,它是独立的修改后不影响全局。github
public interface IDataFilter<TEntity> where TEntity : class { IDataFilter<TEntity> Apply(string filterName, Expression<Func<TEntity, bool>> filterAndValidateExp); IDisposable Enable(params string[] filterName); IDisposable EnableAll(); IDisposable Disable(params string[] filterName); IDisposable DisableAll(); bool IsEnabled(string filterName); }
using (repos1.DataFilter.Disable("test")) { //在这段中,repos1 之 test 过滤器失效 } //repos1 之 test 过滤器从新生效
dotnet add package FreeSql.Repositorysql
var fsql = new FreeSql.FreeSqlBuilder() .UseConnectionString(FreeSql.DataType.Sqlite, @"Data Source=|DataDirectory|\document.db;Pooling=true;Max Pool Size=10") .UseLogger(loggerFactory.CreateLogger<IFreeSql>()) .UseAutoSyncStructure(true) //自动迁移实体的结构到数据库 .Build(); public class Song { [Column(IsIdentity = true)] public int Id { get; set; } public string Title { get; set; } }
一、IFreeSql 的扩展方法;数据库
var curd1 = fsql.GetRepository<Song, int>(); var curd2 = fsql.GetGuidRepository<Song>();
二、继承现实;安全
public class SongRepository : BaseRepository<Song, int> { public SongRepository(IFreeSql fsql) : base(fsql) {} //在这里增长 CURD 之外的方法 }
三、Autofac 注入,使用方法继续往下看【全局过滤器】;ide
假设咱们有User(用户)、Topic(主题)两个实体,在领域类中定义了两个仓储:函数
var userRepository = fsql.GetGuidRepository<User>(); var topicRepository = fsql.GetGuidRepository<Topic>();
在开发过程当中,老是担忧 topicRepository 的数据安全问题,即有可能查询或操做到其余用户的主题。所以咱们在v0.0.7版本进行了改进,增长了 filter lambad 表达式参数。测试
var userRepository = fsql.GetGuidRepository<User>(a => a.Id == 1); var topicRepository = fsql.GetGuidRepository<Topic>(a => a.UserId == 1);
全局过滤器,可帮助实现“软删除”、“租户”等设计,目前使用 Autofac 注入的方式实现的全局过滤器。ui
public IServiceProvider ConfigureServices(IServiceCollection services) { services.AddSingleton<IFreeSql>(fsql); services.AddMvc(); var builder = new ContainerBuilder(); //示范全局过滤的仓储类注入,若是实体中不存在 Title 属性,则条件不生效 builder.RegisterFreeRepository(filter => filter.Apply<Song>("test", a => a.Title == DateTime.Now.ToString() + Thread.CurrentThread.ManagedThreadId) ); builder.Populate(services); var container = builder.Build(); return new AutofacServiceProvider(container); } public class xxxx { public int Id { get; set; } } public class Song { [Column(IsIdentity = true)] public int Id { get; set; } public string Title { get; set; } } //在控制器使用 public SongsController(GuidRepository<Song> repos1, GuidRepository<xxxx> repos2) { //在此打断点,调试 }
第一次请求:设计
repos1.Select.ToSql()
"SELECT a."Id", a."Title" \r\nFROM "Song" a \r\nWHERE (a."Title" = strftime('%Y-%m-%d %H:%M.%f',datetime(current_timestamp,'localtime')) || 21)"
repos2.Select.ToSql()
"SELECT a."Id" \r\nFROM "xxxx" a"
第二次请求:
repos1.Select.ToSql()
"SELECT a."Id", a."Title" \r\nFROM "Song" a \r\nWHERE (a."Title" = strftime('%Y-%m-%d %H:%M.%f',datetime(current_timestamp,'localtime')) || 4)"
repos2.Select.ToSql()
"SELECT a."Id" \r\nFROM "xxxx" a"
//禁用过滤器
repos1.DataFilter.Disable("test")
repos1.Select.ToSql()
"SELECT a."Id", a."Title" \r\nFROM "Song" a"
测试总结:
一、注入的变量值在使用时有了动态变化,每次获取时都是新的(Thread.CurrentThread.ManagedThreadId);
二、设定的全局过滤,若某实体不存在表达式函数中的字段时,不会生效(如上xxxx不存在Title);
三、使用 DataFilter.Disable("test") 可临时关闭过滤器的效果,使用 DataFilter.Enable("test") 可从新开启;
四、仓储对象建立时,从全局过滤器copy进来,而后本身管理本身。修改后不影响其余或全局设置。
github 源码: https://github.com/2881099/FreeSql