【Web】基于Mybatis的SQL注入漏洞利用点学习笔记

目录

MyBatis传参占位符区别

不能直接用#{}的情况

in多参数值查询

[like %%模糊查询](#like %%模糊查询)

[order by列名参数化](#order by列名参数化)


MyBatis传参占位符区别

在 MyBatis 中,#{}${} 都是用于传参的占位符,但它们之间有很大的区别,主要体现在两个方面:参数值的类型和 SQL 注入问题。

1.参数值的类型:

① #{} 是预编译参数,表示使用 PreparedStatement 时,使用 setXXX() 方法设置参数值,会对传入的参数进行类型处理,确保传递的参数类型正确。一般来说,我们在使用 MyBatis 进行 CRUD 操作时,都应该使用 #{} 来传递参数,这样可以有效地避免 SQL 注入问题。

{} 是字符串拼接,表示在 SQL 语句中直接插入传入的参数值,不会对传入的参数类型进行处理。如果传入的参数是字符串类型,那么在 SQL 语句中使用时,需要使用单引号将其括起来,否则会抛出 SQL 语法错误异常。因此,在使用 {} 时,需要特别注意避免 SQL 注入问题。

简而言之:

复制代码
如果传入Z3r'4y
# PrepareStatement ? 填充参数 类型化 转义处理 返回一个'Z3r\'4y'
$ Statement 拼接参数 返回一个Z3r'4y

2.SQL 注入问题:

① #{} 可以避免 SQL 注入问题,因为它会对传入的参数进行类型处理,并将参数值转义后再放到 SQL 语句中,保证了 SQL 语句的安全性。

② ${} 存在 SQL 注入问题,因为 SQL 语句中插入的是传入的参数值,如果参数值中包含 SQL 语句的关键字或特殊字符,可能会导致 SQL 注入攻击。

举个例子:

复制代码
<!-- 使用#,不存在漏洞;默认都是PREPARED-->
    <select id="findUserById" resultType="com.example.mybaits.entity.User" statementType="PREPARED">
        SELECT * FROM tbuser where id= #{id}
    </select>

<!-- 使用$,存在注入漏洞 -->
    <select id="findUserByName" resultType="com.example.mybaits.entity.User">
        SELECT * FROM tbuser where userName='${username}'
    </select>

总结一下:原则就是能用#{}就不要用${}

不能直接用#{}的情况

因为一些sql语言的性质,一些情况不能直接用#{}(传参强制被引号包裹导致sql语义错误)

比如下面这种情况:

如果直接SELECT * FROM tbuser where id in (#{ids})并传参ids=1,2,3就会拼接成:

SELECT * FROM tbuser where id in ('1,2,3');

从而出现只调出一行的谬误

解决方案如下:

in多参数值查询

复制代码
 <!-- in查询,错误写法 -->
    <select id="findUserInIds" resultType="com.example.mybaits.entity.User" >
        SELECT * FROM tbuser where id in (${ids})
    </select>

    <!-- in查询,正确写法 -->
    <select id="findUserInIdsRight" resultType="com.example.mybaits.entity.User" >
        SELECT * FROM tbuser where id in
        <foreach collection="list" item="ids" open="(" close=")" separator=",">
            #{ids}
        </foreach>
    </select>

like %%模糊查询

复制代码
 <!-- 模糊查询,错误写法 -->

<select id="findUserLikeName" resultType="com.example.mybaits.entity.User">
        SELECT * FROM tbuser where userName like '%${username}%'
    </select>

 <!-- 模糊查询,正确写法 -->

<select id="findUserLikeName" resultType="com.example.mybaits.entity.User">
        SELECT * FROM tbuser where userName like concat('%',#{username}, '%')
    </select>

order by列名参数化

复制代码
<!--动态指定列名排序,有注入漏洞-->

<select id="allUserSeq" resultType="com.example.mybaits.entity.User">
    SELECT *FROM tbuser order by ${colName}
</select>

基于MyBatis并不能很好解决这种注入漏洞

相关推荐
牧羊人_myr5 分钟前
Maven核心功能与项目构建详解
java·maven
量子物理学23 分钟前
Eclipse Mosquitto 在小内存下怎么修改配置文件
java·服务器·eclipse
l1t23 分钟前
用parser_tools插件来解析SQL语句
数据库·sql·插件·duckdb
程序员鱼皮35 分钟前
让老弟做个数据同步,结果踩了 7 个大坑!
java·后端·计算机·程序员·编程·职场
TDengine (老段)41 分钟前
TDengine 数学函数 ABS() 用户手册
大数据·数据库·sql·物联网·时序数据库·tdengine·涛思数据
lypzcgf43 分钟前
Coze源码分析-资源库-编辑数据库-后端源码-安全与错误处理
数据库·安全·系统架构·coze·coze源码分析·ai应用平台·agent平台
Iris76144 分钟前
MyBatis一对多关系映射方式
java
程序员清风44 分钟前
滴滴二面:MySQL执行计划中,Key有值,还是很慢怎么办?
java·后端·面试
白鲸开源1 小时前
3.1.8<3.2.0<3.3.1,Apache DolphinScheduler集群升级避坑指南
java·开源·github
Ytadpole1 小时前
客户端加密 和 服务端加密:端到端安全的真正含义
安全·加解密