AspNetCore開源中介軟體-VueRouterHistory

gui.h發表於2022-05-09

前言

用過VueRouter路由元件的應該都知道,VueRouterhashhistory兩種模式。hash模式會在url中插入#history模式下url則看上去更加簡潔美觀。如果想要支援history模式則必須要後端服務進行配合。

常用後端伺服器配置方式請參考 後端配置例子

後端配置例子

注意:下列示例假設你在根目錄服務這個應用。如果想部署到一個子目錄,你需要使用 Vue CLI 的 publicPath 選項 (opens new window)和相關的 router base property (opens new window)。你還需要把下列示例中的根目錄調整成為子目錄 (例如用 RewriteBase /name-of-your-subfolder/ 替換掉 RewriteBase /)。

Apache

<IfModule mod_rewrite.c>
  RewriteEngine On
  RewriteBase /
  RewriteRule ^index\.html$ - [L]
  RewriteCond %{REQUEST_FILENAME} !-f
  RewriteCond %{REQUEST_FILENAME} !-d
  RewriteRule . /index.html [L]
</IfModule>

除了 mod_rewrite,你也可以使用 FallbackResource

nginx

location / {
  try_files $uri $uri/ /index.html;
}

原生 Node.js

const http = require('http')
const fs = require('fs')
const httpPort = 80

http.createServer((req, res) => {
  fs.readFile('index.html', 'utf-8', (err, content) => {
    if (err) {
      console.log('We cannot open "index.html" file.')
    }

    res.writeHead(200, {
      'Content-Type': 'text/html; charset=utf-8'
    })

    res.end(content)
  })
}).listen(httpPort, () => {
  console.log('Server listening on: http://localhost:%s', httpPort)
})

基於 Node.js 的 Express

對於 Node.js/Express,請考慮使用 connect-history-api-fallback 中介軟體 。

Internet Information Services (IIS)

安裝 IIS UrlRewrite

在你的網站根目錄中建立一個 web.config 檔案,內容如下:

<?xml version="1.0" encoding="UTF-8"?>
<configuration>
  <system.webServer>
    <rewrite>
      <rules>
        <rule name="Handle History Mode and custom 404/500" stopProcessing="true">
          <match url="(.*)" />
          <conditions logicalGrouping="MatchAll">
            <add input="{REQUEST_FILENAME}" matchType="IsFile" negate="true" />
            <add input="{REQUEST_FILENAME}" matchType="IsDirectory" negate="true" />
          </conditions>
          <action type="Rewrite" url="/" />
        </rule>
      </rules>
    </rewrite>
  </system.webServer>
</configuration>

Caddy

rewrite {
    regexp .*
    to {path} /
}

Firebase 主機

在你的 firebase.json 中加入:

{
  "hosting": {
    "public": "dist",
    "rewrites": [
      {
        "source": "**",
        "destination": "/index.html"
      }
    ]
  }
}

警告

給個警告,因為這麼做以後,你的伺服器就不再返回 404 錯誤頁面,因為對於所有路徑都會返回 index.html 檔案。為了避免這種情況,你應該在 Vue 應用裡面覆蓋所有的路由情況,然後再給出一個 404 頁面。

const router = new VueRouter({
  mode: 'history',
  routes: [
    { path: '*', component: NotFoundComponent }
  ]
})

或者,如果你使用 Node.js 伺服器,你可以用服務端路由匹配到來的 URL,並在沒有匹配到路由的時候返回 404,以實現回退。更多詳情請查閱 Vue 服務端渲染文件 (opens new window)。

原生AspNetCore實現

現如今AspNetCore完全不需要依賴IIS即可進行部署,如何在AspNetCore原生應用中進行支援VueRouterhistory想必是很多人遇到到的問題之一,也許大部分人選擇使用hash模式,因為它雖然醜點,但是不需要任何配置即可使用。
為了帶給像我一樣強烈需要history模式的使用者,索性寫了箇中介軟體,經過測試,能夠完美支援VueRouter元件history模式的部署。

VueRouterHistory

VueRouterHistory是實現原生AspNetCore下支援VueRouterhistory模式的中介軟體。

原始碼已開源在Github: https://github.com/SpringHgui/VueRouterHistory

使用方法

  1. 通過nuget安裝VueRouterHistory
Install-Package VueRouterHistory -Version 1.0.2
  1. 註冊中介軟體app.UseVueRouterHistory()

app.UseRouting()app.MapControllers()之後新增app.UseVueRouterHistory();

// This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
    if (env.IsDevelopment())
    {
        app.UseDeveloperExceptionPage();
    }
    else
    {
        app.UseExceptionHandler("/Home/Error");
        // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
        app.UseHsts();
    }
    app.UseHttpsRedirection();

    app.UseStaticFiles();

    app.UseRouting();
    
    // ==============新增這一行即可================
    app.UseVueRouterHistory();

    app.UseAuthorization();

    app.UseEndpoints(endpoints =>
    {
        endpoints.MapControllerRoute(
            name: "default",
            pattern: "{controller=Home}/{action=Index}/{id?}");
    });
}
  1. 將Vue編譯後的產物全部放置到wwwroot資料夾下
  2. 開始體驗你的應用吧~

結語

VueRouterHistory中介軟體的使用,讓我們免於對iis進行配置以實現history模式部署,使專案不管是託管在IIS還是直接自託管模式,都不需要進行額外的配置。
歡迎有需要的朋友通過VueRouterHistory包進行支援history模式,如遇到問題,請提交ISSU。

相關文章