想深入瞭解OrchadCore原始碼許久了,但是讀原始碼的時候遇到很多問題而網上的參考資料太少了(幾乎都是OrchadCms不帶OrchardCore的),現在解決得差不多了,做下筆記方便自己檢視,有錯誤之處也請大家幫忙指出,謝謝。
OrchardCore是一個基於asp.net core的cms(廢話),其結構類似於asp.net core(為啥說類似我也不知道,asp.net core的原始碼我只看了一部分,似是而非吧)。因此,我將直接從asp.net core的角度來解析OrchardCore,從asp.net core的角度就是說關於asp.net core的靠自己理解,可以省略1萬字。
廢話結束,開始正題,原始碼直接github搜尋OrchardCore就可以下載了,我直接用visual studio(我不用rider,首先我沒錢買,其次我之前是弄.net framework不是java,我用的vs都是社群版)自帶的git去clone半天(這小水管受不了後面直接碼雲找映象倉庫clone了)。
clone完了,第一個坑出現了,選擇記得選擇版本和分支,說多都是淚,OrchardCore支援的版本(從.net core 1.0到.net core 3.1都有)和自身的分支太多了,第一次接觸沒注意跳坑了導致各種出錯(原因是我沒有裝對應的.net core版本,預設好像2.x),後面發現了就直接用.net core 3.1的dev分支。新手常見錯誤,大家儘管嘲笑,讓我漲漲記性,特麼有次我自己的程式碼都改錯分支!
正文開始,vs載入完OrchardCore解決方案出現很多專案,非常清晰,加粗的OrcharCore.Cms.Web專案就是啟動專案。
啟動專案的內容非常熟悉非常簡潔,就是一個asp.net core的專案。
當然你會發現多了個Localization資料夾和NLog.config配置檔案,除了這兩個之外跟我們正常新建asp.net core專案一模一樣,所有我們直接從asp.net core解讀開始剖析。
程式入口Program.cs檔案的main函式(為啥是這樣我也不知道,反正學c、學java還是c#的時候老師和書都是這樣教的,應該是定義或者約定俗成吧)開啟如下:
namespace OrchardCore.Cms.Web { public class Program { public static Task Main(string[] args) => BuildHost(args).RunAsync(); public static IHost BuildHost(string[] args) => Host.CreateDefaultBuilder(args) .ConfigureLogging(logging => logging.ClearProviders()) .ConfigureWebHostDefaults(webBuilder => webBuilder .UseStartup<Startup>() .UseNLogWeb()) .Build() ; } }
非常明顯的asp.net core程式碼,建立Host(主機),然後Build(生成),然後Run(執行),跟新建asp.net core專案唯一區別就多了ConfigureLogging和ConfigureWebHostDefaults兩個配置了,意思也很清楚(不清楚得去看asp.net core的原始碼了,此處省略1萬個字),就是清除預設日誌提供者然後用NLog代替(感覺就是個人愛好,你用log4net又咋樣),這也是專案裡有NLog.config這個配置檔案的原因。要了解logging.ClearProviders()直接看asp.net core的api去。UseNLogWeb()是OrchardCord自定義的擴充套件方法,程式碼非常清楚就是載入NLog.config這個配置檔案,NLog我也是隻會簡單用,就不分析了,以後有空再去看原始碼
public static IWebHostBuilder UseNLogWeb(this IWebHostBuilder builder) { LayoutRenderer.Register<TenantLayoutRenderer>(TenantLayoutRenderer.LayoutRendererName); builder.UseNLog(); builder.ConfigureAppConfiguration((context, configuration) => { var environment = context.HostingEnvironment; environment.ConfigureNLog($"{environment.ContentRootPath}{Path.DirectorySeparatorChar}NLog.config"); LogManager.Configuration.Variables["configDir"] = environment.ContentRootPath; }); return builder; }
我重點關注還是入口檔案的UseStartup<Startup>(),總所周知這個是asp.net core的核心中的核心,沒錯,就是服務和中介軟體,終於來到重點了,這個會呼叫專案裡Startup.cs的程式碼通過中介軟體生成管道,至於原理又是關於asp.net core的執行原理又可以省略1萬字。
接觸過asp.net core都知道通過Startup的ConfigureServices方法註冊服務,Configure方法配置管道中介軟體。
public class Startup { public void ConfigureServices(IServiceCollection services) { services.AddOrchardCms(); } public void Configure(IApplicationBuilder app, IHostEnvironment env) { if (env.IsDevelopment()) { app.UseDeveloperExceptionPage(); } app.UseStaticFiles(); app.UseOrchardCore(); } }
asp.net core表示服務配置可以沒有。服務部分我是直接跳過了,這裡是直接呼叫擴充套件方法services.AddOrchardCms()配置服務,其實服務按我理解就是依賴注入的註冊,依賴注入是asp.net core的基礎沒啥好說的,反正就是後面可以自動例項化建構函式的類,所以這個地方導致我後面又掉了一個坑(找錯例項的類)。
asp.net core的中介軟體配置一定要有,所以這部分是重中之重,前面的app.UseDeveloperExceptionPage()和app.UseStaticFiles()就不用說了,一般的asp.net core專案都有,就是異常和靜態檔案中介軟體。這個OrchardCore最重要的內容在於app.UseOrchardCore()這個擴充套件方法,這個擴充套件方法主要是後面兩個中介軟體,當然前面那個PoweredByMiddleware也很重要(作者說我辛辛苦苦弄一個cms你居然說我的版權不重要那我就慘了),我這裡只是說程式的執行很重要並不是不尊重版權(我本身很尊重各個版權的,所以vs我只能用社群版、sqlserver只能用epress版)
public static class ApplicationBuilderExtensions { /// <summary> /// Enables multi-tenant requests support for the current path. /// </summary> public static IApplicationBuilder UseOrchardCore(this IApplicationBuilder app, Action<IApplicationBuilder> configure = null) { var env = app.ApplicationServices.GetRequiredService<IHostEnvironment>(); var appContext = app.ApplicationServices.GetRequiredService<IApplicationContext>(); env.ContentRootFileProvider = new CompositeFileProvider( new ModuleEmbeddedFileProvider(appContext), env.ContentRootFileProvider); // Init also the web host 'ContentRootFileProvider'. app.ApplicationServices.GetRequiredService<IWebHostEnvironment>() .ContentRootFileProvider = env.ContentRootFileProvider; app.UseMiddleware<PoweredByMiddleware>(); // Ensure the shell tenants are loaded when a request comes in // and replaces the current service provider for the tenant's one. app.UseMiddleware<ModularTenantContainerMiddleware>(); configure?.Invoke(app); app.UseMiddleware<ModularTenantRouterMiddleware>(app.ServerFeatures); return app; } }
ModularTenantContainerMiddleware和ModularTenantRouterMiddleware這兩個中介軟體撐起OrchardCore一片天,下次就從UseOrchardCore繼續!從asp.net core角度去看真心簡化好多,但是後面估計要惡補下asp.net core了(原始碼的專案太多,這個感覺要看很久,而且目前github上面的asp.net core原始碼都是基於.net 5的,build都要好久)