想深刻了解OrchadCore源碼許久了,可是讀源碼的時候遇到不少問題而網上的參考資料太少了(幾乎都是OrchadCms不帶OrchardCore的),如今解決得差很少了,作下筆記方便本身查看,有錯誤之處也請你們幫忙指出,謝謝。java
OrchardCore是一個基於asp.net core的cms(廢話),其結構相似於asp.net core(爲啥說相似我也不知道,asp.net core的源碼我只看了一部分,似是而非吧)。所以,我將直接從asp.net core的角度來解析OrchardCore,從asp.net core的角度就是說關於asp.net core的靠本身理解,能夠省略1萬字。git
廢話結束,開始正題,源碼直接github搜索OrchardCore就能夠下載了,我直接用visual studio(我不用rider,首先我沒錢買,其次我以前是弄.net framework不是java,我用的vs都是社區版)自帶的git去clone半天(這小水管受不了後面直接碼雲找鏡像倉庫clone了)。github
clone完了,第一個坑出現了,選擇記得選擇版本和分支,說多都是淚,OrchardCore支持的版本(從.net core 1.0到.net core 3.1都有)和自身的分支太多了,第一次接觸沒注意跳坑了致使各類出錯(緣由是我沒有裝對應的.net core版本,默認好像2.x),後面發現了就直接用.net core 3.1的dev分支。新手常見錯誤,你們儘管嘲笑,讓我漲漲記性,特麼有次我本身的代碼都改錯分支!web
正文開始,vs加載完OrchardCore解決方案出現不少項目,很是清晰,加粗的OrcharCore.Cms.Web項目就是啓動項目。sql
啓動項目的內容很是熟悉很是簡潔,就是一個asp.net core的項目。shell
固然你會發現多了個Localization文件夾和NLog.config配置文件,除了這兩個以外跟咱們正常新建asp.net core項目如出一轍,全部咱們直接從asp.net core解讀開始剖析。c#
程序入口Program.cs文件的main函數(爲啥是這樣我也不知道,反正學c、學java仍是c#的時候老師和書都是這樣教的,應該是定義或者約定俗成吧)打開以下:api
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我也是隻會簡單用,就不分析了,之後有空再去看源碼app
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
接觸過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都要很久)