首页 > 其他分享 >最新 umi4-max 如何使用 webpack5 联邦模块

最新 umi4-max 如何使用 webpack5 联邦模块

时间:2023-03-19 21:25:10浏览次数:52  
标签:webpack5 name 项目 max umi4 react eager config true

新项目用 umi4-max 搭建,部分功能想要使用其他项目的功能,不想重新开发,想到了使用 webpack5 的联邦模块,可以直接引用其他项目代码来实现共享代码。

理想很美好,现实很残酷。直接按照 webpack5 联邦模块的使用方法,并不能成功,而官方文档没有明确说明如何使用。

webpack 联邦模块如何使用呢?

理解:

  • 使用场景:项目A有一个功能,项目B也想用。此时可以用。
  • 使用前提:依赖 webpack5,且主要依赖相同(如都依赖react)

说明:项目A需要用项目B的代码,项目A为导入项目,项目B为导出项目。

相关配置字段说明:

字段名 类型 含义
name string 必传值,即输出的模块名,被远程引用时路径为 name/{name}/name/
library string 声明全局变量的方式,name 为 umd 的 name
filename string 构建输出的文件名
remotes object 远程引用的应用名及其别名的映射,使用时以 key 值作为 name
exposes object 被远程引用时可暴露的资源路径及其别名
shared object 与其他应用之间可以共享的第三方依赖,使你的代码中不用重复加载同一份依赖

1. 普通项目

1.1 导出项目

配置要导出的功能模块

// 配置文件

const { ModuleFederationPlugin } = require("webpack").container;
const packageDeps = require('../package.json').dependencies

new ModuleFederationPlugin({
  name: "app1",
  filename: "remoteEntry.js",
  // 表示导出的模块,只有在此申明的模块才可以作为远程依赖被使用。
  exposes: {
    "./CounterAppOne": "./src/components/CounterAppOne",
  },
  shared: {
    ...packageDeps,
    react: { singleton: true, eager: true, requiredVersion: deps.react },
    "react-dom": {
      singleton: true,
      eager: true,
      requiredVersion: deps["react-dom"],
    },
    "react-router-dom": {
      singleton: true,
      eager: true,
      requiredVersion: deps["react-router-dom"],
    },
  },
})

1.2 导入项目

配置要导入的功能模块的文件地址

// 配置文件

const { ModuleFederationPlugin } = require("webpack").container;
const packageDeps = require('../package.json').dependencies

new ModuleFederationPlugin({
  name: "container",
  // 将其它项目的 name 映射到当前项目中
  remotes: {
    app1: 'app1@http://localhost:3001/remoteEntry.js',
  },
  // 是非常重要的参数,制定了这个参数,可以让远程加载的模块对应依赖改为使用本地项目的 React 或 ReactDOM。
  shared: {
    ...packageDeps,
    react: { singleton: true, eager: true, requiredVersion: deps.react },
    "react-dom": {
      singleton: true,
      eager: true,
      requiredVersion: deps["react-dom"],
    },
    "react-router-dom": {
      singleton: true,
      eager: true,
      requiredVersion: deps["react-router-dom"],
    },
  },
})

react 项目中使用

// 通过 webpack 关联其它应用,然后按需加载
const CounterAppOne = React.lazy(() => import("app1/CounterAppOne"))

export default () => {
  return (
  	<React.Suspense fallback={<div>Loading</div>}>
      <CounterAppOne />
    </React.Suspense>
  )
}

2. umi3 项目

2.1 导出项目

配置

// .umirc.ts

publicPath:'http://127.0.0.1:5502/',
webpack5: {}, // 开启 webpack5 
chainWebpack: (config) => {
  config.output.publicPath('auto'); // 路径处理,保证导入项目路径正确

  const { ModuleFederationPlugin } = require("webpack").container;
  const packageDeps = require('./package.json').dependencies

  config.plugin('mf').use(ModuleFederationPlugin, [{
    name: "app1",
    filename: 'remoteEntry.js',
    exposes: {
      "./Test": '@/pages/test.tsx',
    },
    shared: { react: { eager: true }, "react-dom": { eager: true } },
  }])
  return config;
}

2.2 导入项目

安装插件:yarn install umi-plugin-mf-bootstrap 支持入口异步导入,以便支持使用 hooks。如果不安装,会报错 Uncaught Error: Shared module is not available for eager consumption。具体原因为:违背了 hooks 的使用规则,不能用两个 React 实例。

插件内容:

import { IApi } from 'umi';
import { resolve } from 'path';
import { readFileSync } from 'fs';

export default (api: IApi) => {

  api.onGenerateFiles(() => {
    const buffer= readFileSync(resolve('./src/.umi/umi.ts'))
    const c = String(buffer)
    // console.log()
    api.writeTmpFile({
      path: 'index.ts',
      content: c,
    });
    api.writeTmpFile({
      path: 'umi.ts',
      content: 'import("./index")',
    });
  });
};

配置

// .umirc.ts

dynamicImport:{},
webpack5: {}, // 开启 webpack5
chainWebpack: (config) => {
  const { ModuleFederationPlugin } = require("webpack").container;
  config.plugin('mf').use(ModuleFederationPlugin, [{
    name: "app2",
    remotes: {
      "app1": "app1@http://127.0.0.1:5502/dist/remoteEntry.js",
    },
    shared: { react: { eager: true }, "react-dom": { eager: true } },
  }])
  return config;
}

使用和普通项目一致

3. umi4-max 项目

按照 umi3 的方案,没有成功。多方查阅摸索后,最终通过查阅官方 github 代码,看到有个插件中有个 mf 的文件,阅读代码后,摸索出最终的方案了。

导出项目配置和 umi3 的一致,而导入项目只需按照下面的配置即可,使用和普通项目一致。

// .umirc.ts

mf: {
  name: 'app2',
  remotes: [
    {
      name: 'app1',
      entry: 'http://127.0.0.1:5502/dist/remoteEntry.js'
    },
  ],
  shared: { react: { eager: true }, "react-dom": { eager: true } },
}

关于 mf 插件的详细使用:可参考官方 github 代码 Module Federation 插件,后来找到的。

其他相关实现:

  • 源码 mf 插件实现 mf
  • 源码 mfsu mf实现 MFImport

4. vite 项目

安装插件 vite-plugin-federation

4.1 导出项目

配置

// vite.config.js 或 rollup.config.js

import federation from "@originjs/vite-plugin-federation";
export default {
  plugins: [
    federation({
      name: "remote-app",
      filename: "remoteEntry.js",
      // 需要暴露的模块
      exposes: {
        "./Button": "./src/Button.vue",
      },
      shared: ["vue"],
    }),
  ],
};

4.2 导入项目

配置

// vite.config.js 或 rollup.config.js

import federation from "@originjs/vite-plugin-federation";
export default {
  plugins: [
    federation({
      name: "host-app",
      remotes: {
        remote_app: "http://localhost:5001/assets/remoteEntry.js",
      },
      shared: ["vue"],
    }),
  ],
};

react 项目中使用

// dynamic import
const myButton = React.lazy(() => import('remote/myButton'))

// static import
import myButton from 'remote/myButton'

备注:React 使用 federation 问题解决:

建议查看这个 Issue,里面包含了大多数 React 相关的问题

常见问题:远程模块加载本地模块的共享依赖失败,报错:

localhost/:1 Uncaught (in promise) TypeError: Failed to fetch dynamically imported module: http:your url

原因:Vite 在启动服务时对于 IP、Port 有自动获取逻辑,在 Plugin 中还没有找到完全对应的获取逻辑,在部分情况下可能会出现获取失败。

解决:

在本地模块显式到声明 IP、Port、cacheDir,保证我们的 Plugin 可以正确的获取和传递依赖的地址。

// 本地模块的 vite.config.ts

export default defineConfig({
  server:{
    https: "http",
    host: "192.168.56.1",
    port: 5100,
  },
  cacheDir: "node_modules/.cacheDir",
}

建议阅读:

参考:

标签:webpack5,name,项目,max,umi4,react,eager,config,true
From: https://www.cnblogs.com/EnSnail/p/17234326.html

相关文章

  • Expectation-Maximization algorithm
    1.IntroductionTheExpectation-Maximization(EM)algorithmisawidelyusedstatisticalalgorithmformaximumlikelihoodestimationincaseswherethereismi......
  • maxwell直线电机仿真疑难汇总
     一.maxwell2D绕组电流激励问题1.Numberofconductors:线圈导线数一个线圈的导线根数不一定就是匝数,只有并绕根数等于1时,一个线圈的导线根数才等于线圈的匝数。有如......
  • umi4查看webpack配置
    请按照以下步骤进行操作:打开.umirc.ts文件并添加chainWebpack方法。使用console.log()创建一个带有字符串消息的log语句。在console.log()语句后添加以下行以......
  • java8新特性-引用流-max,min
    例子:List<User>users=newArrayList<>();users.add(newUser("张三",30));users.add(newUser("李四",34));users.add(newUser("王五",20));......
  • 实际线程数量大于maximumPoolSize
    在做练习的时候出现了下面的状况注:只是自己的推理哈,如果后面学会了看源码再补充和修改首先复习两个知识:临时线程什么时候创建?新任务提交时,发现核心线程都在忙、任......
  • mysql: Specified key was too long; max key length is 767 bytes
    问题记录:原因如果该字段参与了索引,在对该字段进行拓展长度时会提示超过索引最大值我使用的解决方案,在使用联合索引时使用改字段的前一部分作为联合索引\然后再......
  • 3Dmax和C4D有什么区别
    3Dmax和C4D有什么区别老杨爱说2022-11-2113:02吉林关注 3Dmax和C4D的区别有:  一、性质不同  1、3Dmax:3Dmax作为一个非常经典的三维软件,......
  • vmax vmin
    <body><divclass="boxbox1">box1</div><divclass="boxbox2">box2</div><divclass="boxbox3">box3</div></body>.box1{width:10vw;height:10vh;......
  • MaxCompute SQL使用小技巧之时间日期处理
    --当月第一天SELECTdateadd(lastday(dateadd(getdate(),-1,'mm')),1,'dd');--当月最后一天SELECTlastday(getdate());--上季度第一天SELECTdateadd(to_date......
  • 跟着思兼学习Klipper(21)新一代全能选手: Octopus Max EZ 简要使用记录
    前言本文感谢必趣(BigtreeTech)赞助的OctopusMAXEZ主板,据说是美女一手自用。既往没有系统了解过必趣系的主板,这次和网友们一起一探究竟。【KlipperBoxConfigs配......