MyBatis-Plus防全表更新与删除插件BlockAttackInnerInterceptor

打印 上一主题 下一主题

主题 939|帖子 939|积分 2817

防全表更新与删除插件

BlockAttackInnerInterceptor 是 MyBatis-Plus 框架提供的一个安全插件,专门用于防止恶意的全表更新和删除操作。该插件通过拦截 update 和 delete 语句,确保这些操作不会偶然中影响到整个数据表,从而保护数据的完备性和安全性。

功能特性



  • 阻止全表更新删除:插件能够识别并阻止没有指定条件的 update 和 delete 语句,这些语句可能会导致全表数据被修改或删除。
  • 保护数据安全:通过限制全表操作,减少因误操作或恶意攻击导致的数据丢失风险。
使用方法


  • 注入插件:在 Spring Boot 设置类中,通过 @Bean 注解将 MybatisPlusInterceptor 注入到 Spring 容器中,并添加 BlockAttackInnerInterceptor 作为内部拦截器。
  1. @Configuration
  2. public class MybatisPlusConfig {
  3.     @Bean
  4.     public MybatisPlusInterceptor mybatisPlusInterceptor() {
  5.         MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
  6.         interceptor.addInnerInterceptor(new BlockAttackInnerInterceptor());
  7.         return interceptor;
  8.     }
  9. }
复制代码


  • 设置拦截规则:插件默认拦截没有指定条件的 update 和 delete 语句。假如需要自定义拦截规则,可以参考 MyBatis-Plus 的文档进行设置。
测试示例

全表更新测试

以下测试示例展示了怎样使用 BlockAttackInnerInterceptor 来防止全表更新操作。
  1. @SpringBootTest
  2. public class QueryWrapperTest {
  3.     @Autowired
  4.     private UserService userService;
  5.     /**
  6.      * SQL:UPDATE user  SET name=?,email=?;
  7.      */
  8.     @Test
  9.     public void testFullUpdate() {
  10.         User user = new User();
  11.         user.setId(999L);
  12.         user.setName("custom_name");
  13.         user.setEmail("xxx@mail.com");
  14.         // 由于没有指定更新条件,插件将抛出异常
  15.         // com.baomidou.mybatisplus.core.exceptions.MybatisPlusException: Prohibition of table update operation
  16.         Assertions.assertThrows(MybatisPlusException.class, () -> {
  17.             userService.saveOrUpdate(user, null);
  18.         });
  19.     }
  20. }
复制代码

部门更新测试

以下测试示例展示了怎样正确地执行部门更新操作,插件不会对此类操作进行拦截。
  1. @SpringBootTest
  2. public class QueryWrapperTest {
  3.     @Autowired
  4.     private UserService userService;
  5.     /**
  6.      * SQL:UPDATE user  SET name=?, email=? WHERE id = ?;
  7.      */
  8.     @Test
  9.     public void testPartialUpdate() {
  10.         LambdaUpdateWrapper<User> wrapper = new LambdaUpdateWrapper<>();
  11.         wrapper.eq(User::getId, 1);
  12.         User user = new User();
  13.         user.setId(10L);
  14.         user.setName("custom_name");
  15.         user.setEmail("xxx@mail.com");
  16.         // 由于指定了更新条件,插件不会拦截此操作
  17.         userService.saveOrUpdate(user, wrapper);
  18.     }
  19. }
复制代码

注意


  • 合理设置:确保在设置插件时,思量到项目的实际需求,避免过度限制导致正常操作受阻。
  • 测试验证:在生产环境部署前,应充分测试插件的功能,确保其按预期工作。
BlockAttackInnerInterceptor 插件是 MyBatis-Plus 提供的一个重要的安全工具,它能够有效地防止全表更新和删除操作,保护数据库免受意外或恶意的数据破坏。通过合理设置和使用该插件,可以显着进步应用程序的数据安全性。

源码阅读

  1. public class BlockAttackInnerInterceptor extends JsqlParserSupport implements InnerInterceptor {
  2.     @Override
  3.     public void beforePrepare(StatementHandler sh, Connection connection, Integer transactionTimeout) {
  4.         PluginUtils.MPStatementHandler handler = PluginUtils.mpStatementHandler(sh);
  5.         MappedStatement ms = handler.mappedStatement();
  6.         SqlCommandType sct = ms.getSqlCommandType();
  7.         if (sct == SqlCommandType.UPDATE || sct == SqlCommandType.DELETE) {
  8.             if (InterceptorIgnoreHelper.willIgnoreBlockAttack(ms.getId())) return;
  9.             BoundSql boundSql = handler.boundSql();
  10.             parserMulti(boundSql.getSql(), null);
  11.         }
  12.     }
  13.     @Override
  14.     protected void processDelete(Delete delete, int index, String sql, Object obj) {
  15.         this.checkWhere(delete.getTable().getName(), delete.getWhere(), "Prohibition of full table deletion");
  16.     }
  17.     @Override
  18.     protected void processUpdate(Update update, int index, String sql, Object obj) {
  19.         this.checkWhere(update.getTable().getName(), update.getWhere(), "Prohibition of table update operation");
  20.     }
  21.     protected void checkWhere(String tableName, Expression where, String ex) {
  22.         Assert.isFalse(this.fullMatch(where, this.getTableLogicField(tableName)), ex);
  23.     }
  24.     private boolean fullMatch(Expression where, String logicField) {
  25.         if (where == null) {
  26.             return true;
  27.         }
  28.         if (StringUtils.isNotBlank(logicField) && (where instanceof BinaryExpression)) {
  29.             BinaryExpression binaryExpression = (BinaryExpression) where;
  30.             if (StringUtils.equals(binaryExpression.getLeftExpression().toString(), logicField) || StringUtils.equals(binaryExpression.getRightExpression().toString(), logicField)) {
  31.                 return true;
  32.             }
  33.         }
  34.         if (where instanceof EqualsTo) {
  35.             // example: 1=1
  36.             EqualsTo equalsTo = (EqualsTo) where;
  37.             return StringUtils.equals(equalsTo.getLeftExpression().toString(), equalsTo.getRightExpression().toString());
  38.         } else if (where instanceof NotEqualsTo) {
  39.             // example: 1 != 2
  40.             NotEqualsTo notEqualsTo = (NotEqualsTo) where;
  41.             return !StringUtils.equals(notEqualsTo.getLeftExpression().toString(), notEqualsTo.getRightExpression().toString());
  42.         } else if (where instanceof OrExpression) {
  43.             OrExpression orExpression = (OrExpression) where;
  44.             return fullMatch(orExpression.getLeftExpression(), logicField) || fullMatch(orExpression.getRightExpression(), logicField);
  45.         } else if (where instanceof AndExpression) {
  46.             AndExpression andExpression = (AndExpression) where;
  47.             return fullMatch(andExpression.getLeftExpression(), logicField) && fullMatch(andExpression.getRightExpression(), logicField);
  48.         } else if (where instanceof Parenthesis) {
  49.             // example: (1 = 1)
  50.             Parenthesis parenthesis = (Parenthesis) where;
  51.             return fullMatch(parenthesis.getExpression(), logicField);
  52.         }
  53.         return false;
  54.     }
  55.     /**
  56.      * 获取表名中的逻辑删除字段
  57.      *
  58.      * @param tableName 表名
  59.      * @return 逻辑删除字段
  60.      */
  61.     private String getTableLogicField(String tableName) {
  62.         if (StringUtils.isBlank(tableName)) {
  63.             return StringUtils.EMPTY;
  64.         }
  65.         TableInfo tableInfo = TableInfoHelper.getTableInfo(tableName);
  66.         if (tableInfo == null || !tableInfo.isWithLogicDelete() || tableInfo.getLogicDeleteFieldInfo() == null) {
  67.             return StringUtils.EMPTY;
  68.         }
  69.         return tableInfo.getLogicDeleteFieldInfo().getColumn();
  70.     }
  71. }
复制代码


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

风雨同行

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表