Mybatis Order by動態(tài)參數(shù)防注入方式
一、先提及一下Mybatis動態(tài)參數(shù)
| 參數(shù)符號 | 編譯 | 安全 | 值 |
| #{} | 預(yù)編譯 | 安全 | 處理后的值,字符類型都帶雙引號 |
| ${} | 未預(yù)編譯 | 不安全,存在SQL注入問題 | 傳進來啥就是啥 |
二、order by 動態(tài)參數(shù)
order by 后面參數(shù)值是表字段或者SQL關(guān)鍵字
所以使用#{} 是無效的,只能使用${}
那么SQL注入問題就來了
三、解決Order by動態(tài)參數(shù)注入問題
1、使用正則表達式規(guī)避
特殊字符 * + - / _ 等等
使用indexOf判斷到了直接返回
有可能會存在其它情況,不能完全規(guī)避,但是可以逐步排查
2、技巧解決
- 2.1 先從order by 動態(tài)參數(shù)思考
組合情況只有這兩種
- order by 字段名 (asc直接略掉得了)
- order by 字段名 desc
所以我們只要找到對應(yīng)集合,判斷我們動態(tài)排序條件是否在其中就可以了
- 2.2 獲取排序條件集合
有些勤勞的小伙伴可能就開始寫了
userOrderSet = ['id','id desc','age','age desc',.......] scoreOrderSet = ['yuwen','yuwen desc','shuxue','shuxue desc',.......] .............
要是有n多表n多字段可是要急死人
我們使用注解+映射來獲取
- 2.3 動態(tài)獲取集合
/**
首先改造實體類,有@Column注解映射,也有使用@JsonProperty,都行,沒有用@Column映射,就加個@JsonProperty,不影響,我偷懶使用@JsonProperty
**/
@Data
@Builder
@JsonIgnoreProperties(ignoreUnknown = true)
public class ContentEntity {
@JsonProperty("id")
private Long id;//主鍵ID
@JsonProperty("code")
private String code;//編碼
@JsonProperty("content")
private String content;//內(nèi)容
@JsonProperty("is_del")
private Integer isDel;//是否刪除,0未刪除,1已刪除
@JsonProperty("creator")
private String creator;//創(chuàng)建人
@JsonProperty("creator_id")
@JsonFormat(pattern = DatePattern.NORM_DATETIME_PATTERN, timezone = "GMT+8")
private Date createAt;//創(chuàng)建時間
@JsonProperty("updater")
private String updater;//更新人
@JsonProperty("updater_id")
@JsonFormat(pattern = DatePattern.NORM_DATETIME_PATTERN, timezone = "GMT+8")
private Date updateAt;//更新時間
}/**工具類來了**/
public class MybatisDynamicOrderUtils {
private static final String desc = " desc";
/**
* 獲取對象 JsonProperty 值列表 使用Column替換一下
* @param object
* @return
*/
public static Set<String> getParamJsonPropertyValue(Class<?> object){
try {
//獲取filed數(shù)組
Set<String> resultList = new HashSet<>();
Field[] fields = object.getDeclaredFields();
for (Field field:fields){
//獲取JsonProperty注解
if(field.getAnnotation(JsonProperty.class)!=null){
JsonProperty annotation = field.getAnnotation(JsonProperty.class);
if (annotation != null) {
//獲取JsonProperty 的值
String jsonPropertyValue = annotation.value();
resultList.add(jsonPropertyValue);
}
}
}
return resultList;
}catch (Exception e){
e.printStackTrace();
}
return null;
}
/**
* 判斷動態(tài)order是否是合理
* @param order
* @param object
* @return
*/
public static Boolean isDynamicOrderValue(String order,Class<?> object){
//先獲取JsonProperty 注解中的集合
Set<String> set = getParamJsonPropertyValue(object);
//屬于直屬字段 直接返回
if(set.contains(order)){
return true;
}
//多了倒序,先去除倒序字段再判斷
if(order.lastIndexOf(desc)>0){
String temp = order.substring(0,order.lastIndexOf(desc));
if(set.contains(temp)){
return true;
}
}
return false;
}
}//調(diào)用操作一下
//檢驗動態(tài)order是否合理,防止SQL注入
if(!MybatisDynamicOrderUtils.isDynamicOrderValue(sort,ContentEntity.class)){
log.error("dynamic order is error:{}",sort);
return null;
}
//mapper.class
@Select({"<script>select * from content order by ${sort}</script>"})
List<ContentEntity> getList(@Param("sort") String sort);總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Sentinel結(jié)合Nacos實現(xiàn)數(shù)據(jù)持久化過程詳解
這篇文章主要介紹了Sentinel結(jié)合Nacos實現(xiàn)數(shù)據(jù)持久化過程,要持久化的原因是因為每次啟動Sentinel都會使之前配置的規(guī)則就清空了,這樣每次都要再去設(shè)定規(guī)則顯得非常的麻煩,感興趣想要詳細(xì)了解可以參考下文2023-05-05
Java多線程和并發(fā)基礎(chǔ)面試題(問答形式)
多線程和并發(fā)問題是Java技術(shù)面試中面試官比較喜歡問的問題之一。在這里,從面試的角度列出了大部分重要的問題,感興趣的小伙伴們可以參考一下2016-06-06
SpringBoot 配置文件加載位置與優(yōu)先級問題詳解
這篇文章主要介紹了SpringBoot 配置文件加載位置與優(yōu)先級問題,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-09-09
java數(shù)據(jù)結(jié)構(gòu)基礎(chǔ):順序隊列和循環(huán)隊列
下面小編就為大家分享一篇java隊列實現(xiàn)方法(順序隊列,循環(huán)隊列),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-08-08
Spring?@Conditional通過條件控制bean注冊過程
這篇文章主要為大家介紹了Spring?@Conditional通過條件控制bean注冊過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-02-02
Mybatis Interceptor 攔截器的實現(xiàn)
這篇文章主要介紹了Mybatis Interceptor 攔截器的實現(xiàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-12-12

