Owin Katana 的底層源碼分析

来源:https://www.cnblogs.com/neilhu/archive/2020/05/11/12802038.html
-Advertisement-
Play Games

最近看了一下開源項目asp.net katana,感覺公開的介面非常的簡潔優雅,channel 9 說是受到node.js的啟發設計的,Katana是一個比較老的項目,現在已經整合到asp.net core中。 從github克隆下來的項目,這個博客專門是從代碼角度去理解katana項目,所以本篇隨 ...


最近看了一下開源項目asp.net katana,感覺公開的介面非常的簡潔優雅,channel 9 說是受到node.js的啟發設計的,Katana是一個比較老的項目,現在已經整合到asp.net core中。

github克隆下來的項目,這個博客專門是從代碼角度去理解katana項目,所以本篇隨筆針對已經對OWIN有所瞭解的人,如果只是入門的話可以跑一下MSDN的源碼再來閱讀本篇文章。

代碼結構如上,簡單分析一下各個文件夾的含義,這對於理解katana項目的整體結構有一個大的輪廓。

  .build文件夾顧名思義就是編譯的文件夾,在沒使用vs的時候你可以單擊build.cmd 去編譯這個項目,十分的方便。

  .Nuget就是包管理工具的配置文件,這個我們可以忽略。同理.Prerelease。

  Development是本次的研究重點,當你打開這個文件夾的時候你會發現一個類庫Microsoft.Owin的類庫,這個是OWIN組件的經典實現。

  FunctionTests是單元測試的類庫

  Hosting 是server的抽象層,OWIN 將伺服器進行抽象化,Hosting 就是能夠管理Server的一層,像WebApp就能開啟一個httplister服務,詳細稍後再講。

  Middleware是一些中間件的實現,在katana已經將管道模型虛擬化成中間件

  Performance 和Sandbox 是微軟的一些測試工具

  Security 是微軟已經寫好的驗證中間件,其中包括JWT和Oauth的驗證方式

  Server 就是伺服器的實現

  Owin.Analysis是我本人建的web程式用來debug

上面已經介紹了各個文件夾所對應的功能,相必大部分人都是一臉矇蔽,但是不用擔心,下麵就來看看具體的代碼,當然是從最小的例子出發。點擊 getting started with owin and katan 你就能跳到MSDN得到最小的例子。裡面的一系列操作就為了添加下麵的一個類和幾個reference.現在我們看一下這個類。

 

 1 using Microsoft.Owin;
 2 
 3 [assembly: OwinStartup(typeof(Owin.Analysis.Startup))]
 4 namespace Owin.Analysis
 5 {
 6     public class Startup
 7     {
 8         public void Configuration(IAppBuilder app)
 9         {
10             app.Run(context =>
11             {
12                 context.Response.ContentType = "text/plain";
13                 return context.Response.WriteAsync("Hello World");
14             });
15         }
16     }
17 }

看起來這個代碼十分的優雅,添加幾個reference和一個類就讓請求到達Hello World。我們先分析這個類,首先程式集特性OwinStartupAtribute將當前類保存在元數據中。然後寫了一個Configuration方法,獲取一個IAppBuilder 參數調用Run方法,Run方法傳遞一個委托進去,我們的處理邏輯就在這一個委托里。

這裡面我們分析一下核心介面IAppBuilder的經典實現者AppBuilder,IAppBuilder的介面如下,

using System;
using System.Collections.Generic;

namespace Owin
{
    public interface IAppBuilder
    {
        IDictionary<string, object> Properties { get; }//請求的參數

        object Build(Type returnType);//中間件鏈接
        IAppBuilder New();//創建一個新的對象
        IAppBuilder Use(object middleware, params object[] args);//註冊中間件
    }
}

好的我們來分析一下AppBuilder中間件的註冊實現。在app.Run 打完break point你就可以進入app.use方法,首先在AppBuilderUseExtensions這個類里對use的入口寫了一大堆擴展方法。app.Run就是其中的一個,當你用app.Run註冊中間件的時候是沒有下一個中間件的引用的。

        public static void Run(this IAppBuilder app, Func<IOwinContext, Task> handler)
        {
            if (app == null)
            {
                throw new ArgumentNullException("app");
            }
            if (handler == null)
            {
                throw new ArgumentNullException("handler");
            }

            app.Use<UseHandlerMiddleware>(handler);
        }

在經典的實現中,參數middleware會有兩種情況,一種是delegate,一種是type,如果是type類型,則他的構造方法接受next為參數,並且裡面有一個公開的Invoke方法。如果是委托,當前委托作為參數傳遞到next中。

 public IAppBuilder Use(object middleware, params object[] args)
        {
            _middleware.Add(ToMiddlewareFactory(middleware, args));
            return this;
        }

 

下麵的代碼是ToMiddlewareFactory的實現,在第9行和第27行分別判斷了中間件對象是委托類型還是type類型,由於本題例子是type對象,我們分析一下ToConstructorMiddlewareFactory方法。

 1 private static Tuple<Type, Delegate, object[]> ToMiddlewareFactory(object middlewareObject, object[] args)
 2         {
 3             if (middlewareObject == null)
 4             {
 5                 throw new ArgumentNullException("middlewareObject");
 6             }
 7 
 8             var middlewareDelegate = middlewareObject as Delegate;
 9             if (middlewareDelegate != null)
10             {
11                 return Tuple.Create(GetParameterType(middlewareDelegate), middlewareDelegate, args);
12             }
13 
14             Tuple<Type, Delegate, object[]> factory = ToInstanceMiddlewareFactory(middlewareObject, args);
15             if (factory != null)
16             {
17                 return factory;
18             }
19 
20             factory = ToGeneratorMiddlewareFactory(middlewareObject, args);
21             if (factory != null)
22             {
23                 return factory;
24             }
25 
26             if (middlewareObject is Type)
27             {
28                 return ToConstructorMiddlewareFactory(middlewareObject, args, ref middlewareDelegate);
29             }
30 
31             throw new NotSupportedException(string.Format(CultureInfo.CurrentCulture,
32                 Resources.Exception_MiddlewareNotSupported, middlewareObject.GetType().FullName));
33         }

 

 

在第5行獲取type類型的所有構造方法,第8行獲取構造方法的所有參數,在第14行有個trick,用zip方法判斷參數類型是否和構造方法類型是一致的。如果是一致的則繼續往下走,在第22行和第23行利用委托將構造方法創建成lambda表達式,然後生成元祖,第一個是next的類型,第二個是type的構造方法,第三個是type構造方法所需的參數。然後將元祖加入到AppBuild所維護的中間件對象。

 1  
 2         private static Tuple<Type, Delegate, object[]> ToConstructorMiddlewareFactory(object middlewareObject, object[] args, ref Delegate middlewareDelegate)
 3         {
 4             var middlewareType = middlewareObject as Type;
 5             ConstructorInfo[] constructors = middlewareType.GetConstructors();
 6             foreach (var constructor in constructors)
 7             {
 8                 ParameterInfo[] parameters = constructor.GetParameters();
 9                 Type[] parameterTypes = parameters.Select(p => p.ParameterType).ToArray();
10                 if (parameterTypes.Length != args.Length + 1)
11                 {
12                     continue;
13                 }
14                 if (!parameterTypes
15                     .Skip(1)
16                     .Zip(args, TestArgForParameter)
17                     .All(x => x))
18                 {
19                     continue;
20                 }
21 
22                 ParameterExpression[] parameterExpressions = parameters.Select(p => Expression.Parameter(p.ParameterType, p.Name)).ToArray();
23                 NewExpression callConstructor = Expression.New(constructor, parameterExpressions);
24                 middlewareDelegate = Expression.Lambda(callConstructor, parameterExpressions).Compile();
25                 return Tuple.Create(parameters[0].ParameterType, middlewareDelegate, args);
26             }
27 
28             throw new MissingMethodException(string.Format(CultureInfo.CurrentCulture,
29                 Resources.Exception_NoConstructorFound, middlewareType.FullName, args.Length + 1));
30         }

 

這個時候我們已經將中間件註冊到AppBuilder對象了。註冊完中間件的對象我們還需要做一件事就是將這些中間件chained together,這些實現就是Build 方法中,而Build方法BuildInternal方法,這個時候會產生一個entry point供調用。

現在我們重點看一下這個build方法。

    public object Build(Type returnType)
        {
            return BuildInternal(returnType);
        }

Build方法調用私有的BuildInternal的方法。

private object BuildInternal(Type signature)
        {
            object app;
            if (!_properties.TryGetValue(Constants.BuilderDefaultApp, out app))
            {
                app = NotFound;
            }

            foreach (var middleware in _middleware.Reverse())
          {
                Type neededSignature = middleware.Item1;
                Delegate middlewareDelegate = middleware.Item2;
                object[] middlewareArgs = middleware.Item3;

                app = Convert(neededSignature, app);
                object[] invokeParameters = new[] { app }.Concat(middlewareArgs).ToArray();
                app = middlewareDelegate.DynamicInvoke(invokeParameters);
                app = Convert(neededSignature, app);
            }

            return Convert(signature, app);
        }

我們可以看到它是怎樣將中間件chained together的,在我們之前註冊的時候實際上middleware元祖會保存三個信息,第一個type就是構造函數的第一個類型,第二個委托是useHandlerMiddleware的構造方法,第三個是構造方法的參數(除了第一個),Reverse的方法會將中間件逆序,這樣保證調用的順序就是你註冊的順序,後面的是chain的邏輯,app的變數實際上就是下一個中間件構造函數的next,當得到第一個中間件的時候,裡面的next會保存第二個中間件的處理邏輯,同樣第二個next就是第三個...,這樣chained together得到的就是第一個中間件的邏輯,所以你們在用app.Use的方法就會有一個參數next,並且需要手動調用一下。

得到這些之後需要的就是要將中間件註冊到application管道事件呢。因為asp.net的是一個大的切麵框架。

 public void Initialize(HttpApplication application)
        {
            for (IntegratedPipelineBlueprintStage stage = _blueprint.FirstStage; stage != null; stage = stage.NextStage)
            {
                var segment = new IntegratedPipelineContextStage(this, stage);
                switch (stage.Name)
                {
                    case Constants.StageAuthenticate:
                        application.AddOnAuthenticateRequestAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StagePostAuthenticate:
                        application.AddOnPostAuthenticateRequestAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StageAuthorize:
                        application.AddOnAuthorizeRequestAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StagePostAuthorize:
                        application.AddOnPostAuthorizeRequestAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StageResolveCache:
                        application.AddOnResolveRequestCacheAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StagePostResolveCache:
                        application.AddOnPostResolveRequestCacheAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StageMapHandler:
                        application.AddOnMapRequestHandlerAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StagePostMapHandler:
                        application.AddOnPostMapRequestHandlerAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StageAcquireState:
                        application.AddOnAcquireRequestStateAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StagePostAcquireState:
                        application.AddOnPostAcquireRequestStateAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    case Constants.StagePreHandlerExecute:
                        application.AddOnPreRequestHandlerExecuteAsync(segment.BeginEvent, segment.EndEvent);
                        break;
                    default:
                        throw new NotSupportedException(
                            string.Format(CultureInfo.InvariantCulture, Resources.Exception_UnsupportedPipelineStage, stage.Name));
                }
            }
            // application.PreSendRequestHeaders += PreSendRequestHeaders; // Null refs for async un-buffered requests with bodies.
            application.AddOnEndRequestAsync(BeginFinalWork, EndFinalWork);
        }

這裡面有一個概念就是IntegratedPipelineBlueprintStage,這個是一個鏈表結構,每個對應的就是管道事件,每個stage都有entry point,這樣方便我們在不同的管道事件中運行中間件,在BeginEvent里我們得到stage 的entry point,然後非同步調用得到結果。entry point 就是我們上例build得到的結果。

  private async Task RunApp(AppFunc entryPoint, IDictionary<string, object> environment, TaskCompletionSource<object> tcs, StageAsyncResult result)
        {
            try
            {
                await entryPoint(environment);
                tcs.TrySetResult(null);
                result.TryComplete();
            }
            catch (Exception ex)
            {
                // Flow the exception back through the OWIN pipeline.
                tcs.TrySetException(ex);
                result.TryComplete();
            }
        }

然後我們分析一下怎麼在不同的管道中註冊事件。在MSDN的文檔描述的。

app.UseStageMarker(PipelineStage.Authenticate)

這個api會創建一個IntegratedPipelineBlueprintStage,上文說這是一個鏈表結構,之間用next屬性連接。在不同的stage中會有entry point,然後在上面的例子中註冊到不同的管道中去調用。下圖是api的代碼。

  public static IAppBuilder UseStageMarker(this IAppBuilder app, string stageName)
        {
            if (app == null)
            {
                throw new ArgumentNullException("app");
            }

            object obj;
            if (app.Properties.TryGetValue(IntegratedPipelineStageMarker, out obj))
            {
                var addMarker = (Action<IAppBuilder, string>)obj;
                addMarker(app, stageName);
            }
            return app;
        }

好的,到這裡了,謝謝大家閱讀,如果有任何不理解的歡迎交流:)


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

-Advertisement-
Play Games
更多相關文章
  • 一個小伙伴最近參加某一線互聯網公司的面試,被問到了一些併發相關的問題,看看大家能否答出來: (1)synchronized的CPU原語級別是如何實現的? (2)無鎖、偏向鎖、輕量級鎖、重量級鎖有什麼差別,升級過程如何? (3)線程間通信,同機器進程間通信,跨機器進程間通信,各有什麼方法? (4)下列 ...
  • 先扯些這些內容: TCP/IP TCP/IP是個協議組,可分為三個層次:網路層、傳輸層和應用層。 在網路層有IP協議、ICMP協議、ARP協議、RARP協議和BOOTP協議。 在傳輸層中有TCP協議與UDP協議。 在應用層有: TCP包括FTP、HTTP、TELNET、SMTP等協議 UDP包括DN ...
  • 1.學前知識 1.1視頻碼率值 碼率公式: 碼率(kbps)=文件大度小(KB)*8/時間(秒) 所以碼率和視頻文件大小成正比的,不過碼率超過一定值後,人眼是看不出效果的. 接下來,我們便先來學習ffmpeg命令使用 2.ffmpeg常用命令使用 ffmpeg命令- 用於轉碼的應用程式, 也可以從u ...
  • Python 問世至今已經三十年左右了,但其僅在過去幾年人氣迅速飆升超過了除 java 和 C 以外的其他語言。總的來說,Python 已經成為教學、學習編程和軟體開發的優秀起點,而且其可以成為任何技術棧中有價值的一部分。 另外大家要註意:光理論是不夠的。這裡順便總大家一套2020最新python入 ...
  • FAE parse : 一成不變 FAE Value : interp的最終轉讓值 numV: value closureV: param FAE(或value,或function) pair list DefrdSub : 傳達environment 以前WAE中使用的DefrdSub只用於wit ...
  • 一、基礎知識 1.1、簡介 Log4Net是一個開源日誌框架,它的功能很強大,可以將日誌分為不同的等級,以不同的格式輸出到不同的存儲介質中,比如:資料庫、txt文件、記憶體緩衝區、郵件、控制台、ANSI終端、遠程接收端等等。 Log4Net將日誌分為五個級別,從高到低分別是:FATAL(致命錯誤)、E ...
  • 視頻:https://www.bilibili.com/video/BV15x411x7WN?p=6 功能點:表格分組,彙總。 結果圖: 操作步驟: 分組實現:設置GroupIndex 表格分數彙總:設置GridView的ShowFooter=true。設置分數列的彙總方式。 在分組裡實現彙總功能: ...
  • 集成.NET Core+Swagger+Consul+Polly+Ocelot+IdentityServer4+Exceptionless+Apollo的微服務開發框架 Github源代碼地址 https://github.com/PeyShine/Demo.MicroServer Apollo配置 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...