精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Java安全編碼之SQL注入

安全 應用安全
本文以Java項目廣泛采用的兩個框架Hibernate和MyBatis 為例來介紹,如何在編碼過程中避免SQL注入的幾種編碼方法,包括對預編譯的深度解析,以及對預編譯理解的幾個“誤區”進行了解釋。

隨著互聯網的發展,Java語言在金融服務業、電子商務、大數據技術等方面的應用極其廣泛。Java安全編碼規范早已成為SDL中不可或缺的一部分。本文以Java項目廣泛采用的兩個框架Hibernate和MyBatis 為例來介紹,如何在編碼過程中避免SQL注入的幾種編碼方法,包括對預編譯的深度解析,以及對預編譯理解的幾個“誤區”進行了解釋。

備注,本文是Java語言安全編碼會是系列文章的第一篇。

1. 框架介紹

目前Hibernate和MyBatis為java項目廣泛采用的兩個框架。由于Hibernate使用方便,以前的項目采用Hibernate非常的廣泛,但是后面由于Hibernate的侵入式特性,后面慢慢被MyBatis所取代 。下面我們會以SpringBoot為基礎,分別搭建Hibernate和MyBatis的漏洞環境。

2. 配置說明

SpringBoot采用2.3.1.RELEASE,MySQL版本為5.7.20。數據庫有一張表user_tbl。數據如下:

1596704219.png!small

3. Hibernate

Hibernate 是一個開放源代碼的對象關系映射框架,它對 JDBC 進行了非常輕量級的對象封裝,是一個全自動的 ORM 框架。Hibernate 自動生成 SQL 語句,自動執行。

(1) 環境搭建

結構如下,ctl為控制層,service為服務層,dao為持久層。為了方便沒有按照標準的接口實現,我們只關注漏洞的部分。

1596704559.png!small

Beans下User.java對用為user_tbl表結構。

1596704752.png!small

我們使用/inject 接口,p為接受外部的參數,來查詢User的列表,使用fastjson來格化式輸出。

1596704781.png!small

我們回到dao層。

1)SQL注入

SQL注入我們使用字符串拼接方式:

1596704819.png!small

訪問http://localhost:8080/inject?p=m 直接用SQLMap跑一下:

1596704885.png!small

很容易就注入出數據來了。

2)HQL注入

HQL(Hibernate Query Language)是Hibernate專門用于查詢數據的語句,有別于SQL,HQL 更接近于面向對象的思維方式。表名就是對應我們上面的entity配置的。HQL注入利用比SQL注入利用難度大,比如一般程序員不會對系統表進行映射,那么通過系統表獲取屬性的幾乎不可能的,同時由于HQL對于復雜的語句支持比較差,對攻擊者來說需要花費更多時間去構造可用的payload,更多詳細的語法可以參考:

https://docs.huihoo.com/Hibernate/reference-v3_zh-cn/queryhql.html

1596704962.png!small

3)預編譯

我們使用setParameter的方式,也就是我們熟知的預編譯的方式。

  1. Query query = (Query) this.entityManager.createQuery("from User u where u.userName like :userName ",User.class);  
  2. query.setParameter("userName","%"+username+"%"); 

訪問http://localhost:8080/inject?p=m后得到正常結果。

1596705006.png!small

執行注入語句:

http://localhost:8080/inject?p=m’ or ‘1’ like ‘1 返回為空。

1596705041.png!small

我們來看看setParameter的方式到底對我們的SQL語句做了什么。我們將斷點打至Loader.class的bindPreparedStatement。發現通過預編譯后,SQL變為了:

  1. select user0_.id as id1_0_, user0_.password as password2_0_, user0_.username as username3_0_ from user_tbl user0_ where user0_.username like '%'' or ''1'' like ''1%', 

然后交給hikari處理。發現將我們的單引號變成了兩個單引號,也就是說把傳入的數據變為字符串。

1596705072.png!small

將斷點斷至mysql-connector-java(也就是我們熟知的JDBC驅動包)的ClientPreparedQueryBindings.setString.這里就是參數設置的地方。

1596705107.png!small

看一下算法:

  1. String parameterAsString = x
  2.  
  3.             boolean needsQuoted = true
  4.  
  5.             if (this.isLoadDataQuery || this.isEscapeNeededForString(x, stringLength)) { 
  6.  
  7.                 needsQuoted = false
  8.  
  9.                 StringBuilder buf = new StringBuilder((int)((double)x.length() * 1.1D)); 
  10.  
  11.                 buf.append('\''); 
  12.  
  13.                 for(int i = 0; i < stringLength; ++i) { 
  14.  
  15.                     char c = x.charAt(i); 
  16.  
  17.                     switch(c) { 
  18.  
  19.                     case '\u0000': 
  20.  
  21.                         buf.append('\\'); 
  22.  
  23.                         buf.append('0'); 
  24.  
  25.                         break; 
  26.  
  27.                     case '\n': 
  28.  
  29.                         buf.append('\\'); 
  30.  
  31.                         buf.append('n'); 
  32.  
  33.                         break; 
  34.  
  35.                     case '\r': 
  36.  
  37.                         buf.append('\\'); 
  38.  
  39.                         buf.append('r'); 
  40.  
  41.                         break; 
  42.  
  43.                     case '\u001a': 
  44.  
  45.                         buf.append('\\'); 
  46.  
  47.                         buf.append('Z'); 
  48.  
  49.                         break; 
  50.  
  51.                     case '"': 
  52.  
  53.                         if (this.session.getServerSession().useAnsiQuotedIdentifiers()) { 
  54.  
  55.                             buf.append('\\'); 
  56.  
  57.                         } 
  58.  
  59.                         buf.append('"'); 
  60.  
  61.                         break; 
  62.  
  63.                     case '\'': 
  64.  
  65.                         buf.append('\''); 
  66.  
  67.                         buf.append('\''); 
  68.  
  69.                         break; 
  70.  
  71.                     case '\\': 
  72.  
  73.                         buf.append('\\'); 
  74.  
  75.                         buf.append('\\'); 
  76.  
  77.                         break; 
  78.  
  79.                     case '¥': 
  80.  
  81.                     case '₩': 
  82.  
  83.                         if (this.charsetEncoder != null) { 
  84.  
  85.                             CharBuffer cbuf = CharBuffer.allocate(1); 
  86.  
  87.                             ByteBuffer bbuf = ByteBuffer.allocate(1); 
  88.  
  89.                             cbuf.put(c); 
  90.  
  91.                             cbuf.position(0); 
  92.  
  93.                             this.charsetEncoder.encode(cbuf, bbuf, true); 
  94.  
  95.                             if (bbuf.get(0) == 92) { 
  96.  
  97.                                 buf.append('\\'); 
  98.  
  99.                             } 
  100.  
  101.                         } 
  102.  
  103.                         buf.append(c); 
  104.  
  105.                         break; 
  106.  
  107.                     default: 
  108.  
  109.                         buf.append(c); 
  110.  
  111.                     } 
  112.  
  113.                 } 
  114.  
  115.                 buf.append('\''); 

可以看到mysql-connector-java主要是將將我們’轉為了’’,對于轉義的\會變為\\,比如對于這種SQL:

  1. SELECT user0_.id AS id1_0_,user0_. PASSWORD AS password2_0_,user0_.username AS username3_0_ 
  2.  
  3. FROM user_tbl user0_ WHERE user0_.username LIKE '%\' or username = 0x6d #%' 

也會變為:

  1. SELECT user0_.id AS id1_0_,user0_. PASSWORD AS password2_0_,user0_.username AS username3_0_ 
  2.  
  3. FROM user_tbl user0_ WHERE user0_.username LIKE '%\\'' or username = 0x6d #%' 

有人會說那我們使用select * from user_tbl where id = 1 and user() = 0x726f6f74406c6f63616c686f7374 這種類似的語句,全程沒有jdbc里面的危險字符是不是就可以繞過了?mysql-connector-java里面有個非常巧妙的點是,他會根據你傳入的類型判斷。比如傳入的為int類型。就會走setInt。傳入的為string就會走setString。所以這段語句還是會被select * from user_tbl where id = 1 ‘and user() = 0x726f6f74406c6f63616c686f7374’

我們看到SQL預編譯的算法也是非常簡單。

4. MyBatis

MyBatis是一流的持久性框架,支持自定義SQL,存儲過程和高級映射。MyBatis可以使用簡單的XML或注釋進行配置。現在目前國內大部分公司都是采用的MyBatis框架。

(1) 環境搭建:

下面為我們項目目錄結構:

1596705239.png!small

(2) 使用#{}的方式

#{}也就是我們熟知的預編譯方式。

1596705338.png!small

訪問http://localhost:8080/getList?p=m 后正常的返回:

1596705362.png!small

使用http://localhost:8080/getList?p=m' or ‘1’ like ‘1

結果返回為空。不存在注入。

我們將斷點斷在PreparedStatementLogger的invoke方法上面,其實這里就是一個代理方法。這里我們看到完整的SQL語句。

1596705390.png!small

同樣我們將斷點斷在:ClientPreparedQueryBindings.setString同樣會進去

1596705454.png!small

Hibernate和MyBatis的預編譯機制是一樣的。

(3) 使用${}的方式

${}的方式也就是MyBatis的字符串連接方式。

1596705500.png!small

使用SQLMap很容易就能跑出數據:

1596705527.png!small

(4) 關于OrderBy

之前有聽人說Order By后面的語句是不會參與預編譯?這句話是錯誤的。Order By也是會參與預編譯的。從我們上面的jdbc的setString算法可以看到,是因為setString會在參數的前后加上’’,變成字符串。導致Order By失去了原本的意義。只能說是預編譯方式的Order By不適用而已。所以對于這種Order By的防御的話建議是直接寫死在代碼里面。對于Order By方式的注入我們可以通過返回數據的順序的不同來獲取數據。

1596705578.png!small

(5) 關于useServerPrepStmts

其實在只有JDBC在開啟了useServerPrepStmts=true的情況下才算是真正的預編譯。但是如果是字符串的拼接方式,預編譯是沒有效果的。從MySQL的查詢日志就可以開看到。可以看到Prepare的語句。一樣是存在SQL注入的。

1596705612.png!small

我們使用占位符的方式:

1596705641.png!small

上面的語句就不存在SQL注入了。

我想這就是JDBC默認為啥不開啟useServerPrepStmts=true的原因吧。

5. 總結

在能使用預編譯的情況下我們應該要使用預編譯。在不能使用預編譯的情況下,可以對特定類型做規范,比如傳數字的需要規范為Integer,Long等。這樣會在進入數據庫前會提前拋出異常。或者使用Spring的AOP機制,添加一個前置的fitler,對有害的字符清洗或者過濾。但是這樣有點籠統,會對全局參數進行清洗。

還有一種比較好的方式是,通過注解的方式,這樣會比較方便,可復用性也很好。對不能進行預編譯的參數加上過濾有害字符的注解。我們就不在這里做代碼的實現,網上有很多可以參考的教程。可以使用Apache Jakarta Commons提供的很多方便的方法來過濾有害字符。

 

責任編輯:趙寧寧 來源: FreeBuf
相關推薦

2021-05-08 15:22:31

網絡安全Web安全SQL

2013-07-27 14:14:25

2009-02-12 10:14:16

2020-12-16 13:22:37

Web安全SQL工具

2013-11-12 16:38:22

2012-11-14 17:18:58

2010-12-20 16:04:30

2010-04-12 08:59:00

2013-04-19 10:56:54

2023-08-01 08:00:00

SQLWeb應用安全

2009-03-14 16:50:38

網站安全meter程序

2014-04-14 10:03:16

2017-08-10 10:23:59

2019-09-17 10:06:46

數據庫程序員網絡安全

2012-11-08 17:02:58

2016-05-18 09:52:20

2013-01-11 16:31:27

2020-08-19 15:30:04

PHP網絡安全代碼

2010-04-13 14:35:17

2011-12-30 11:04:14

點贊
收藏

51CTO技術棧公眾號

中文日本在线观看| 日本黄色网址大全| 亚洲七七久久综合桃花剧情介绍| 国产一区二区精品久久99| 久久影院在线观看| 一级全黄裸体片| 成人免费看视频网站| 欧美国产丝袜视频| av色综合网| 亚洲av无码不卡| 午夜久久久久| 亚洲一区二区精品| 日韩女优在线视频| 成人在线免费电影网站| 亚洲无线码一区二区三区| 欧美婷婷久久| 国产黄色片av| 日本美女一区二区三区| 欧美激情欧美狂野欧美精品| 日韩精品电影一区二区| 久久伊人精品| 欧美亚洲国产一区在线观看网站| 国产麻豆电影在线观看| 无码h黄肉3d动漫在线观看| 蜜桃精品视频在线| 38少妇精品导航| 在线观看美女av| 蜜桃a∨噜噜一区二区三区| 91精品一区二区三区久久久久久| 99色精品视频| 91av久久| 亚洲主播在线播放| 日本久久高清视频| 成年在线观看免费人视频| 91在线小视频| 亚洲一区二区三区成人在线视频精品| 欧美性猛交xxxx乱大交hd | 中文字幕日韩在线观看| 日本黄色动态图| 中文字幕一区二区三区中文字幕| 欧美日韩精品一二三区| 欧美 国产 小说 另类| sm性调教片在线观看 | 亚洲深夜福利视频| 中文在线字幕观看| 久久久精品区| 日韩三级中文字幕| 免费黄频在线观看| 亚洲精品伦理| 欧美老人xxxx18| 最新天堂中文在线| 国产a亚洲精品| 国产在线精品一区二区不卡了| 一级精品视频在线观看宜春院 | 日本55丰满熟妇厨房伦| 台湾天天综合人成在线| 欧美日韩中文一区| 伊人网在线综合| 欧美成人三级| 91精品国产高清一区二区三区蜜臀| 一级片视频免费观看| 成人自拍视频网| 欧美日韩小视频| 99精品999| 免费看一区二区三区| 日韩欧美在线影院| 日本美女视频网站| 欧美wwwsss9999| 亚洲精品资源在线| a级片在线观看| 久久国产亚洲精品| 另类视频在线观看| 国产极品美女高潮无套嗷嗷叫酒店| 影音先锋国产精品| 91成人天堂久久成人| 无码日韩精品一区二区| 老司机精品视频一区二区三区| 国产精品综合不卡av| 国产裸体永久免费无遮挡| 国产精品1区2区3区| 国产伦精品一区二区三区视频孕妇| 色网站免费观看| 国产欧美日韩激情| 国产欧美综合一区| 欧美xxxx性xxxxx高清| 亚洲高清不卡在线| 日韩精品无码一区二区三区免费 | 欧洲一区二区三区免费视频| 欧美成人福利在线观看| 日韩高清在线观看一区二区| 亚洲精品国产电影| 天堂资源在线视频| 亚洲视频日本| 国产成人精品一区二区在线| 国产精品怡红院| av动漫一区二区| 亚洲乱码一区二区三区| 欧美videosex性欧美黑吊| 欧美日韩激情小视频| 成年网站在线播放| 国产毛片久久久| 色噜噜狠狠狠综合曰曰曰| 久久这里只有精品国产| 日韩**一区毛片| 国产精品制服诱惑| 在线免费看黄| 欧美日韩一区二区免费视频| 亚洲a级黄色片| 在线日韩一区| 九九热精品视频| 亚洲中文无码av在线| 成人h动漫精品一区二| 亚洲一区二区三区精品视频| 欧洲一区精品| 日韩一区国产二区欧美三区| 精品人妻中文无码av在线| 欧美日韩国产免费观看| 国产精品一二三在线| 五月天激情婷婷| 一区二区在线观看av| 一区二区三区国产免费| 色狼人综合干| 欧美日韩国产123| 91久久久久国产一区二区| 久久蜜桃香蕉精品一区二区三区| 成人免费a级片| 91成人短视频在线观看| 中文欧美日本在线资源| 中文字幕精品无码一区二区| 成人免费视频免费观看| 日韩第一页在线观看| 日本综合久久| 亚洲欧美一区二区三区情侣bbw| 欧美激情国产精品免费| 狠狠色丁香久久婷婷综合_中| 日韩一区二区电影在线观看| 手机在线理论片| 亚洲激情小视频| 国产无码精品一区二区| 国产91精品一区二区| 婷婷视频在线播放| 色噜噜成人av在线| 色悠悠国产精品| 中文字幕在线播放不卡| 久久精品在线观看| 精品少妇一区二区三区在线| 亚洲一区二区三区四区电影| 欧美成人h版在线观看| 国产精品爽爽久久| 亚洲视频在线一区观看| 免费成年人高清视频| 99久久www免费| 国产日韩欧美在线观看| 日本www在线观看| 91精品国产手机| 九九视频免费看| 成人午夜精品一区二区三区| 很污的网站在线观看| 极品国产人妖chinesets亚洲人妖| 色综合五月天导航| 狠狠躁夜夜躁av无码中文幕| 亚洲.国产.中文慕字在线| 四季av综合网站| 午夜一级久久| 性欧美精品一区二区三区在线播放 | 丁香六月色婷婷| 亚洲狠狠爱一区二区三区| 亚洲精品激情视频| 国产精品主播| 日韩影片在线播放| 日韩精品中文字幕吗一区二区| 欧美大片欧美激情性色a∨久久| 性一交一乱一精一晶| 亚洲成人综合网站| b站大片免费直播| 免费高清视频精品| 黄色一级片av| 欧美影院天天5g天天爽| 国产福利成人在线| caopeng在线| 日韩av网站导航| 性色av一区二区三区四区| 亚洲欧洲三级电影| 亚洲成a人片在线www| 国产深夜精品| 一本一道久久a久久精品综合| 日韩不卡在线视频| 日韩免费观看网站| 4438x成人网全国最大| 精品性高朝久久久久久久| 久久精品国产亚洲av麻豆蜜芽| 亚洲视频免费观看| yy1111111| 久久电影网电视剧免费观看| 777av视频| 欧美3p在线观看| 精品国产一区二区三区四区vr| 日本综合视频| 久久久久久网址| av色图一区| 日韩成人免费视频| 国产ts变态重口人妖hd| 欧美性生活大片免费观看网址| 日本 欧美 国产| 91首页免费视频| 三级av免费看| 视频一区二区国产| 亚洲精品无码国产| 欧美高清视频手机在在线| 激情小说综合区| 人人爱人人干婷婷丁香亚洲| 国产va免费精品高清在线观看| 影音先锋在线视频| 在线精品视频视频中文字幕| 三级在线观看网站| 欧美一区二区三区四区高清| 波多野结衣二区三区| 亚洲va韩国va欧美va| 中文字幕在线观看2018| 日本一区二区三区免费乱视频| 黄色在线免费播放| 国产麻豆精品theporn| 一区二区在线播放视频| 亚洲欧美日韩国产一区| www.在线观看av| 婷婷综合在线| 亚洲午夜精品一区二区| 国产伦精品一区二区三区千人斩| 国产日韩精品久久| 91九色鹿精品国产综合久久香蕉| 成人激情黄色网| 国产成人免费精品| 欧美亚洲国产精品| 成人福利电影| 久久久久久久久久久免费| 亚洲精品天堂| 久久综合五月天| 国产视频中文字幕在线观看| 色哟哟网站入口亚洲精品| 91在线视频| 色哟哟亚洲精品一区二区| 北岛玲日韩精品一区二区三区| 亚洲老板91色精品久久| 欧美日韩在线精品一区二区三区激情综 | 欧美专区在线观看| 中文av在线全新| 欧洲一区二区视频| 暖暖成人免费视频| 国产精品99久久久久久久久| 三上悠亚激情av一区二区三区| 日本成人激情视频| 日韩电影网站| 国产精品视频一| 精品欧美日韩精品| 成人黄色在线免费| 精品国产鲁一鲁****| 亚洲最大福利视频网站| 18国产精品| 国产在线欧美日韩| 校花撩起jk露出白色内裤国产精品 | 神马久久影院| 欧美日韩一区在线观看视频| 国产伦精品一区二区三区视频| 亚洲精品乱码久久久久久蜜桃91 | 免费日韩一区二区三区| 久久精品一二三区| 国产精品一国产精品| 水蜜桃亚洲精品| 亚洲h色精品| 男人天堂av片| 久久久久久久波多野高潮日日| 久久午夜夜伦鲁鲁一区二区| 久久99久久久欧美国产| 91精品国产高清91久久久久久 | 久久精品二区亚洲w码| 日本高清一区二区视频| 国产99精品国产| 少妇精品一区二区| 国产精品乱人伦| 久久亚洲国产成人精品性色| 一本久久综合亚洲鲁鲁五月天| 中文字幕+乱码+中文| 日韩欧美国产麻豆| 青青草免费在线视频| 日韩中文av在线| 成人性生交大片免费看网站| 国产97在线视频| 久久视频免费| 欧美三日本三级少妇三99| 亚洲老妇激情| 久久久久久久久久福利| 国产精品原创巨作av| 在线观看福利片| 自拍偷拍欧美精品| 免费黄色网址在线| 日韩一区二区三区av| 欧美老女人性开放| 欧美成人午夜剧场免费观看| 欧美极品videos大乳护士| 成人激情免费在线| 亚洲涩涩av| 蜜桃视频一区二区在线观看| 视频在线观看91| 三级黄色片免费看| 国产日产欧产精品推荐色| 国产一级生活片| 在线播放欧美女士性生活| 亚洲女人18毛片水真多| 尤物yw午夜国产精品视频| 国产网红在线观看| 成人在线观看视频网站| 一本色道久久综合狠狠躁的番外| 久久久99精品视频| 蜜臀91精品一区二区三区| 菠萝菠萝蜜网站| 亚洲综合一二区| 国产视频手机在线| 在线亚洲国产精品网| 亚洲一级少妇| 国产精品视频免费一区| 91精品推荐| 国产原创精品在线| 国产亚洲欧美激情| 毛片视频网站在线观看| 日韩精品一区二区三区在线播放| 在线观看国产原创自拍视频| 热久久免费视频精品| 欧美理伦片在线播放| 18禁裸男晨勃露j毛免费观看| 韩国v欧美v日本v亚洲v| 青青草自拍偷拍| 欧美伊人精品成人久久综合97| 五月天丁香视频| 69精品小视频| 女同另类激情重口| 野外做受又硬又粗又大视频√| 国产一区二区精品久久99| 日本黄色免费片| 欧美精品乱码久久久久久| 国产福利电影在线| 国产精品福利在线观看| 久久不见久久见国语| 国产真实乱子伦| 久久综合久久鬼色| 日日摸天天添天天添破| 日韩福利在线播放| 中文在线а√在线8| 乱一区二区三区在线播放| 香蕉精品999视频一区二区 | 一本久道中文无码字幕av| 久久久不卡网国产精品一区| 日韩特级黄色片| 亚洲欧美一区二区三区久久| 女生影院久久| 一区二区三区在线视频看| 久久国产精品99精品国产| 我要看黄色一级片| 日韩免费性生活视频播放| 好久没做在线观看| 国外成人在线视频网站| 国产亚洲毛片| 欧美波霸videosex极品| 欧美日韩午夜精品| 91麻豆国产福利在线观看宅福利| 不卡的av一区| 国产精品日韩久久久| 亚洲精品国产一区黑色丝袜| 欧美日韩精品综合在线| www红色一片_亚洲成a人片在线观看_| 91久久精品www人人做人人爽| 亚洲人体偷拍| 少妇精品无码一区二区免费视频| 欧美日韩一区二区三区免费看| а√天堂8资源在线官网| 国产99午夜精品一区二区三区 | 欧美中文在线| 国产精品视频色| 欧美日韩精品| jizz日本免费| 欧美三级中文字| 日韩精品亚洲人成在线观看| 久久99九九| 另类人妖一区二区av| 精品无码久久久久| 亚洲午夜精品久久久久久性色| 亚洲精品成a人ⅴ香蕉片| 97超碰人人澡| 国产精品久久久久久妇女6080| 性一交一乱一伧老太| 国产精品扒开腿做爽爽爽视频 | 中文幕一区二区三区久久蜜桃| 97人妻精品一区二区三区| 午夜精品在线视频| 日韩国产综合| 丰满岳乱妇一区二区| 欧美日韩情趣电影| 少妇视频一区| av磁力番号网| 国产女人水真多18毛片18精品视频 | a级片在线免费观看|