黄色电影一区二区,韩国少妇自慰A片免费看,精品人妻少妇一级毛片免费蜜桃AV按摩师 ,超碰 香蕉

ASP.NET Core 集成 React SPA應(yīng)用的步驟

asp.net core 集成 react spa應(yīng)用的步驟

agileconfig的ui使用react重寫快完成了。上次搞定了基于jwt的登錄模式(antdesign pro + .net core 實(shí)現(xiàn)基于jwt的登錄認(rèn)證),但是還有點(diǎn)問題。現(xiàn)在使用react重寫后,agileconfig成了個(gè)確確實(shí)實(shí)的前后端分離項(xiàng)目。那么其實(shí)部署的話要分2個(gè)站點(diǎn)部署,把前端build完的靜態(tài)內(nèi)容部署在一個(gè)網(wǎng)站,把server端也部署在一個(gè)站點(diǎn)。然后修改前端的baseurl讓spa的api請求都指向server的網(wǎng)站。
這樣做也不是不行,但是這不符合agileconfig的精神,那就是簡單。asp.net core程序本身其實(shí)就是一個(gè)http服務(wù)器,所以完全可以把spa網(wǎng)站使用它來承載。這樣只需要部署一個(gè)站點(diǎn)就可以同時(shí)跑spa跟后端server了。
其實(shí)最簡單的辦法就是把build完的文件全部丟wwwroot文件夾下面。然后訪問:

http://localhost:5000/index.html

但是這樣我們的入口是index.html,這樣看起來比較別扭,不夠友好。而且這些文件直接丟在wwwroot的根目錄下,會(huì)跟網(wǎng)站其他js、css等內(nèi)容混合在一起,也很混亂。
那么下面我們就要解決這兩個(gè)文件,我們要達(dá)到的目的有2個(gè):

  • spa的入口path友好,比如http://localhost:5000/ui
  • spa靜態(tài)文件存放的目錄獨(dú)立,比如存放在wwwroot/ui文件夾下,或者別的什么目錄下。
  • 要實(shí)現(xiàn)以上內(nèi)容只需要一個(gè)自定義中間件就可以了。

     

    wwwroot\ui

    wwwroot\ui

    我們把build完的靜態(tài)文件全部復(fù)制到wwwroot\ui文件夾內(nèi),以跟其他靜態(tài)資源進(jìn)行區(qū)分。當(dāng)然你也可以放在任意目錄下,只要是能讀取到就可以。

     

    reactuimiddleware

    namespace agileconfig.server.apisite.uiextension
    {
      public class reactuimiddleware
      {
          private static dictionary<string, string> _contenttypes = new dictionary<string, string>
          {
              {".html", "text/html; charset=utf-8"},
              {".css", "text/css; charset=utf-8"},
              {".js", "application/javascript"},
              {".png", "image/png"},
              {".svg", "image/svg+xml"},
              { ".json","application/json;charset=utf-8"},
              { ".ico","image/x-icon"}
          };
          private static concurrentdictionary<string, byte[]> _staticfilescache = new concurrentdictionary<string, byte[]>();
          private readonly requestdelegate _next;
          private readonly ilogger _logger;
          public reactuimiddleware(
             requestdelegate next,
             iloggerfactory loggerfactory
         )
          {
              _next = next;
              _logger = loggerfactory.
                  createlogger<reactuimiddleware>();
          }
    
          private bool shouldhandleuirequest(httpcontext context)
          {
              return context.request.path.hasvalue && context.request.path.value.equals("/ui", stringcomparison.ordinalignorecase);
          }
    
          private bool shouldhandleuistaticfilesrequest(httpcontext context)
          {
              //請求的的referer為 0.0.0.0/ui ,以此為依據(jù)判斷是否是reactui需要的靜態(tài)文件
              if (context.request.path.hasvalue && context.request.path.value.contains("."))
              {
                  context.request.headers.trygetvalue("referer", out stringvalues referervalues);
                  if (referervalues.any())
                  {
                      var referervalue = referervalues.first();
                      if (referervalue.endswith("/ui", stringcomparison.ordinalignorecase))
                      {
                          return true;
                      }
                  }
              }
    
              return false;
          }
    
          public async task invoke(httpcontext context)
          {
              const string uidirectory = "wwwroot/ui";
              //handle /ui request
              var filepath = "";
              if (shouldhandleuirequest(context))
              {
                  filepath = uidirectory + "/index.html";
              }
              //handle static files that referer = xxx/ui
              if (shouldhandleuistaticfilesrequest(context))
              {
                  filepath = uidirectory + context.request.path;
              }
    
              if (string.isnullorempty(filepath))
              {
                  await _next(context);
              }
              else
              {
                  //output the file bytes
    
                  if (!file.exists(filepath))
                  {
                      context.response.statuscode = 404;
                      return;
                  }
    
                  context.response.onstarting(() =>
                  {
                      var exttype = path.getextension(filepath);
                      if (_contenttypes.trygetvalue(exttype, out string contenttype))
                      {
                          context.response.contenttype = contenttype;
                      }
                      return task.completedtask;
                  });
    
                  await context.response.startasync();
    
                  byte[] filedata = null;
                  if (_staticfilescache.trygetvalue(filepath, out byte[] outfiledata))
                  {
                      filedata = outfiledata;
                  }
                  else
                  {
                      filedata = await file.readallbytesasync(filepath);
                      _staticfilescache.tryadd(filepath, filedata);
                  }
                  await context.response.bodywriter.writeasync(filedata);
    
                  return;
              }
          }
      }
    }

    大概解釋下這個(gè)中間件的思路。這個(gè)中間件的邏輯大概是分量部分。
    1.攔截請求的路徑為/ui的請求,直接從ui文件夾讀取index.html靜態(tài)文件的內(nèi)容然后輸出出去,這就相當(dāng)于直接訪問/index.html。但是這樣的路徑形式看起來更加友好。
    2.攔截react spa需要的靜態(tài)資源文件,比如css文件,js文件等。這里比較麻煩,因?yàn)閟pa拉靜態(tài)文件的時(shí)候path是直接從網(wǎng)站root開始的,比如http://localhost:5000/xxx.js,那么怎么區(qū)分出來這個(gè)文件是react spa需要的呢?我們判斷一下請求的referer頭部,如果referer的path是/ui,那么就說明是react spa需要的靜態(tài)資源,同樣從ui文件夾去讀取。
    這里還需要給每個(gè)response設(shè)置指定的contenttype不然瀏覽器無法準(zhǔn)確識(shí)別資源。

     public void configure(iapplicationbuilder app, iwebhostenvironment env, iserviceprovider serviceprovider)
          {
              if (env.isdevelopment())
              {
                  app.usedeveloperexceptionpage();
              }
              else
              {
                  app.usemiddleware<exceptionhandlermiddleware>();
              }
              app.usemiddleware<reactuimiddleware>();
          
          ...
          ...
    
          }

    在startup類的configure方法內(nèi)使用這個(gè)中間件。這樣我們的改造就差不多了。

     

    運(yùn)行一下

    訪問下http://localhost:5000/ui 可以看到spa成功加載進(jìn)來了。

     

    總結(jié)

    為了能讓asp.net core承載react spa應(yīng)用,我們使用一個(gè)中間件進(jìn)行攔截。當(dāng)訪問對應(yīng)path的時(shí)候從本地文件夾內(nèi)讀取靜態(tài)資源返回給瀏覽器,從而完成spa所需要資源的加載。這次使用react spa來演示,其實(shí)換成任何spa應(yīng)用都是一樣的操作。
    代碼在這:reactuimiddleware

    以上就是asp.net core 集成 react spa應(yīng)用的步驟的詳細(xì)內(nèi)容,更多關(guān)于asp.net core 集成 react spa的資料請關(guān)注碩編程其它相關(guān)文章!

    下一節(jié):.net數(shù)據(jù)庫操作框架sqlsugar的簡單入門

    asp.net編程技術(shù)

    相關(guān)文章