基于umijs多模块系统调试开发方案

基于umijs多模块系统调试开发方案

背景

​ 随着前端开发过程工程化的深入,前端SPA应用已经越来越多的在各种信息系统内部采用,而且随着微服务架构的深入,前端SPA应用的使用又更加普及了。如果只是一个独立应用的开发,那么前端在开发阶段,往往是以下形态存在的:

spa app产品形态.png

​ 本地npm run start,通过了devServer开启了一个web服务器,另外通过本地模拟或者proxy的方式,把向后端请求接口代理到了后端开发者提供给我们的地址。

​ 这种方式的确简单和方便,但是有个前提:一个独立的应用。如果开发的是一个大型系统的一个模块。可以参考下图:

信息系统部署形态.png

​ 那么开发以上单个模块的前端工程,往往需要和多方进行交互,甚至还可能存在前端运行的特定容器。这些交互包括:请求后端其他(非本模块)接口服务、接口请求权限认证、前端调用其他模块页面等等。如果还是采用简单的方式进行本地开发,伴随而来的就需要解决诸如:添加多个proxy、前端模拟登录、前端跨域问题等等,而这种问题的解决,同时也伴随着工程化配置的复杂化。

​ 本文就基于umijs脚手架前端应用为基础,讲解伴随着前端工程化的深入,前端应用本地开发三种解决以上问题的方式。

可行方案迭代

前端Http全局跨域

​ 前端开发相关性最强的是后端请求接口以及接口引起的权限和跨域问题。那么如果借助浏览同一个窗口,同域cookie共享的机制,自然而然可以想象把前端页面对后端接口的请求,通过全地址的方式进行处理(而不是devServer代理一次)。这样可以提前登陆多模块的平台系统,然后设置浏览器全局跨域的方式,绕过接口的授权和跨域限制。

​ 简单来说就是两步:

  • 修改前端异步请求库,添加后端接口全地址前缀,比如axiosjquery的设置

    import axios from 'axios';
    
    // 后端接口全路径前缀
    let serverHost = "http://192.168.91.66/"
    
    if (process.env.NODE_ENV !== 'development') {
      axios.defaults.baseURL = serverHost;
      jquery.ajaxSettings.baseURL = serverHost;
    }
    
  • 修改浏览器跨域限制,默认开发阶段支持接口调用全局跨域。

    chrome浏览器,右键属性,在目标栏,添加--args --disable-web-security --user-data-dir="C:/chromeDev"

chrome 跨域.png

这样处理以后,保证了后端接口的合理调用,但是这种处理方式也存在一定的限制,比如,如果是前端引用其他模块的页面,跨页面之间的交互仍旧是限制的(iframe内部window.parent)。

前端devServer代理

既然第一种方案不能那么完美的解决跨页面交互跨域的问题,那么第二种方案自然而然的产生了,深度使用devServer的代理功能。

我们通过umijs配置的proxy属性,把全部需要依赖的后端接口都添加上代理,包括前端跨页面访问的其他模块页面地址,一般存在的形式可能如下:

proxy:{
    '/xxService': {
        target: 'http://10.30.21.32:8080/',
        changeOrigin: true
    }
}

同时每次开发调试前,登录多模块系统,浏览器端获取到cookie信息(chrome F12 Appliaction),添加到proxy代理头:

const msServiceCookie = 'language=zh_CN; JSESSIONID=EBE18455DA8A776B097DE467B12E6833; GWSessionId=839AEF9FA35BCAF24466CF7A6C85063C; CASTGC=TGT-44-UUEoZ9yhQc1vNNzdeng2YQewyvsErWbjCLPqfTGAYGIFqYiaML';
const isDev = process.env.NODE_ENV === 'development';

const proxyHeaders = isDev ? {
    'Cookie': msServiceCookie,
} : {};

proxy:{
    '/xxService': {
        target: 'http://10.30.21.32:8080/',
        changeOrigin: true,
        headers: proxyHeaders
    }
}

这样,本地前端开发调试过程,无论是接口调用还是跨域问题,都可以解决了。唯一不太方便的地方就是,cookie是有有效期的,每次失效得从新赋值,后端接口如果前缀规则很多,在proxy内部需要配置多次。

Nginx同域代理

行文至此,似乎本文的话题应该结束了,但是总觉得以上两种方案还是不够完美,同时又让我想起了一句万变不离其中的真理:

计算机科学体系内,没有加一层解决不了的问题

对,下面描述的就是基于多一层(Nginx),来实现更合适这种场景的解决方案。

Nginx的特点就不再赘述,我们今天用到的就是其最普通的功能:代理。最终我们的目标就是,把我们本地的调试站点和多模块的平台系统实现同域化。当然还要适当的应用umijs非根目录部署特性。大概原理如下:

架构流程图-产品形态的代理.png

我们在本地开发环境运行一个Nginx服务(http://127.0.0.1:9528),配置其代理,使得路径/app-local//sockjs-node/代理到本地开发启动的nodejs服务(本地代码http://127.0.0.1:9527),其他路径直接代理到目标多模块平台系统(http://10.30.5.36:8080/)。这样借助nginx代理自动携带cookie的特色,我们可以直接访问本地http://127.0.0.1:9528,既可以做到同访问http://10.30.5.36:8080的效果了。而只有/app-local/和/sockjs-node/代理到了我们的本地,这也正是我们需要达到的目标。参考nginx示例如下:


    # 开发代理
    server {
    
        listen 9528;
        server_name_in_redirect off;

        # 本地开发启动的站点
        location ^~/app-local/{
            proxy_pass http://127.0.0.1:9527;
        }
        
        # 本地调试webpack hot功能的websockt
        location ^~/sockjs-node/{
            proxy_set_header X-Real_IP $remote_addr;
            proxy_set_header Host $host;
            proxy_set_header X_Forward_For $proxy_add_x_forwarded_for;
            proxy_http_version 1.1;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection 'upgrade';
            proxy_pass http://127.0.0.1:9527;
        }

        # BAP远程环境
        location /{
            proxy_pass http://10.30.5.36:8080/;
        }
            
                
    }

介绍到这里,熟悉umijs路由的同学会发现,原先我们的应用的产品态的路由,现在必须添加/app-local前缀了,否则前端的路由是没有办法定位到我们之前的功能的。直白点讲,之前我们的功能可能设置的路由为/appx/emp/list,那么我们现在开发阶段,必须修改为/app-local/appx/emp/list才能正常访问。同时,本地应用的所有静态资源,也必须添加/app-local前缀。幸好,umijs已经提供了合适的机制来实现这个功能。还是umijs的配置:

import routes from './routes';

let buildDevBase = `/app-local/`;
const isDev = process.env.NODE_ENV === 'development';

// ++ 功能点添加
const chainWebpack = (config) => {
  // 添加静态资源非根目录部署路径(开发阶段(yarn start:custom)umi的publicPath没有生效)
  if (process.env.NODE_ENV === 'development') {
    config.merge({
      output: {
        publicPath: buildDevBase,
      },
    });
  }
}

let appConfig = {
  treeShaking: true,
  routes,
  targets: {
    ie: 11,
  },
  plugins: [
    [
      'umi-plugin-react',
      {
        dva: {
          immer: true
        },
        dynamicImport: {
          webpackChunkName: true,
          loadingComponent: './platform/loading',
        },
        title: 'MES-APP',
        dll: false,
        routes: {
          exclude: [
            /models\//,
            /services\//,
            /model\.(t|j)sx?$/,
            /service\.(t|j)sx?$/,
            /components\//
          ],
        },
      },
    ]
  ],
  chainWebpack: chainWebpack
};

// ++ 功能点添加
if(isDev){
  appConfig = {...appConfig,base: buildDevBase,publicPath: buildDevBase}
}

export default appConfig;

其实以上的修改,说到底就是实现umijs应用非根目录部署过程。

那么本地开发过程中,第一步当然是通过本地9528端口,登录平台系统,然后切换到本地应用地址进行正常打开调试,此时已经是同域场景,接口调用也会直接代理到了平台,对其他模块页面访问也不再造成跨域问题了,似乎我们需要实现的功能都实现了。当然这个方案还是有待真实场景试验,比如sockjs-node/这样的路径也是在不断尝试过程中发现的。

后记

前端模块化、组件化和工程化,已经把前端开发过程变的不再那么简单,各种基础环境配置也被引入了前端开发过程,本文的http代理,就是个例子。但是我相信,本质还是不变的,无论是前端路由,还是React的虚拟Dom,还是nodejsdevServer都离不开之前的技术体系,话说回来了,前端基础还是那三把剑:html,css,javascript

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351