提问人:xunitc 提问时间:11/23/2018 更新时间:10/31/2023 访问量:1284
如何防止使用 groovy 进行 sql 注入?
How can I prevent sql injection with groovy?
问:
我有一个像这样的sql:
String sql = """SELECT id, name, sex, age, bron_year, address, phone, state, comment, is_hbp, is_dm, is_cva, is_copd, is_chd, is_cancer, is_floating, is_poor, is_disability, is_mental
FROM statistics_stin WHERE 1=1
${p.team_num == 0 ? "" : "AND team_num = ${p.team_num}"}
${p.zone == 0 ? "" : "AND team_id = ${p.zone}"}
${p.is_hbp == 2 ? "" : "AND is_hbp = ${p.is_hbp}"}
${p.is_dm == 2 ? "" : "AND is_dm = ${p.is_dm}"}
${p.is_chd == 2 ? "" : "AND is_chd = ${p.is_chd}"}
${p.is_cva == 2 ? "" : "AND is_cva = ${p.is_cva}"}
${p.is_copd == 2 ? "" : "AND is_copd = ${p.is_copd}"}
${p.is_cancer == 2 ? "" : "AND is_cancer = ${p.is_cancer}"}
${p.is_floating == 2 ? "" : "AND is_floating = ${p.is_floating}"}
${p.is_poor == 2 ? "" : "AND is_poor = ${p.is_poor}"}
${p.is_disability == 2 ? "" : "AND is_disability = ${p.is_disability}"}
${p.is_mental == 2 ? "" : "AND is_mental = ${p.is_mental}"}
${p.is_aged == 2 ? "" : (p.is_aged == 1 ? " AND age >= 65" : " AND age < 65")}
${p.is_prep_aged == 2 ? "" : (p.is_prep_aged == 1 ? "AND (age BETWEEN 60 AND 64)" : "AND (age < 60 OR age > 64)")}
${p.is_young == 2 ? "" : (p.is_young == 1 ? " AND age < 60" : " AND age >= 60")}
ORDER BY team_id ASC, id ASC
LIMIT ${start}, ${page_size}
""";
然后使用:
def datasource = ctx.lookup("jdbc/mysql");
def executer = Sql.newInstance(datasource);
def rows = executer.rows(sql);
这里的 p 是一个 json 对象,如下所示:
p = {is_aged=2, is_cancer=2, is_chd=1, is_copd=2, is_cva=2, is_disability=2, is_dm=2, is_floating=2, is_hbp=1, is_mental=2, is_poor=2, pn=1, team_num=0, zone=0}
这种方式有sql注入。我知道我可以使用像这样的参数类型:
executer.rows('SELECT * from statistics_stin WHERE is_chd=:is_chd', [is_chd: 1]);
但是这种情况有很多AND条件,使用与否将由json p来决定。
请问该怎么做?
答:
2赞
Marmite Bomber
11/23/2018
#1
您有一个动态 SQL 绑定的问题,即绑定参数的数量不是恒定的,而是取决于输入。
Tom Kyte 有一个优雅的解决方案,它在 Groovy 中实现了更优雅的实现
基本思想是简单地绑定所有变量,具有输入值并且应该使用的变量被正常处理,例如
col1 = :col1
没有输入(应忽略)的变量使用虚拟构造进行绑定:
(1=1 or :col2 is NULL)
也就是说,它们被快捷方式评估有效地忽略了。
这里是三列的两个示例
def p = ["col1" : 1, "col2" : 2, "col3" : 3]
此输入将导致完整查询
SELECT col1, col2, col3
FROM tab WHERE
col1 = :col1 AND
col2 = :col2 AND
col3 = :col3
ORDER by col1,col2,col3
对于有限的输入
p = [ "col3" : 3]
你得到这个查询
SELECT col1, col2, col3
FROM tab WHERE
(1=1 or :col1 is NULL) AND
(1=1 or :col2 is NULL) AND
col3 = :col3
ORDER by col1,col2,col3
这里是 Groovy 创建的 SQL 语句
String sql = """SELECT col1, col2, col3
FROM tab WHERE
${(!p.col1) ? "(1=1 or :col1 is NULL)" : "col1 = :col1"} AND
${(!p.col2) ? "(1=1 or :col2 is NULL)" : "col2 = :col2"} AND
${(!p.col3) ? "(1=1 or :col3 is NULL)" : "col3 = :col3"}
ORDER by col1,col2,col3
"""
你甚至可以摆脱丑陋的谓语;1=1
评论
0赞
xunitc
11/25/2018
谢谢。这样,它就可以工作了,甚至SQL也不清楚。
1赞
Mike W
11/24/2018
#2
另一种选择是在生成查询时生成绑定,然后执行rows
def query = new StringBuilder( "SELECT id, name, sex, age, bron_year, address, phone, state, comment, is_hbp, is_dm, is_cva, is_copd, is_chd, is_cancer, is_floating, is_poor, is_disability, is_mental FROM statistics_stin WHERE 1=1" )
def binds = []
if ( p.team_num == 0 ) {
query.append( ' AND team_num = ? ' )
binds << p.team_num
}
if ( p.zone == 0 ) {
query.append( ' AND team_id = ? ' )
binds << p.zone == 0
}
...
executer.rows(query.toString(), binds);
评论
0赞
Marmite Bomber
11/25/2018
构建绑定数组的不错选择。
评论