Vite模块联邦(vite-plugin-federation)实现去中心化微前端后台管理系统架构

一、项目概述

1.1 前言

在现代大型前端项目开发中,多团队协作时往往面临代码隔离与集成的挑战。为了解决这一问题,我们需要一种能够让各个微前端模块既能独立开发部署,又能作为完整系统一部分的解决方案。基于Vite的模块联邦插件@originjs/vite-plugin-federation提供了一种去中心化的微前端架构实现方式,实现了组件、路由的跨应用共享和动态加载。本文将结合实际项目经验,详细介绍如何利用模块联邦技术构建真正去中心化的微前端架构。

1.2 依赖版本要求

node: 18.20.5

json 复制代码
{
  "vite": "^6.0.2",
  "@originjs/vite-plugin-federation": "1.4.1",
}

二、架构设计

2.1 模块联邦基础概念

2.1.1 什么是模块联邦

@originjs/vite-plugin-federation是Vite生态中实现模块联邦功能的插件,它允许多个独立应用在运行时共享模块和代码,无需重复构建或打包依赖。这一特性在去中心化微前端架构中尤为关键。

2.1.2 基本配置示例

一个微前端模块的典型配置如下:

typescript 复制代码
federation({
  name: "pageAModule",        // 微前端模块名称
  filename: "pageAEntry.js",  // 入口文件名
  exposes: {                  // 暴露的模块,此处为路由
    './routes': './src/routes/index.ts'
  },
  remote: {
    menuModule: ModuleUrl, // 引入的远程模块
  }
  shared: ['vue', 'vue-router', 'pinia']  // 共享依赖
})
// 使用远程模块
import 'menuModule/xxx'

2.2 去中心化微前端架构设计

2.2.1 传统微前端 vs 去中心化微前端

传统微前端架构通常采用"基座模式",即一个主应用作为基座控制其他子应用的加载和渲染。而去中心化微前端架构没有明确的主应用,各个子应用可以独立部署、独立运行,同时又能无缝协作。

去中心化微前端架构的主要特点:

  1. 无中心基座:没有固定的主应用控制全局状态
  2. 平等协作:每个应用都可作为入口,并动态加载其他应用模块
  3. 路由共享机制:所有微前端模块共享路由表,实现无缝导航
  4. 共享运行时:应用间共享关键依赖和状态

2.3 特殊架构设计:menuModule的角色

虽然我们采用去中心化的微前端架构,但在后台管理系统中,我们仍然需要一个统一的菜单管理模块(menuModule)。来提供系统的layout和管理一些公共文件。

2.3.1 为什么需要menuModule?

  • 统一的菜单管理:后台管理系统需要一个统一的菜单体系,确保用户体验的一致性
  • 权限控制中心:集中管理各个页面模块的访问权限
  • 导航状态维护:统一处理菜单的激活状态、展开状态等
  • 系统级功能集成:包含用户信息、全局设置等系统级功能

menuModule模块需要在所有的页面项目中引入,且所有项目在vite的federation中都只需引入这一个menuModule模块就行。 其他剩余的模块引入方式,在下文有提到。

menuModule承担以下职责:

  • 提供统一的布局容器(Layout)
  • 管理全局导航菜单
  • 处理用户认证和权限
  • 提供公共组件和工具函数

三、技术实现

3.1 模块联邦配置详解

在实际项目中,每个页面的微前端模块的联邦配置通常包含以下部分:

typescript 复制代码
federation({
  name: config.projectPrefix + 'Module',      // 微前端模块名称
  filename: config.projectPrefix + 'Entry.js', // 入口文件名
  exposes: {
    './routes': './src/routes/index.ts',      // 暴露路由配置(关键)
  },
  remotes: {
    // 后台管理系统中需要单独引入菜单模块
    menuModule: ModuleUrl,
  },
  shared: ['vue', 'vue-router', 'pinia', 'dayjs', 'axios', 'sass', 'element-plus'], // 共享依赖
})

3.1.1 关键配置参数解析

  1. name:微前端模块的唯一标识符,其他模块通过此名称引用
  2. filename:构建后的入口文件名,通常以Entry.js结尾
  3. exposes:暴露的模块,主要是路由配置,这是实现去中心化的关键
  4. remotes:需要引用的其他微前端模块,指定模块的URL或加载方式
  5. shared:所有微前端模块间共享的依赖,确保单一实例,避免重复加载

3.2 构建时处理

在构建微前端模块时,vite-plugin-federation会进行以下处理:

  1. 识别exposes配置中声明的路由模块
  2. 为每个模块生成独立的构建产物
  3. 创建容器模块,管理模块的导出和依赖关系
  4. 生成远程入口文件,处理模块的加载和解析

四、动态模块的导入与路由整合

4.1 传统导入方式的局限

传统的模块联邦使用方式是预先声明远程模块并直接导入:

typescript 复制代码
// 常规的模块联邦导入方法
import Component from 'remoteModule/Component'

这种方式存在一个重要限制:无法使用动态字符串拼接模块名。例如,以下代码在模块联邦中不起作用:

typescript 复制代码
// 这种写法在模块联邦中不支持
const moduleName = 'remoteModule'
import Component from `${moduleName}/Component` // 不支持!

这样就会导致一个问题就是 在去中心化微前端架构中,每个项目模块在开发时并不知道全局系统中到底有多少个联邦模块,也无法预先确定所有模块的名称和地址。为了支持新增模块的灵活扩展,需要一个动态的机制来发现和加载模块。通常,我们会通过一个配置文件(如 registry.json)来集中管理所有联邦模块的注册信息,允许新模块随时加入系统。然而,官方的模块联邦导入方式,它不支持动态拼接模块名称的字符串。

为了解决这一问题,我实现了一个支持动态拼接模块名称的加载函数 getRemoteEntries。通过该函数,我们可以在运行时根据配置文件动态获取模块的 URL 并加载模块,从而实现真正的动态模块发现和集成。这种方式不仅解决了官方导入方式的限制,还为系统的扩展性和灵活性提供了强有力的支持。

4.2 动态模块实现

为解决上述限制,我实现了getRemoteEntries函数,通过@originjs/vite-plugin-federationgetAPI实现,运行时动态的模块声明和加载:

typescript 复制代码
// src/utils/federation.ts
export const getRemoteEntries = async (name?: string, moduleName?: string): Promise<any[]> => {
  try {
    // 从注册中心获取所有可用模块的信息
    const response = await axios.get(`${baseUrl}/federation/registry.json`)

    // 运行时根据条件筛选模块
    const filteredModules = response.data.filter(module => !name || module.name.includes(name))

    // 动态加载匹配的模块
    const loadedComponents = []
    for (const moduleInfo of filteredModules) {
      // 动态构建模块URL
      const moduleUrl = buildModuleUrl(moduleInfo)

      // 使用模块联邦API动态加载模块
      const remote = await import(/* @vite-ignore */ moduleUrl)

      // 加载特定模块
      const moduleFactory = await remote.get('./' + moduleName)
      const module = await moduleFactory()

      loadedComponents.push(module)
    }

    return loadedComponents
  } catch (error) {
    console.error('获取远程模块失败:', error)
    return []
  }
}

这种动态模块发现机制的优势在于:

  1. 运行时集成:应用可以在运行时发现并加载其他模块
  2. 可扩展性:新模块可以随时添加到系统中,无需修改全部现有模块
  3. 自动注册:新增的微前端模块自动成为整体系统的一部分
  4. 动态模块名:支持通过字符串拼接方式构建模块URL

4.3 路由表的共享与整合

共享路由表是去中心化架构的核心,使每个模块都能作为独立入口。在我们的实现中,路由整合分两步进行:

  1. 路由收集:收集所有微前端模块的路由配置
  2. 路由合并:将所有路由合并到主布局路由下
typescript 复制代码
// src/routes/index.ts
export const createRouterInstance = async () => {
  // 存储布局路由
  let layoutRoute: RouteRecordRaw | undefined

  if (ENV_TYPE === 'federation') {
    // 获取远程入口配置
    const remoteRoutes = await getRemoteEntries('', 'routes')
    // 找到Layout路由并合并
    layoutRoute = mergeLayoutChildren(remoteRoutes || [])
  }

  // 创建路由实例
  const router = createRouter({
    history: createWebHashHistory(),
    routes: ENV_TYPE === 'federation' ? (layoutRoute as any).default : baseRoutes,
  })

  // 全局前置守卫
  router.beforeEach(async (to, from, next) => {
    // 路由守卫逻辑...
    next()
  })

  return router
}

4.4 路由合并核心逻辑

路由合并的核心实现在mergeLayoutChildren函数中,它实现了以下功能:

  1. 识别布局路由:找到包含主布局的路由模块
  2. 处理独立页面:将需要独立展示的页面(outLayout)提取出来
  3. 嵌套路由合并:将所有子模块的路由合并到主布局的children中
  4. 顶层路由保留:保留顶层路由如登录页、404页面等
typescript 复制代码
// 合并路由
const mergeLayoutChildren = (data: RouteRecordRaw[]) => {
  let layoutIndex = -1 // 添加变量记录Layout的索引
  const outLayoutItems: { moduleIndex: number; route: any }[] = [] // 存储所有outLayoutItem及其位置

  // 首先处理 baseRoutes 中的 outLayout 路由
  const baseOutLayoutRoutes = baseRoutes.filter((route) => route.meta?.outLayout)
  baseOutLayoutRoutes.forEach((route) => {
    outLayoutItems.push({ moduleIndex: -1, route })
  })

  // 遍历远程数据处理布局路由和独立路由
  data.forEach((item, index) => {
    const defaultRoutes = (item as any).default
    const layoutItem = defaultRoutes.find((route: any) => route.name === 'Layout')
    const outLayoutItem = defaultRoutes.find((route: any) => route.meta?.outLayout)

    if (layoutItem) {
      layoutIndex = index // 记录Layout所在的索引
    }

    if (outLayoutItem) {
      outLayoutItems.push({ moduleIndex: index, route: outLayoutItem })
      // 从原数组中移除outLayoutItem避免重复
      const outLayoutIndex = defaultRoutes.findIndex((route: any) => route.meta?.outLayout)
      if (outLayoutIndex > -1) {
        defaultRoutes.splice(outLayoutIndex, 1)
      }
    }
  })

  // 获取Layout路由作为合并基础
  const layoutRoute = (data[layoutIndex] as any).default.find((route: any) => route.name === 'Layout')

  if (layoutRoute) {
    // 将所有outLayoutItem添加到data[layoutIndex].default中作为顶层路由
    outLayoutItems.forEach((item) => {
      ;(data[layoutIndex] as any).default.push(item.route)
    })

    if (layoutRoute.children) {
      // 将主应用中非outLayout的路由添加到Layout的children中
      const nonOutLayoutBaseRoutes = baseRoutes.filter((route) => !route.meta?.outLayout)
      layoutRoute.children.push(...nonOutLayoutBaseRoutes)

      // 遍历所有其他模块的路由添加到Layout的children中
      data.forEach((item, index) => {
        if (index !== layoutIndex) {
          const routes = (item as any).default
          if (Array.isArray(routes)) {
            layoutRoute.children.push(...routes)
          }
        }
      })
    }
  }

  return data[layoutIndex]
}

五、状态管理与模块通信

在去中心化微前端架构中,各模块间的状态管理和通信是一个关键问题。本项目采用了基于pinia持久化缓存的状态共享机制,主要通过src/stores/modules/cache.ts实现:

5.1 基于缓存的状态管理

我们使用了专门的缓存存储模块来管理跨模块的状态:

typescript 复制代码
// stores/modules/cache.ts
import { defineStore } from 'pinia'
import { ref } from 'vue'
import config from '@/config'

interface CacheData {
  value: unknown
  expire?: number
}

export const useCacheStore = defineStore(
  'cache',
  () => {
    // 状态定义
    const prefix = ref<string>(config.cachePrefix)
    const cacheData = ref<Record<string, CacheData>>({})

    // 获取当前时间戳
    const getTime = (): number => {
      return Math.round(new Date().getTime() / 1000)
    }

    // 获取完整的键名
    const getKey = (key: string, customPrefix?: string): string => {
      return (customPrefix ?? prefix.value) + key
    }

    // 设置缓存数据
    const set = (key: string, value: unknown, expire?: number, customPrefix?: string) => {
      const fullKey = getKey(key, customPrefix)
      cacheData.value[fullKey] = {
        value,
        expire: expire ? getTime() + expire : undefined,
      }
    }

    // 获取缓存数据
    const get = (key: string, customPrefix?: string) => {
      const fullKey = getKey(key, customPrefix)
      const data = cacheData.value[fullKey]

      if (!data) return null

      if (data.expire && data.expire < getTime()) {
        remove(key, customPrefix)
        return null
      }

      return data.value
    }

    // 删除缓存数据
    const remove = (key: string, customPrefix?: string) => {
      const fullKey = getKey(key, customPrefix)
      delete cacheData.value[fullKey]
    }

    return {
      prefix,
      cacheData,
      set,
      get,
      remove,
      getKey,
      getTime,
    }
  },
  {
    persist: {
      storage: localStorage,
    },
  }
)

5.2 缓存模块的核心特性

缓存存储模块具有以下特点:

  1. 命名空间隔离:通过前缀机制,确保不同模块间缓存键名不冲突
  2. 过期时间控制:支持设置数据过期时间,自动清理失效数据
  3. 本地持久化:使用Pinia的持久化功能,确保页面刷新后状态不丢失
  4. 跨模块共享:所有微前端模块都可以访问同一缓存存储,实现数据共享

5.3 模块间通信方式

基于缓存存储的通信机制主要有以下几种模式:

  1. 直接状态共享:模块间通过相同的缓存键读写数据

    typescript 复制代码
    // 页面A中设置数据
    const cacheStore = useCacheStore()
    cacheStore.set('sharedData', { count: 1 })
    
    // 页面B中读取数据
    const cacheStore = useCacheStore()
    const data = cacheStore.get('sharedData')
  2. 基于过期时间的临时共享:对于临时性数据,设置合适的过期时间

    typescript 复制代码
    // 设置1小时过期的共享数据
    cacheStore.set('temporaryData', value, 3600)
  3. 跨应用的全局状态:通过特定前缀获取不同微前端的状态

    typescript 复制代码
    // 获取user微前端模块的状态
    cacheStore.set('userInfo', settings, undefined, 'user')
    
    // 读取user微前端模块的状态
    const settings = cacheStore.get('userInfo', 'user')

六、项目结构与配置

6.1 微前端模块文件架构

scss 复制代码
├── 菜单模块 (被当前模块引用)
│   ├── vite.config.ts (模块联邦配置)
│   ├── src/
│   │   ├── layout/
│   │   ├── routes/
│   │   │   └── index.ts (导出路由配置)
│   │   ├── views/ (页面组件)
│   │   ├── stores/ (状态管理)
│   │   └── utils/
│   │       └── federation.ts (模块加载器)
│   └── package.json
├── user模块(页面模块)
│   ├── vite.config.ts (模块联邦配置)
│   ├── src/
│   │   ├── routes/
│   │   │   └── index.ts (导出路由配置)
│   │   ├── login/ (登录组件)
│   │   ├── views/ (页面组件)
│   │   ├── stores/ (状态管理)
│   │   └── utils/
│   │       └── federation.ts (模块加载器)
│   └── package.json
├── xxx模块(页面模块)
│   ├── vite.config.ts (模块联邦配置)
│   ├── src/
│   │   ├── routes/
│   │   │   └── index.ts (导出路由配置)
│   │   ├── views/ (页面组件)
│   │   ├── stores/ (状态管理)
│   │   └── utils/
│   │       └── federation.ts (模块加载器)
│   └── package.json
│

6.2 完整的vite.config.ts示例

typescript 复制代码
// vite.config.ts
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'
import { federation } from '@originjs/vite-plugin-federation'
import path from 'path'

import { config } from './config'

const baseUrl = process.env.VITE_APP_BASE_URL || ''

// 需要暴露的模块
const exposes: Record<string, string> = {
  // 暴露路由表,供其他应用使用
  './routes': './src/routes/index.ts',
}

export default defineConfig({
  // 基础配置
  base: baseUrl || './', // 通过构建脚本传入实际静态文件地址,所有联邦模块文件都会遵循这个地址
  // 插件配置
  plugins: [
    vue(),

    // 模块联邦配置
    federation({
      name: config.projectPrefix + 'Module', // 微前端模块名称
      filename: config.projectPrefix + 'Entry.js', // 入口文件名
      exposes, // 暴露的模块
      remotes: {
        // 只需单独引入menuModule,其他的模块走federation
        menuModule: ModuleUrl,
      },
      // 共享依赖配置
      shared: [
        'vue',
        'vue-router',
        'pinia',
        'dayjs',
        'axios',
        'sass',
      ],
    })
  ],
  // 解析配置
  resolve: {
    alias: {
      '@': path.resolve(__dirname, 'src')
    }
  },
})

6.3 部署与注册机制

去中心化微前端架构采用集中式的模块注册机制,通过registry.json文件管理所有可用模块:

scss 复制代码
CDN/静态资源服务器
├── federation/
│   └── registry.json (模块注册信息)
│
├── front-pc-page/ (页面模块)
│   └── pageEntry.js (入口文件)
│
├── front-pc-menu/ (菜单模块)
    └── menuEntry.js (菜单模块入口)

registry.json示例:

json 复制代码
[
  {
    "name": "pageModule",
    "path": "front-pc-page"
  },
  {
    "name": "menuModule",
    "path": "front-pc-menu"
  }
]

七、总结

@originjs/vite-plugin-federation为实现去中心化微前端架构提供了强大的技术基础。通过共享路由表机制,我们可以构建出真正去中心化、高度灵活且可扩展的微前端系统,使得不同团队可以独立开发、测试和部署自己的模块,同时保持系统整体的一致性和可用性。

相关推荐
LaughingZhu16 分钟前
PH热榜 | 2025-04-09
前端·数据库·人工智能·mysql·开源
枫super29 分钟前
Day-03 前端 Web-Vue & Axios 基础
前端·javascript·vue.js
程序猿chen1 小时前
Vue.js组件安全工程化演进:从防御体系构建到安全性能融合
前端·vue.js·安全·面试·前端框架·跳槽·安全架构
你也来冲浪吗1 小时前
MD编辑器用法讲解
前端
小小小小宇1 小时前
十万字总结所有React hooks(含简单原理)
前端
MariaH1 小时前
MySQL数据库DQL
前端
Enjoy10241 小时前
v8垃圾回收机制
前端
Georgewu1 小时前
【HarmonyOS 5】敏感信息本地存储详解
前端·harmonyos
_Le_1 小时前
css 小师系列:一种新的影响样式优先级的方式😍
前端·css
wordbaby1 小时前
从前端视角看 MCP:解锁 LLM 工具调用与结构化交互
前端