.NET MVC CSRF/XSRF 漏洞

来源:https://www.cnblogs.com/caligula/archive/2018/08/17/9492675.html
-Advertisement-
Play Games

最近我跟一個漏洞還有一群阿三幹起來了…… 背景: 我的客戶是一個世界知名的藥企,最近這個客戶上臺了一位阿三管理者,這個貨上線第一個事兒就是要把現有的軟體供應商重新洗牌一遍。由於我們的客戶關係維護的非常好,直接對口人提前透露給我們這個管理者就是想讓一個阿三公司壟斷他們的軟體供應,並且表示了非常鄙視。我 ...


最近我跟一個漏洞還有一群阿三幹起來了……

背景:

我的客戶是一個世界知名的藥企,最近這個客戶上臺了一位阿三管理者,這個貨上線第一個事兒就是要把現有的軟體供應商重新洗牌一遍。由於我們的客戶關係維護的非常好,直接對口人提前透露給我們這個管理者就是想讓一個阿三公司壟斷他們的軟體供應,並且表示了非常鄙視。我們表示了理解,畢竟任意一家公司只要進去一個阿三,慢慢的。。。慢慢的。。。就變成滿屋都是阿三。。。

然後某一家阿三公司就暗地裡中標了,然後我們就面臨KT。由於我們維護著12個高活躍系統,所以KT的工作量也是非常的大。

BUT! 阿三的牛逼之處就在這時候體現出來了,他會從各個維度找你的事兒,其中一個就是找漏洞(自己找了一家阿三的漏洞檢測公司免費做)報給客戶並威脅說解決不完不接手,用以拉長KT的周期(本來KT只有三周時間)。

然後客戶的阿三頭頭就同意了。。。

這個漏洞本來就有,客戶一直表示不想處理,因為大多數網站太老舊了,很多都不是我們一手開發的。

但是這回看來是不幹不行了,還好客戶表示會付費,行吧。。。 那就整

 

現在有請漏洞登場!

大家好!我叫CSRF,全名是 Cross-Site Request Forgery (CSRF) Prevention Cheat Sheet

這是我的簡歷:https://www.owasp.org/index.php/Cross-Site_Request_Forgery_(CSRF)_Prevention_Cheat_Sheet#Viewstate_.28ASP.NET.29

(Google Translate 瞭解一下)

這個玩意說白了就是一個偽裝攻擊,偽裝工具是Cookie。

這個玩意是這樣運作的:

(請不要在意這個醜逼的圖。。。)

 

簡單描述就是

其他網站用你的身份(Cookie)假裝是你幹了你不知道的事兒,這時候請想想你在網上銀行轉賬的時候

那麼這裡面就出現一個重大的疑點:

為啥WebSiteB發過來的請求WebSiteA會收到呢? IIS吃了髒東西不管事兒了?

因為我們的網站支持跨域請求!(是不是看著賊扎眼!畫重點了啊)

現在毛病基本OK了,剩的就是出方案。

對與CSRF這個東西知名度還是很高的,網上一搜一大把

.NET MVC就自帶瞭解決方案,此方案只針對常規的MVC項目,前後端分離的繞行,以後我要是解決了我再回來寫。。。

解決方案也很粗暴,一句話來說就是:

我們的伺服器只接收來自我們自己頁面發過來的請求

放到實現上就是:每個頁面都按照一定規則生成一個Token,然後再發請求的時候帶過去,伺服器先看Token再乾別的

這時候有人說了:要是別的網站偽造Token怎麼辦?

有道是孔子曰:不怕賊偷就怕賊惦記,他要是就想搞你,你早晚是防不住的啊,兄die

下麵介紹關鍵代碼:

@Html.AntiForgeryToken()

這個是cshtml的頁面的代碼,aspx的差不多

這東西的作用是會在頁面上生成一個 Hidden,Value就是Token

最後變成Html長介樣兒:

<input name="__RequestVerificationToken" 
type="hidden" value="MbnNdB3T64quXYviXLsvoi_FlbM2SihwiiPCgSzaWAL0duMy7H6SbuF0lkUAxOD-DwF4P_4kxlyravohGXsQ_ERVPm5f3Oa3owG6LZ26WRw1" />

 那一球亂糟糟的就是Token

那麼這玩意怎麼用呢?

Type 1,Form Request:

@using (Html.BeginForm("Action", "Controller", null, FormMethod.Post, new { id = "formId" }))
{
    @Html.AntiForgeryToken();
    Other Code......
}

  

Type 2,Ajax Request:

var token = $('@Html.AntiForgeryToken()').val();
var headers = {};
headers["__RequestVerificationToken"] = token;

            $.ajax({
                type: "post",
                headers: headers,
                url: "@Url.Action("Action","Controller")",
                data: { },
                dataType: "json",
                success: function (response) {
                   
                }
            });

  

說到底就是頁面上生成了Token之後,想盡一切辦法發到後臺去,不拘泥與形式

form就是直接包到裡面了,後臺直接用name拿就ok了,Ajax是放在header里了。

接下來就是後臺驗證,由於絕大多數Action都需要堵這個漏洞,所以直接寫了一個Filter

    using System.Net;
    using System.Web.Helpers;
    using System.Web.Mvc;

    public class ExtendedValidateAntiForgeryToken : AuthorizeAttribute
    {
        public override void OnAuthorization(AuthorizationContext filterContext)
        {
            var request = filterContext.HttpContext.Request;
            if (request.HttpMethod != WebRequestMethods.Http.Post) return;
            if (request.IsAjaxRequest())
            {
                var antiForgeryCookie = request.Cookies[AntiForgeryConfig.CookieName];
                var cookieValue = antiForgeryCookie != null ? antiForgeryCookie.Value : null;
                //從cookies 和 Headers 中 驗證防偽標記  
                //這裡可以加try-catch  
                //try
                //{
                AntiForgery.Validate(cookieValue, request.Headers["__RequestVerificationToken"]);
                //}
                //catch (Exception e)
                //{
                //    //filterContext.Result = new RedirectResult("/Account/Login?returnUrl=" +
                //    // HttpUtility.UrlEncode(filterContext.HttpContext.Request.Url.ToString()));
                //    ContentResult result = new ContentResult();
                //    result.Content = "<div style='text-align:center;padding:1em;' >當前已經處於退出狀態,請重新登錄</div>";
                //    filterContext.Result = result;
                //}
            }
            else
            {
                //try
                //{
                new ValidateAntiForgeryTokenAttribute().OnAuthorization(filterContext);
                //}
                //catch (Exception ex)
                //{
                //    //
                //}
            }
        }
    }

  裡面代碼核心就是驗證Token的有效性,用的是官方API方法,但是要區別一個事兒,就是前文提到了咱們Ajax和Form帶Token的方式不一樣,所以需要判斷是不是AJAX Request,走兩個分支。

然後就是把Filter掛到Action上就行了。

好了,漏洞堵上了,用時2天,客戶賊開心,正在準備去找阿三乾仗的時候出岔子了。

細心的老鐵可能發現了,上面的解決方案都是POST請求啊,GET呢?

這個就是個事兒了,從網上調查的時候得知,這個CSRF全是針對POST的,壓根就不管GET。

比如這個文章:

https://stackoverflow.com/questions/35473856/asp-net-mvc-csrf-on-a-get-request

阿三哪個什麼漏洞檢測公司發回來一堆GET的URL。。。

在跟客戶說明原委之後,客戶炸了。。。 要乾阿三,然後就發了一系列言辭犀利的郵件,也CC了他們哪個阿三頭頭

最後阿三們看有點失控,一個是我們POST改的太快了(47處),第二個是,沒想到客戶的IT急眼了。。。

這時的阿三很尷尬,在郵件里回:我們有很豐富的修改漏洞的經驗

WTF?!!

 

還沒等我們說話,客戶直接回了一句:好!我現在約一個會,你們說說GET請求是怎麼回事兒

行了。。。 我去幫客戶乾仗了。。。

想想跟印度人、南韓人、澳大利亞人加上我一個中國人開英語的會我就腦仁兒疼。。。。。。

 

另外附加一個連接:

https://weblogs.asp.net/dixin/anti-forgery-request-recipes-for-asp-net-mvc-and-ajax

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本文使用的DEV版本為10.1版本 首先需要添加Dll引用 DevExpress.Data.v10.1 DevExpress.XtraPrinting.v10.1 DevExpress.XtraReports.v10.1 在winfrom的環境下 XtraReport 是可以直接調用 Print() ...
  • default 關鍵字有兩類用法 1. switch語句中指定預設標簽 2. 預設值表達式 switch 語句 預設值表達式 default對應各種類型生成預設值列表如下: |類型|預設值| | | | |任何引用類型| | |數值類型|零| |bool| | |enum|表達式 生成的值,其中 是 ...
  • 解決方法:在執行的任務方法前加上Mutex特性即可,如果作業未完成,新作業開啟的話,新作業會放入計劃中的作業隊列中,直到前面的作業完成。 必須使用Hangfire.Pro.Redis 和 Hangfire.SqlServer 作為資料庫。 參考:https://github.com/Hangfire ...
  • 本節對事件進行總結。 二、事件: 1、概念:Event:A member that enables an object or class to provide notifications;官方的解釋是這樣,就是說在C#中,事件是使 對象或者類具備通知能力的成員。比如說手機收到簡訊提醒我去開會,那麼手 ...
  • 紙殼CMS(ZKEACMS)里的Grid是一個TagHelper,是對jQuery插件datatables的一個配置封裝。 grid的使用很簡單,設置Model,使用 ...
  • @{ ViewBag.Title = "UsersList"; } @{ Html.RenderAction("SearchRole", "Shared"); } 賬號(編號或姓名): ... ...
  • 問題:考慮屏幕大小,一般都是會在表單問卷的頁面使用ScrollViewer。問卷中問題漏填漏選時,在提交時校驗不過,需要滾動跳轉至漏填漏選項。 頁面如下: 每個選項使用StackPanel,並對覆選框和單選的勾選事件進行答案記錄,使用全局變數記錄 將所有選項使用StackPanel指定垂直樣式,思路 ...
  • 索引 NET Core應用框架之BitAdminCore框架應用篇系列 框架演示:https://www.bitadmincore.com 框架源碼:https://github.com/chenyinxin/cookiecutter-bitadmin-core 什麼是.NET Core Globa ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...