Monorepo单仓多包&独立部署

仓就是仓库(repository,简称 repo),通常我们使用多个仓库(简称多仓,multi-repo)来管理项目代码,也就是每个仓库负责一个模块或包的编码、构建、测试和发布,代码规模相对较小,所以也称为小型规模仓库(简称小仓)。而单一(mono)仓库(简称单仓,mono-repo)是指在一个仓库中管理多个模块或包,当代码规模达到一定程度后可称为大型规模仓库(简称大仓)。

1. 目录结构「monorepo」

bash 复制代码
project-root/
├── packages/                      # 公共模块目录(跨子项目共享)
│   └── shared/                   # 公共模块(可跨子项目复用)
│       ├── api/                  # 公共接口封装
│       ├── utils/                # 工具函数、通用方法
│       ├── components/           # 公共 Vue/React 组件
│       └── styles/               # 公共样式文件,例如变量、mixin等
│
├── apps/                          # 子项目目录,每个子项目独立打包部署
│   ├── mobile/                    # mobile 子项目
│   │   ├── assets/               # 页面专属静态资源(图片、图标等)
│   │   ├── router/               # 页面专属路由配置
│   │   ├── store/                # 页面专属状态管理(vuex、pinia 等)
│   │   ├── views/                # 页面视图组件
│   │   ├── App.vue               # 根组件
│   │   └── main.js               # 入口文件,挂载 App.vue 并初始化子项目        
│   │
│   └── ...                       # 其他子项目,结构同 home/about
│
├── build/                         # webpack 配置、打包脚本、构建工具等
├── package.json                   # 项目依赖、npm scripts、monorepo 配置
└── README.md                      # 项目说明文档

2. 入口、出口以及别名

js 复制代码
const path = require("path");
const fs = require("fs");

// 根目录 apps 路径
const appsPath = path.resolve(__dirname, "../apps");

// 获取所有子包目录
const appDirs = fs
  .readdirSync(appsPath, { withFileTypes: true })
  .filter(dirent => dirent.isDirectory())
  .map(dirent => dirent.name);

// 构建别名
const alias = {
  // 每个 app,比如 @mobile @test
  ...Object.fromEntries(appDirs.map(name => [`@${name}`, path.resolve(appsPath, name)]))
};

module.exports = { apps: appDirs, alias };

// webpack 入口、出口以及别名
const APP_NAME = process.env.APP_NAME || "mobile";
if (!apps.includes(APP_NAME)) {
  throw new Error(`未找到${APP_NAME}应用,请检查apps目录`);
} else {
  console.log(`🚀 开始编译${APP_NAME}应用`);
}
entry: {
  base: ["core-js/stable", "regenerator-runtime/runtime"],
  main: [resolve(`apps/${APP_NAME}/main.js`)]
},
output: {
  filename: "js/[name].[chunkhash:8].js",
  path: resolve(`${process.env.OUTPUT_DIR || "dist"}/${APP_NAME}`)
},
alias: {
  "@": resolve("packages/shared"),
  "~": resolve(""),
  ...alias
}

3. 子包单独编译

1. 根目录统一管控

js 复制代码
"build:test": "pnpm run -r build:test",
"build:prod": "pnpm run -r build:prod",
"dev:mobile": "pnpm --filter mobile dev",
"dev:wxmp": "pnpm --filter wxmp dev",

2. 子包打包配置

js 复制代码
{
  "name": "mobile",
  "version": "1.0.0",
  "private": true,
  "scripts": {
    "build:prod": "cross-env NODE_ENV=production BUILD_ENV=prod APP_NAME=mobile webpack --progress --config ../../build/webpack.prod.js",
    "build:test": "cross-env NODE_ENV=production BUILD_ENV=test APP_NAME=mobile webpack --progress --config ../../build/webpack.prod.js",
    "dev": "cross-env NODE_ENV=development BUILD_ENV=dev APP_NAME=mobile webpack serve --progress --config ../../build/webpack.dev.js"
  }
}
相关推荐
saber_andlibert3 小时前
TCMalloc底层实现
java·前端·网络
逍遥德3 小时前
如何学编程之01.理论篇.如何通过阅读代码来提高自己的编程能力?
前端·后端·程序人生·重构·软件构建·代码规范
冻感糕人~3 小时前
【珍藏必备】ReAct框架实战指南:从零开始构建AI智能体,让大模型学会思考与行动
java·前端·人工智能·react.js·大模型·就业·大模型学习
程序员agions3 小时前
2026年,“配置工程师“终于死绝了
前端·程序人生
alice--小文子3 小时前
cursor-mcp工具使用
java·服务器·前端
晚霞的不甘3 小时前
揭秘 CANN 内存管理:如何让大模型在小设备上“轻装上阵”?
前端·数据库·经验分享·flutter·3d
小迷糊的学习记录4 小时前
0.1 + 0.2 不等于 0.3
前端·javascript·面试
梦帮科技4 小时前
Node.js配置生成器CLI工具开发实战
前端·人工智能·windows·前端框架·node.js·json
VT.馒头5 小时前
【力扣】2695. 包装数组
前端·javascript·算法·leetcode·职场和发展·typescript
css趣多多5 小时前
一个UI内置组件el-scrollbar
前端·javascript·vue.js