想深入了解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都要好久)
轉載請註明出處,本文鏈接:https://www.uj5u.com/net/3520.html
標籤:.NET Core
