经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JS/JS库/框架 » webpack » 查看文章
最新 umi4-max 如何使用 webpack5 联邦模块
来源:cnblogs  作者:ESnail  时间:2023/3/20 8:41:46  对本文有异议

新项目用 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 导出项目

配置要导出的功能模块

  1. // 配置文件
  2. const { ModuleFederationPlugin } = require("webpack").container;
  3. const packageDeps = require('../package.json').dependencies
  4. new ModuleFederationPlugin({
  5. name: "app1",
  6. filename: "remoteEntry.js",
  7. // 表示导出的模块,只有在此申明的模块才可以作为远程依赖被使用。
  8. exposes: {
  9. "./CounterAppOne": "./src/components/CounterAppOne",
  10. },
  11. shared: {
  12. ...packageDeps,
  13. react: { singleton: true, eager: true, requiredVersion: deps.react },
  14. "react-dom": {
  15. singleton: true,
  16. eager: true,
  17. requiredVersion: deps["react-dom"],
  18. },
  19. "react-router-dom": {
  20. singleton: true,
  21. eager: true,
  22. requiredVersion: deps["react-router-dom"],
  23. },
  24. },
  25. })

1.2 导入项目

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

  1. // 配置文件
  2. const { ModuleFederationPlugin } = require("webpack").container;
  3. const packageDeps = require('../package.json').dependencies
  4. new ModuleFederationPlugin({
  5. name: "container",
  6. // 将其它项目的 name 映射到当前项目中
  7. remotes: {
  8. app1: 'app1@http://localhost:3001/remoteEntry.js',
  9. },
  10. // 是非常重要的参数,制定了这个参数,可以让远程加载的模块对应依赖改为使用本地项目的 React 或 ReactDOM。
  11. shared: {
  12. ...packageDeps,
  13. react: { singleton: true, eager: true, requiredVersion: deps.react },
  14. "react-dom": {
  15. singleton: true,
  16. eager: true,
  17. requiredVersion: deps["react-dom"],
  18. },
  19. "react-router-dom": {
  20. singleton: true,
  21. eager: true,
  22. requiredVersion: deps["react-router-dom"],
  23. },
  24. },
  25. })

react 项目中使用

  1. // 通过 webpack 关联其它应用,然后按需加载
  2. const CounterAppOne = React.lazy(() => import("app1/CounterAppOne"))
  3. export default () => {
  4. return (
  5. <React.Suspense fallback={<div>Loading</div>}>
  6. <CounterAppOne />
  7. </React.Suspense>
  8. )
  9. }

2. umi3 项目

2.1 导出项目

配置

  1. // .umirc.ts
  2. publicPath:'http://127.0.0.1:5502/',
  3. webpack5: {}, // 开启 webpack5
  4. chainWebpack: (config) => {
  5. config.output.publicPath('auto'); // 路径处理,保证导入项目路径正确
  6. const { ModuleFederationPlugin } = require("webpack").container;
  7. const packageDeps = require('./package.json').dependencies
  8. config.plugin('mf').use(ModuleFederationPlugin, [{
  9. name: "app1",
  10. filename: 'remoteEntry.js',
  11. exposes: {
  12. "./Test": '@/pages/test.tsx',
  13. },
  14. shared: { react: { eager: true }, "react-dom": { eager: true } },
  15. }])
  16. return config;
  17. }

2.2 导入项目

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

插件内容:

  1. import { IApi } from 'umi';
  2. import { resolve } from 'path';
  3. import { readFileSync } from 'fs';
  4. export default (api: IApi) => {
  5. api.onGenerateFiles(() => {
  6. const buffer= readFileSync(resolve('./src/.umi/umi.ts'))
  7. const c = String(buffer)
  8. // console.log()
  9. api.writeTmpFile({
  10. path: 'index.ts',
  11. content: c,
  12. });
  13. api.writeTmpFile({
  14. path: 'umi.ts',
  15. content: 'import("./index")',
  16. });
  17. });
  18. };

配置

  1. // .umirc.ts
  2. dynamicImport:{},
  3. webpack5: {}, // 开启 webpack5
  4. chainWebpack: (config) => {
  5. const { ModuleFederationPlugin } = require("webpack").container;
  6. config.plugin('mf').use(ModuleFederationPlugin, [{
  7. name: "app2",
  8. remotes: {
  9. "app1": "app1@http://127.0.0.1:5502/dist/remoteEntry.js",
  10. },
  11. shared: { react: { eager: true }, "react-dom": { eager: true } },
  12. }])
  13. return config;
  14. }

使用和普通项目一致

3. umi4-max 项目

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

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

  1. // .umirc.ts
  2. mf: {
  3. name: 'app2',
  4. remotes: [
  5. {
  6. name: 'app1',
  7. entry: 'http://127.0.0.1:5502/dist/remoteEntry.js'
  8. },
  9. ],
  10. shared: { react: { eager: true }, "react-dom": { eager: true } },
  11. }

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

其他相关实现:

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

4. vite 项目

安装插件 vite-plugin-federation

4.1 导出项目

配置

  1. // vite.config.js 或 rollup.config.js
  2. import federation from "@originjs/vite-plugin-federation";
  3. export default {
  4. plugins: [
  5. federation({
  6. name: "remote-app",
  7. filename: "remoteEntry.js",
  8. // 需要暴露的模块
  9. exposes: {
  10. "./Button": "./src/Button.vue",
  11. },
  12. shared: ["vue"],
  13. }),
  14. ],
  15. };

4.2 导入项目

配置

  1. // vite.config.js 或 rollup.config.js
  2. import federation from "@originjs/vite-plugin-federation";
  3. export default {
  4. plugins: [
  5. federation({
  6. name: "host-app",
  7. remotes: {
  8. remote_app: "http://localhost:5001/assets/remoteEntry.js",
  9. },
  10. shared: ["vue"],
  11. }),
  12. ],
  13. };

react 项目中使用

  1. // dynamic import
  2. const myButton = React.lazy(() => import('remote/myButton'))
  3. // static import
  4. import myButton from 'remote/myButton'

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

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

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

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

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

解决:

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

  1. // 本地模块的 vite.config.ts
  2. export default defineConfig({
  3. server:{
  4. https: "http",
  5. host: "192.168.56.1",
  6. port: 5100,
  7. },
  8. cacheDir: "node_modules/.cacheDir",
  9. }

建议阅读:

参考:

原文链接:https://www.cnblogs.com/EnSnail/p/17234326.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号