從一個簡單的ASP.NET 5站點開啟.NET跨平臺之旅
在經歷了阿里云上“黑色1秒”的空歡喜之后,我們“被迫”考慮實現.NET的跨平臺,將Web服務器由Windows換成Linux。而這種“被迫”在一個存在已久的愿望下,變得水到渠成。這個愿望就是 —— “Mac上寫.NET程序,Linux上跑.NET程序”。
既然水也到了,渠也成了,那我們還等什么,動身起程吧。
今天我們以我們邁出的***步——一個部署在Linux上基于dnx/corefx/coreclr的非常簡單的ASP.NET 5/MVC 6站點——宣布“.NET跨平臺之旅”開啟了!
這個基于跨平臺.NET的站點已經上線,訪問網址:http://about.cnblogs.com/ 。
該站點部署在CentOS服務器上(部署步驟),服務器上只安裝了dnx,沒有安裝mono,所以是完全基于.NET Core運行。后端Web服務器用的是Kestrel,也是目前跨平臺.NET在非Windows平臺上***能用的Web服務器。
CentOS服務器上運行情況如下:
[root@about-server AboutUs]# dnx . kestrel Started
前端Web服務器用的是阿里云SLB(負載均衡),如果不用SLB,可以直接在CentOS上用nginx做反向代理。為什么要用前端Web服務器?因為Kestrel Web服務器實在太簡陋了,連keep-alive與http compression的功能都沒有。
該站點的ASP.NET 5程序是在Ubuntu服務器上用vim進行開發的。
項目文件結構如下:
- .
- ├── Controllers
- │ ├── AboutController.cs
- │ └── HomeController.cs
- ├── Extensions
- │ └── HtmlHelperExtensions.cs
- ├── project.json
- ├── project.lock.json
- ├── Startup.cs
- ├── Views
- │ ├── About
- │ │ ├── Ad.cshtml
- │ │ ├── Contact.cshtml
- │ │ ├── Intro.cshtml
- │ │ └── Job.cshtml
- │ ├── Shared
- │ │ └── _Layout.cshtml
- │ └── _ViewStart.cshtml
- └── wwwroot
- ├── images
- │ ├── about_cnbogs.gif
- │ ├── icon_arrow.gif
- │ └── icon_triangle.gif
- └── styles
- └── about.css
project.json文件中的配置:
frameworks中只有dnxcore50,說明程序是完全基于.NET Core的。但由于基于coreclr的dnu restore功能目前無法使用,所以在開發環境中不得不安裝mono,用基于mono的dnu retore安裝nuget包包。
Startup.cs中的代碼如下:
- using Microsoft.AspNet.Builder;
- using Microsoft.Framework.DependencyInjection;
- namespace CNBlogs.AbouUs.Web
- {
- public class Startup
- {
- public void Configure(IApplicationBuilder app)
- {
- app.UseErrorPage();
- app.UseMvcWithDefaultRoute();
- app.UseStaticFiles();
- }
- public void ConfigureServices(IServiceCollection services)
- {
- services.AddMvc();
- }
- }
- }
(注:project.json與Startup.cs中都沒有多余的配置與代碼)
程序非常簡單,沒有數據庫操作,主要就是顯示文字內容。稍微復雜些的就是一個HtmlHelpder擴展方法(代碼是從現有項目中移植過來的),根據訪問的URL自動高亮左側的導航標簽,代碼如下:
- using Microsoft.AspNet.Mvc.Razor;
- using Microsoft.AspNet.Mvc.Rendering;
- namespace Microsoft.AspNet.Mvc.Rendering
- {
- public static class HtmlHelperExtensions
- {
- public static HtmlString TabLink(this IHtmlHelper htmlHelper, string linkText, string linkUrl, string viewName)
- {
- var view = htmlHelper.ViewContext.View as RazorView;
- if (view != null && view.Path.IndexOf("/" + viewName + ".", System.StringComparison.OrdinalIgnoreCase) > -1)
- {
- return htmlHelper.Raw(string.Format("<a href=\"{0}\" class=\"current\">{1}</a>", linkUrl, linkText));
- }
- else
- {
- return htmlHelper.Raw(string.Format("<a href=\"{0}\">{1}</a>", linkUrl, linkText));
- }
- }
- }
- }
這個ASP.NET 5程序的代碼是一步一步從無到有用vim手寫出來的(除了視圖與HtmlHelperExtensions), 從中更深刻地了解了ASP.NET 5的一些工作原理,從而也得到了一個運行這個簡單的ASP.NET 5程序所需的最小配置。
在開發過程中最痛苦的是修改代碼后ASP.NET 5不會自動重新編譯,需要重新用dnx運行程序;而且Kestrel目前有bug,無法退出,即使關閉ssh窗口,也照樣運行,必須用非常規手段強制結束 進程(ps all; kill -9 [PID])。但Kestrel的這個bug卻帶來一個讓人驚喜的副作用,正因為它一啟動就一直運行,怎么也不會退出,相當于以一種后臺服務的方式運行, 一下子解決了部署時如何后臺運行ASP.NET 5站點的問題。
雖然只是一個非常簡單的ASP.NET 5程序,雖然只是.NET跨平臺之旅非常非常小的一步,但它卻是重要的一步,因為它讓我們實實在在地感受到了——.NET跨平臺,路在腳下。
【更新】
15:35左右,出現異常造成kestrel退出,重新運行dnx之后恢復正常。異常信息如下:
Unhandled Exception: System.NullReferenceException: Object reference not set to an instance of an object. at Microsoft.AspNet.Server.Kestrel.Networking.UvShutdownReq.UvShutdownCb(IntPtr ptr, Int32 status)