新项目用 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",
}

建议阅读:

参考:

最新 umi4-max 如何使用 webpack5 联邦模块的更多相关文章

  1. 手把手教你如何使用 webpack5 的模块联邦新特性

    想象一下,在webpack5还没出来前,前端使用第三方组件库,例如使用 dayjs 日期处理库,都是通过 npm i dayjs -s 安装 dayjs 模块到项目里,然后就可以通过 require ...

  2. python利用requests和threading模块,实现多线程爬取电影天堂最新电影信息。

    利用爬到的数据,基于Django搭建的一个最新电影信息网站: n1celll.xyz  (用的花生壳动态域名解析,服务器在自己的电脑上,纯属自娱自乐哈.) 今天想利用所学知识来爬取电影天堂所有最新电影 ...

  3. [转] npm 模块安装机制简介

    npm 是 Node 的模块管理器,功能极其强大.它是 Node 获得成功的重要原因之一. 正因为有了npm,我们只要一行命令,就能安装别人写好的模块 . $ npm install 本文介绍 npm ...

  4. npm 模块安装机制简介

    npm 是 Node 的模块管理器,功能极其强大.它是 Node 获得成功的重要原因之一. 正因为有了npm,我们只要一行命令,就能安装别人写好的模块 . $ npm install 本文介绍 npm ...

  5. Nginx的HTTP模块

    1.HTTP的核心模块.这些HTTP模块会在编译Nginx时自动编译进来,除非使用configure命令禁止编译这些模块.(1)alias指令.该指令用于在URL和文件系统路径之间实现映射.它与roo ...

  6. spring boot / cloud (十九) 并发消费消息,如何保证入库的数据是最新的?

    spring boot / cloud (十九) 并发消费消息,如何保证入库的数据是最新的? 消息中间件在解决异步处理,模块间解耦和,和高流量场景的削峰,等情况下有着很广泛的应用 . 本文将跟大家一起 ...

  7. Orchard详解--第七篇 拓展模块(译)

    Orchard作为一个组件化的CMS,它能够在运行时加载任意模块. Orchard和其它ASP.NET MVC应用一样,支持通过Visual Studio来加载已经编译为程序集的模块,且它还提供了自定 ...

  8. Python 中 logging 日志模块在多进程环境下的使用

    因为我的个人网站 restran.net 已经启用,博客园的内容已经不再更新.请访问我的个人网站获取这篇文章的最新内容,Python 中 logging 日志模块在多进程环境下的使用 使用 Pytho ...

  9. kivy学习一:安装kivy模块

    现在是看脸的时代,一个程序没有一个漂亮的UI,就像一个深闺中的美女没人欣赏. 当然作为一个小小.............白,没有那么高的要求,当前要先有脸是不? 首选python自家的模块tkinte ...

  10. npm模块安装机制简介

    npm是node的模块管理器,功能及其强大,它是node获得成功的重要原因之一. 正因为有了nom,我们只要一行命令,就能安装别人写好的模块. $ npm install 本文介绍npm模块安装机制的 ...

随机推荐

  1. linux配置定时备份mysql数据库

    1.创建备份目录 # mkdir /mydata/bak2.创建备份脚本 # vi /usr/sbin/bakmysql.sh #!/bin/bash #Name:bakmysql.sh #This ...

  2. 1903021126 申文骏 Java 第三周作业 编写代码及运行

    项目 内容 课程班级博客链接 19级信计班(本) 作业要求链接 第三周作业要求 博客名称 1903021126 申文骏 Java 第三周作业 编写代码及运行 要求 每道题要有题目,代码(使用插入代码, ...

  3. 基于Python接口自动化测试持续集成----在jenkins创建任务->检出git的代码-->生成报告-->发送邮件

    步骤一:先在jenkins创建一个自由风格的任务 步骤二:然后在源码管理选择git检出代码的方式,需要提供代码所在git的路径和登录git的账号和密码 步骤三:返回到任务配置的界面,先在构建后操作增加 ...

  4. javase我的笔记

    学习java编程 |--需要遵循其语法规范 |--在其开发环境下编程 准备开发java环境 |--安装jdk |--配置jdk jdk:含义是java提供的一套开发工具,包含jre\编译器等 |--编 ...

  5. jdkman(jdk版本管理工具)安装和使用(mac)

    1.安装jdkman 1.1.下载命令 curl -s "https://get.sdkman.io" | bash 执行后,sdkman安装到目录$HOME/.sdkman/,比 ...

  6. java_web----1

    一.HTML 1.网页的构成 结构:HTML 超文本标记语言  标签语言 标签写法: <标签名> </标签名> <标签名/> 表现:CSS 行为:JavaScrip ...

  7. Linux系列---【设置ssh连接超时时间】

    设置ssh连接超时时间 shell工具总是不断的超时,上个厕所就断开了,很不方便,这里根据自己习惯设置一下. echo "export TMOUT=600" >> /e ...

  8. 2003031118-李伟-Python数据分析第三周作业-第一次作业

    项目 NumPy数值计算基础 博客名称 2003031118-李伟-Python数据分析第三周作业-第一次作业 课程班级博客链接 https://edu.cnblogs.com/campus/pexy ...

  9. Centos8 中安装GitLab

    Centos8 中安装GitLab 1,安装依赖 yum install -y curl policycoreutils-python openssh-server centos8没有policyco ...

  10. java的3中代理模式

    代理模式是什么 代理模式是一种设计模式,简单说即是在不改变源码的情况下,实现对目标对象的功能扩展. 比如有个歌手对象叫Singer,这个对象有一个唱歌方法叫sing(). 1 public class ...