本章我們將講解ASP.NET5項(xiàng)目發(fā)布部署相關(guān)的內(nèi)容,示例項(xiàng)目以我們前一章創(chuàng)建的BookStore項(xiàng)目為例。
發(fā)布前的設(shè)置
由于新版ASP.NET5支持多版本DNX運(yùn)行環(huán)境的發(fā)布和部署,所以在部署之前,我們需要設(shè)定部署的目標(biāo)DNX(即之前的KRE)。
步驟:右鍵BookStore項(xiàng)目->屬性->Application選項(xiàng)卡,選擇DNX的版本,本例中,選擇dnx-coreclr-win-x64.1.0.0-beta4
。
在project.json
文件的commands
節(jié)點(diǎn),我們可以看到,系統(tǒng)默認(rèn)配置了3個(gè)調(diào)試命令,分別如下:
命令 | 描述 |
---|---|
web | 啟動(dòng)WebListener服務(wù),該服務(wù)可以讓web程序脫離IIS運(yùn)行,默認(rèn)地址是http://localhost:5000。 |
gen | 使用該命令可以生成MVC相關(guān)的代碼,比如Controller,目前還用不到。 |
ef | Entity Framework遷移命令,用于遷移數(shù)據(jù)使用,本例我們還用戶不到。 |
理論上來(lái)說(shuō),我們F5運(yùn)行的時(shí)候,應(yīng)該是啟動(dòng)web命令,但是在VS2015中,默認(rèn)的運(yùn)行環(huán)境依然是IIS Express,所以F5調(diào)試的時(shí)候,會(huì)默認(rèn)啟動(dòng)IIS Express。
注意:web模式和IIS Express模式的程序運(yùn)行端口不一樣。
我們先F5調(diào)試運(yùn)行,啟動(dòng)IIS Express,打開(kāi)頁(yè)面,一切正常。重新選擇默認(rèn)模擬器環(huán)境為web,再F5運(yùn)行,這時(shí)候發(fā)現(xiàn)彈出了一個(gè)命令行窗口,并提示如下文字:
[INFORMATION:Microsoft.NET.Http.Server.WebListener] Start[INFORMATION:Microsoft.NET.Http.Server.WebListener] Listening on prefix: http://localhost:5000/Started
代碼沒(méi)有出錯(cuò),但是并沒(méi)有打開(kāi)瀏覽器窗口,我們手工打開(kāi)一個(gè)瀏覽器訪問(wèn)上述網(wǎng)址,即可看到該示例程序的界面,此時(shí)說(shuō)明,該BookStore已經(jīng)成功運(yùn)行在5000端口了。其實(shí)該模式下的瀏覽器自動(dòng)打開(kāi)功能默認(rèn)是關(guān)閉的,可以通過(guò)如下方式開(kāi)啟自動(dòng)打開(kāi)功能:
步驟:右鍵BookStore項(xiàng)目->屬性->Debug選項(xiàng)卡,勾選Launch Brower復(fù)選框,并在輸入框里輸入上述網(wǎng)址即可(此時(shí)會(huì)在項(xiàng)目的Properties目錄下生成一個(gè)debugSettings.json文件來(lái)保存上述信息)。
再次F5運(yùn)行,即可看到自動(dòng)打開(kāi)的瀏覽器界面。
應(yīng)用程序參數(shù)
在該Debug選項(xiàng)卡中,我們還看到一個(gè)應(yīng)用程序參數(shù)(Application Arguments
)輸入框,該輸入框可以傳入多種參數(shù),這些參數(shù)可以在Startup.cs
里,通過(guò)Configuration
的AddCommandLine
方法進(jìn)行收集并利用。
環(huán)境變量
同理,在Debug選項(xiàng)卡的最下面還有一個(gè)環(huán)境變量(Environment Variables
)輸入框,可以讓我們?cè)谡{(diào)試的時(shí)候自定義一些環(huán)境變量的值(key/value),然后通過(guò)Configuration
的AddEnvironmentVariables
方法進(jìn)行收集并利用。
上述參數(shù)和環(huán)境變量的具體使用方式,請(qǐng)參考配置信息管理章節(jié)。
發(fā)布流程分析
在之前的MVC程序中,我們一般都是通過(guò)右鍵項(xiàng)目,選擇發(fā)布(Publish)的方式來(lái)發(fā)布程序的,這一次我們也來(lái)看看這種方式。
首先,右鍵->發(fā)布->Profile(選擇File System)->選擇D:/BookStore
->選擇Release/coreclr
->下一步,最終點(diǎn)擊發(fā)布。在在Output面板,我們看到出錯(cuò)了,錯(cuò)誤信息如下:
正在連接到 D:/Documents/Visual Studio 2015/Projects/BookStore/BookStore/../artifacts/bin/BookStore/Release/Publish...C:/Program Files (x86)/MSBuild/Microsoft/VisualStudio/v14.0/Web/Microsoft.DNX.Publishing.targets(342,5): 錯(cuò)誤 : 錯(cuò)誤: 無(wú)法識(shí)別規(guī)則“BackupRule”。C:/Program Files (x86)/MSBuild/Microsoft/VisualStudio/v14.0/Web/Microsoft.DNX.Publishing.targets(342,5): 錯(cuò)誤 : 錯(cuò)誤計(jì)數(shù): 1。C:/Program Files (x86)/MSBuild/Microsoft/VisualStudio/v14.0/Web/Microsoft.DNX.Publishing.targets(342,5): 錯(cuò)誤 : An error occured during publish.The command ["C:/Program Files (x86)/IIS/Microsoft Web Deploy/msdeploy.exe" -source:contentPath='C:/Users/Administrator/AppData/Local/Temp/PublishTemp/' -dest:contentPath='D:/Documents/Visual Studio 2015/Projects/BookStore/artifacts/bin/BookStore/Release/Publish' -verb:sync -enableRule:DoNotDeleteRule -retryAttempts:2 -disablerule:BackupRule ] exited with code [-1]。
通過(guò)查看輸出信息,可以發(fā)現(xiàn),編譯成功,但復(fù)制的時(shí)候出錯(cuò),可能是powershell的問(wèn)題,所以返回上述步驟,在設(shè)置(Settings)選項(xiàng)卡下,將取消發(fā)布腳本(Publish Scripts)下的使用PowerShell腳本發(fā)布的復(fù)選框。重新發(fā)布,成功了。
打開(kāi)發(fā)布目錄D:/BookStore,發(fā)現(xiàn)生成了如下目錄和文件:
目錄或文件 | 描述 |
---|---|
approot | 應(yīng)用程序目錄 |
wwwroot | 靜態(tài)文件目錄 |
gen | linux shell命令文件 |
gen.cmd | cmd命令文件 |
web | linux shell命令文件 |
web.cmd | cmd命令文件 |
看到cmd文件的擴(kuò)展名,我們可以猜想這些命令是用于執(zhí)行相關(guān)的命令,比如web.cmd
可能就是用于啟動(dòng)程序的;而非cmd擴(kuò)展名文件,我們則猜想可能是用于linux/mac運(yùn)行的命令。
我們來(lái)試一下,點(diǎn)擊web.cmd
文件,該文件執(zhí)行以后顯示的信息和我們?cè)贒ebug程序時(shí)彈出的信息一樣,通過(guò)訪問(wèn)提示中的網(wǎng)址,我們可以驗(yàn)證應(yīng)用程序已經(jīng)正常運(yùn)行了。這種模式即時(shí)我們所說(shuō)的自宿主(Self-Host)運(yùn)行模式。
再試一下IIS是否能夠運(yùn)行該程序,將IIS站點(diǎn)指向到wwwroot
目錄,打開(kāi)網(wǎng)址,也是可以正常訪問(wèn)的。打開(kāi)wwwroot
文件夾進(jìn)行查看,靜態(tài)文件一應(yīng)俱全,但是發(fā)現(xiàn)bin目錄下并沒(méi)有我們的項(xiàng)目DLL(BookStore.dll
),而是多了一個(gè)AspNet.Loader.dll
,而且根目錄下還多了一個(gè)web.config
文件,內(nèi)容如下:
<?xml version="1.0" encoding="utf-8"?><configuration> <appSettings> <add key="bootstrapper-version" value="1.0.0-beta4" /> <add key="runtime-path" value="../approot/packages" /> <add key="dnx-version" value="1.0.0-beta4" /> <add key="dnx-clr" value="coreclr" /> <add key="dnx-app-base" value="../approot/src/BookStore" /> </appSettings></configuration>
通過(guò)查詢相關(guān)信息(訪問(wèn)詳情) ,得知AspNet.Loader.dll
文件只是一個(gè)橋接文件,用于接收IIS轉(zhuǎn)發(fā)過(guò)來(lái)的請(qǐng)求,然后將其轉(zhuǎn)交給dnx
進(jìn)行運(yùn)行,這里的web.config
里的dnx以及項(xiàng)目信息的配置文件是AspNet.Loader.dll
在轉(zhuǎn)交請(qǐng)求時(shí)所需要的配置信息。
通過(guò)配置文件我們可以看到,這里配置了dnx
的類型、版本號(hào),程序集的路徑和app的路徑。打開(kāi)approot/src/BookStore
目錄,我們發(fā)現(xiàn),這里居然都是cs源碼,雖然有個(gè)bin
目錄,但是里面也沒(méi)有dll文件。而且在approot/packages
文件夾下,居然有90個(gè)程序集文件夾(將近30M文件)。
通過(guò)查詢網(wǎng)站的資料得知(這一部分內(nèi)容,我們?cè)谙乱还?jié)進(jìn)行講解),目前真正運(yùn)行程序的運(yùn)行環(huán)境是DNX
,也被復(fù)制到approot/packages/dnx-coreclr-win-x64.1.0.0-beta4
目錄中, 而該項(xiàng)目依賴的所有程序集(包括System開(kāi)頭的)都被復(fù)制到該packages目錄下了。目的就是要做到真正的跨平臺(tái)運(yùn)行,也就是說(shuō),將這些文件復(fù)制到linux系統(tǒng)下,只要有對(duì)應(yīng)版本的KRE(本例中的DNX是Windows版本的)的話,就可以正常運(yùn)行該程序。
而bin目錄下沒(méi)有dll文件,則是使用了微軟最新的動(dòng)態(tài)編譯技術(shù),即在運(yùn)行的過(guò)程中,自動(dòng)編譯cs文件,而且一旦修改這些cs文件的話,系統(tǒng)將會(huì)自動(dòng)再次進(jìn)行編譯。(感覺(jué)有點(diǎn)像php等腳本語(yǔ)言了)。雖然動(dòng)態(tài)編譯很高效,但是還是沒(méi)有編譯好的dll高效,所以微軟還提供了一個(gè)選項(xiàng)讓開(kāi)發(fā)人員在調(diào)試的時(shí)候生成dll文件。具體步驟如下:
右鍵BookStore->屬性->Build選項(xiàng)卡,勾選編譯時(shí)生成輸出(Produce outputs on build)復(fù)選框。
重新編譯程序,發(fā)現(xiàn)在BookStore/artifacts/bin/BookStore/Debug
目錄下的2個(gè)DNX版本文件夾下都分別生成了BookStore.dll文件了,而且還順帶了Nuget的spec文件。
如果在發(fā)布的時(shí)候也要生成dll文件,則需要在發(fā)布(Publish)設(shè)置里進(jìn)行修改,步驟如下:
右鍵BookStore->發(fā)布(Publish)->Settings選項(xiàng)卡->File Publish Options->勾選Precompile during publishing復(fù)選框。
這樣就可以生成響應(yīng)的dll文件, 但是這些dll文件依然不在wwwroot/bin目錄下,而是在approot/packages/BookStore/1.0.0
目錄下,在該目錄下有2個(gè)文件夾,分別是lib
和root
,以及相關(guān)的Nuget的spec文件,在lib目錄下,生成的是不同dnx版本的dll文件,而root則是類似于之前的web根目錄,因?yàn)樵谠撃夸浵鲁擞幸晥D文件以外,還和以前的結(jié)構(gòu)一樣,保留了bin目錄,并且在bin目錄下的Release文件夾下,也有一份針對(duì)不同dnx版本的dll文件副本。
提示:上述選擇中,另外一個(gè)Delete all existing files prior to publish也可以勾選上,以便在發(fā)布時(shí)將之前發(fā)布版本的所有文件全部清空。
此時(shí),我們通過(guò)web.cmd文件或者IIS模式來(lái)驗(yàn)證發(fā)布的文件,經(jīng)驗(yàn)證,均可以正常運(yùn)行。再仔細(xì)對(duì)比兩份不同設(shè)在的發(fā)布文件,發(fā)現(xiàn),除了dll文件以外,web.config文件的應(yīng)用程序路徑也變了,即從原來(lái)的:
<add key="kre-app-base" value="../approot/src/BookStore" />
變成了如下版本:
<add key="kre-app-base" value="../approot/packages/BookStore/1.0.0/root" />
而web.cmd文件的內(nèi)容,也從如下內(nèi)容:
@"%~dp0approot/packages/dnx-coreclr-win-x64.1.0.0-beta4/bin/dnx.exe" --appbase "%~dp0approot/src/BookStore" Microsoft.Framework.ApplicationHost web %*
變成了如下內(nèi)容:
@"%~dp0approot/packages/kre-coreclr-win-x64.1.0.0-beta4/bin/dnx.exe" --appbase "%~dp0approot/packages/BookStore/1.0.0/root" Microsoft.Framework.ApplicationHost web %*
上述變化,我們是可以理解的,即將src源碼動(dòng)態(tài)編譯運(yùn)行的模式修改為預(yù)編譯dll程序集的模式。所以,在這里我們可以看到,在源碼動(dòng)態(tài)編譯模式下,其發(fā)布后的文件夾結(jié)構(gòu)如下:
//源碼動(dòng)態(tài)編譯模式wwwroot/bin/Microsoft.AspNet.Loader.IIS.dllwwwroot/Contents/site.csswwwroot/Contents/...............................................................................................wwwroot/Scripts/jquery.jswwwroot/Scripts/........................................................................................................................................................approot/src/BootStore/project.jsonapproot/src/BootStore/...............................approot/src/BootStore.Data/project.jsonapproot/src/BootStore.Data/..............................approot/src/BootStore.Bussiness/project.jsonapproot/src/BootStore.Bussiness/.........................approot/packages/Elmah/{version}/...............................................................................
而dll預(yù)編譯模式下的發(fā)布文件夾結(jié)構(gòu)如下:
//dll預(yù)編譯模式wwwroot/bin/Microsoft.AspNet.Loader.IIS.dllwwwroot/Contents/site.csswwwroot/Contents/...............................................................................................wwwroot/Scripts/jquery.jswwwroot/Scripts/........................................................................................................................................................approot/packages/BootStore/{version}/...................approot/packages/BootStore.Data/{version}/..............approot/packages/BootStore.Bussiness/{version}/.........approot/packages/Elmah/{version}/.......................
IIS和web.cmd模式的不同
雖然我們對(duì)dnx內(nèi)容的原理不太理解,但有一點(diǎn)內(nèi)容,我們要記住,那就是兩種模式下,對(duì)靜態(tài)文件的訪問(wèn)模式可能不太一樣。原因是因?yàn)椋m然IIS模式的根目錄就是存放靜態(tài)文件的地方,但是web.cmd文件事先啟動(dòng)的卻是approot/src/BookStore
目錄或approot/packages/BookStore/1.0.0/root
目錄,兩個(gè)目錄下均沒(méi)有靜態(tài)文件,因?yàn)殪o態(tài)文件時(shí)在wwwroot
目錄下的,我們猜想,在這種模式下,肯定會(huì)有一種機(jī)制在來(lái)映射這些靜態(tài)文件,通過(guò)查找文件發(fā)現(xiàn),在approot/src/BookStore
目錄下的project.json
文件中的webroot
鍵的值,從解決方案中默認(rèn)的wwwroot
變成了"../../../wwwroot"
,也就是說(shuō)kre在映射靜態(tài)文件的時(shí)候,應(yīng)該是根據(jù)這個(gè)相對(duì)目錄來(lái)查找這些文件的。
同理,approot/packages/BookStore/1.0.0/root
目錄下的project.json
文件中的webroot
鍵的值,也從wwwroot
變成了"../../../../../wwwroot"
(因?yàn)楸緛?lái)project.json
文件的層級(jí)就深)。
由于IIS是通過(guò)AspNet.Loader.dll
做中轉(zhuǎn),將請(qǐng)求轉(zhuǎn)交給DNX來(lái)運(yùn)行的,那么在IIS模式下,靜態(tài)文件的請(qǐng)求到底是IIS來(lái)處理,還是KRE來(lái)處理呢?我們來(lái)驗(yàn)證一下,驗(yàn)證步驟如下:
創(chuàng)建一個(gè)wwwroot2
文件夾和wwwroot
同級(jí),并將wwwrooot
目錄下的靜態(tài)文件剪切到wwwroot2
目錄下。將project.json
(如果是預(yù)編譯模式,則需要修改root目錄下的project.json)文件中的webroot
值中的wwwroot
修改為wwwroot2
。繼續(xù)以IIS模式運(yùn)行該站點(diǎn)
結(jié)果發(fā)現(xiàn),靜態(tài)文件訪問(wèn)不了了(CSS、JS、Images均失效了),但我們?cè)偻ㄟ^(guò)web.cmd運(yùn)行時(shí),這些靜態(tài)文件卻又可以訪問(wèn)了。由此得知,在IIS模式下,靜態(tài)文件走的是IIS的管線Pipeline,而不是DNX的關(guān)系Pipeline。
兩種發(fā)布模式下的project.json文件不同
動(dòng)態(tài)編譯模式和預(yù)編譯dll模式這兩種模式的自動(dòng)發(fā)布程序,生成后的project.json文件有一些變化,具體變化如下。
動(dòng)態(tài)編譯模式
基本上和解決方案里的project.json文件相同,唯一的不同就是webroot的相對(duì)路徑的修改。
預(yù)編譯dll模式
原來(lái)引用的眾多程序集從dependencies節(jié)點(diǎn)中移除了,取而代之的是BookStore程序集引用,示例如下:
"dependencies": { "BookStore": "1.0.0"},
另外,還多了如下兩個(gè)節(jié)點(diǎn)值(具體功能暫不明確):
"entryPoint": "BookStore","loadable": false
猜想,這些不同,可能是因?yàn)樵趧?dòng)態(tài)編譯模式下需要引用這些被移除的程序集進(jìn)行編譯,而預(yù)編譯dll模式下,都已經(jīng)編譯好了,所以就不再需要這些程序集了,而root目錄只需要引用BookStore程序集就可以了,而BookStore程序集對(duì)這些程序集的依賴,詳細(xì)在該dll程序集的nupkg文件里是可以自動(dòng)解析并下載的吧(這一點(diǎn)待驗(yàn)證)。
以上是新版ASP.NET5項(xiàng)目在發(fā)布流程和相關(guān)技術(shù)的一些內(nèi)容,從這里大家可以看到,ASP.NET5是徹底模塊化了,IIS不再是運(yùn)行MVC程序的唯一容器,任何兼容DNX的運(yùn)行容器都可以運(yùn)行MVC程序,程序發(fā)布包被分為approot和wwwroot兩個(gè)部分,分別存放應(yīng)用程序集(或源碼)和靜態(tài)文件,從而做到更好的分離。在下一章,我們會(huì)討論,ASP.NET 5的運(yùn)行原理。
注意:目前還沒(méi)有辦法通過(guò)復(fù)制源碼的形式來(lái)進(jìn)行調(diào)試,同時(shí)也沒(méi)辦法將IIS指向到源碼中進(jìn)行調(diào)試,這將會(huì)改變開(kāi)發(fā)人員的開(kāi)發(fā)習(xí)慣。
新聞熱點(diǎn)
疑難解答
圖片精選