从0开始搭建一个Monorepo模版,基于Turborepo+pnpm+changesets+dumi

Monorepo

  • 前言
  • 开始
    • 一、使用turborepo初始化项目
    • 二、调整目录结构及文件
      • 1. 调整`package.json`文件
      • 2. 调整app目录
      • 3. 调整`eslint`包
      • 4. 调整`ui`包
      • 5. 调整eslint配置
      • 6. 调整.npmrc
      • 7. 使用commitizen规范代码提交
      • 8. 使用commitlint+husky进行 commit提交信息校验
      • 9. 使用husky进行commit前的规范校验
    • 三、dev run
    • 四、打包
      • ts与eslint
      • UI
    • 五、发版
      • 修改所有的包名
      • 使用Changesets进行版本管理并发版
  • 总结

前言

当你想要搭建一个Monorepo时,相信你已经调研过很多技术选型,之前我也使用过yarn+workspace+lerna+dumi开发过一套组件库,后续有时间把这套工具也抽一个模版。

因为pnpm本身支持workspace并且lerna已经停止维护,因此决定使用新的工具从新搭建一套Monorepo模版,那么这套工具就是:

  • pnpm—包管理器
  • turborepo—外壳
  • changesets—版本号管理、changelog管理
  • dumi—文档预览
  • 其他的规范代码的工具如:—commitizeneslintlint-stagedcommitlinthusky

你可能对Turborepo不太了解,他的官方解释是高性能的JavaScript和TypeScript代码库构建系统,他在这只是作为一个壳子用来提升开发及打包效率,不用了解太多。

开始

一、使用turborepo初始化项目

  1. pnpm dlx create-turbo@latest
  2. 选择目录(项目名)、选择包管理器
    在这里插入图片描述
  3. 项目初始化完成,具体的目录结构什么意思可以去Turborepo看一下,不看也没关系,因为后续我们要调整它。
    在这里插入图片描述

我现在这里的turbo的version是1.10.12,如果你很晚看见这篇文章,可能版本有了很大不同。

  1. 链接gitHub仓库,当然这一步不是必须的,链接它的目的只是为了测试规范代码的工具是否生效。
    git remote add origin https://github.com/Atw-Lee/monorepo-template.git

不要划走:Monorepo可以用来管理多应用工程(app目录下的各种工程),所有工程依赖相同的UI,而Turborepo初始化的项目也是这么干的。但是,我需要的是搭建一个我的UI组件库,这是与它不同的地方,app下我只需要一个文档预览工程(dumi),因此,我需要调整它的目录结构

二、调整目录结构及文件

1. 调整package.json文件

  • 安装eslint@latest(eslint提到全局,并更新到最新版本)
  • 安装@changesets/cli(版本号、changelog管理)
  • 安装commitizen cz-conventional-changelog(规范commit信息)
  • 安装@commitlint/cli @commitlint/config-conventiona husky(检验commit信息)
  • 安装lint-staged stylelint(校验规范格式)
  • 安装father(打包)
  • 增加"preinstall": "npx only-allow pnpm"只允许使用pnpm包管理器

pnpm i -Dw eslint@latest @changesets/cli commitizen cz-conventional-changelog @commitlint/cli @commitlint/config-conventional husky lint-staged stylelint father


2. 调整app目录

  • 删除web及docs目录的所有内容
  • 新建docs目录并初始化dumi项目
    在这里插入图片描述
  • 进入docs目录执行git init
  • 删除.fatherrc.ts文件(不需要在这打包组件)
  • .gitignore文件增加docs-dist(该目录为dumi的build产物)
  • 修改package.json文件:
    • 删除docs:build
    • 修改build:dumi build(打包直接是该文档工程打包,不打包组件)
    • 删除moduletypesfiles配置(用不着)
    • 依赖删除与外边重复的(commitlint、eslint、father、husky、lint-staged、stylelint等)

3. 调整eslint

我不准备使用它默认的eslint,因为我用不到nextjs相关的内容,我选择使用Tencent的eslint-config-alloy

  • 进入packages/eslint-config-custom目录,手动删除不需要的依赖,pnpm i eslint-config-alloy安装alloy
  • 更新eslint-plugin-react(后续补充,不是最新的出错了)

  1. 调整tsconfig
  • 删除package.json的private属性(为了后续可以正常发版)
  • 删除nextjs.js文件(不需要next的ts配置)

在这里插入图片描述


4. 调整ui

  • 删除turbo目录(用不着)
  • package.json文件删除generate脚本(用不着)
  • package.json文件删除eslint依赖(外边已经有了)
  • package.json文件新增脚本"build": "father build"(打包组件,app里的docs不需要了,因为UI组件包的开发在这里)
  • package.json文件修改main属性"main": "./src/index",
  • package.json文件修改types属性"main": "./src/index",
  • 新建src目录,增加button组件
    在这里插入图片描述
  • 修改tsconfig.json文件
{"extends": "tsconfig/react-library.json","include": ["src"],"exclude": ["lib", "dist", "es", "node_modules"]
}
  • 新增.fatherrc.ts文件
import { defineConfig } from "father";
export default defineConfig({// more father config: https://github.com/umijs/father/blob/master/docs/config.mdesm: { output: "es" },cjs: { output: "lib" },
});
  • .gitignore文件增加lib及es(该目录为ui通过father打包产物,具体名称还是根据上一步的配置来)
  • .gitignore文件删除next相关的(用不到)

5. 调整eslint配置

module.exports = {root: true,extends: ["custom"],
};

6. 调整.npmrc

  • 增加registry=你公司的私有源,开发的UI组件库发布到这(在这直接定义注册地址就比较方便了)

7. 使用commitizen规范代码提交

  • package.json文件增加"commit": "cz"脚本
  • package.json文件增加config属性
  "config": {"commitizen": {"path": "node_modules/cz-conventional-changelog"}}
  • 执行 pnpm commit测试
    在这里插入图片描述

8. 使用commitlint+husky进行 commit提交信息校验

  • 根目录新增commitlint.config.js文件
module.exports = { extends: ["@commitlint/config-conventional"] };
  • package.json文件增加"postinstall": "husky install"
  • 执行pnpm i生成.husky文件,初始化husky。完成后可删除postinstall属性
  • 执行npx husky add .husky/commit-msg 'npx --no-install commitlint --edit "$1"'生成commit-msg的hook
  • package.json文件增加commitlintlint-staged属性
	"commitlint": {"extends": ["@commitlint/config-conventional"]},"lint-staged": {"*.{md,json}": ["prettier --write --no-error-on-unmatched-pattern"],"*.{css,less}": ["stylelint --fix","prettier --write"],"*.{js,jsx}": ["eslint --fix","prettier --write"],"*.{ts,tsx}": ["eslint --fix","prettier --parser=typescript --write"]},
  • 执行错误的commit信息测试
    在这里插入图片描述

9. 使用husky进行commit前的规范校验

  • 执行npx husky add .husky/pre-commit "npx --no-install lint-staged"
  • 上一步已经添加完成了lint-staged属性
  • 执行commit进行校验
    在这里插入图片描述

三、dev run

  • 执行pnpm dev:页面正常打开dumi运行的应用
  • docs的package.json增加ui的依赖"ui": "workspace:*"
  • 页面中使用ui包中的button组件
    在这里插入图片描述
    到这我们的运行时的组件与文档预览已经ok了,接下来就是ui组件的打包

四、打包

ts与eslint

他俩不需要打包,都是些配置项

UI

上文我们已经设置好了father打包的配置文件,直接执行pnpm build
在这里插入图片描述
生成了对应的lib和es文件夹,docs也跟着打包了,这里是turborepo的操作逻辑,我们不用关心它

  • ui的package.json中新增files属性
  • ui的package.jsonmain、types属性也要做一个修改,之前是./src
  "main": "./lib/index","types": "./lib/index","files": ["es","lib","dist"],

五、发版

接下来就需要把我们的包发布到我们的私有源上了,当然开源也没有问题

我们现在有三个包,分别是eslint、typescript、ui
我们的包肯定不能叫这个名字,那我们给他们改下名字,加上组织的前缀,我这里以@you-org为例

修改所有的包名

在这里插入图片描述
该变更中的文件均有需要修改

使用Changesets进行版本管理并发版

  • 执行pnpm changeset init生成.changeset文件
  • 执行pnpm changeset选择要发布的包
    在这里插入图片描述
  • 选择版本(一版fix是patch,feat是minor)
    在这里插入图片描述
  • 执行changeset version增加版本号
    在这里插入图片描述
  • commit提交
  • 执行pnpm -r publish --tag alpha0.1.0发布版本
    在这里插入图片描述
    OK到这里就结束了~

总结

收获:从0开始搭建了一个多包(Monorepo)的UI组件模版,了解了一个简单开源UI工程应该具有的工具及搭建步骤。

不足之处在于:

  • 组件库的打包使用rollup可能会更好,但是因为额外的学习成本,还是使用了dumi上自带的father
  • 除了ui包,应该在添加一个utils包,这样才更加具有Monorepo的特性
  • tsconfigeslint包不确定是否有必要单独提成一个包,如果项目很复杂的话,eslint和tsconfig需要灵活配置的话,可能会更有用吧
  • 缺少了一些脚本,通过脚本来减少复杂的原生命令
  • 需要额外的脚手架来协调模版工作,以便不需要更改额外的package.json文件,后续也会做这方面的工作

参考文章

  • 打造高效Monorepo:Turborepo、pnpm、Changesets实践
  • pnpm + workspace + changesets 构建你的 monorepo 工程
  • 在 pnpm 中使用 Changesets
  • Creating a new monorepo

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/91094.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Rust 重载运算符|复数结构的“加减乘除”四则运算

复数 基本概念 复数定义 由实数部分和虚数部分所组成的数,形如a+bi 。 其中a、b为实数,i 为“虚数单位”,i -1,即虚数单位的平方等于-1。 a、b分别叫做复数a+bi的实部和虚部。 当b0时,a&…

利用python实现批量登录网络设备进行日常巡检

利用python实现批量登录网络设备 实现ensp与物理机互通ensp 配置配置网络设备远程登录 用python实现批量登录常见问题 通过阅读本文可以学习自动化运维相关知识,本文章代码可以直接使用,通过批量登录功能后,可以按照自己意愿进行功能更改与完…

OpenCV图像处理——边缘检测

目录 原理Sobel检测算子方法应用 Laplacian算子Canny边缘检测原理 原理 Sobel检测算子 方法 应用 sobel_x_or_ycv.Sobel(src,ddepth,dx,dy,dst,ksize,scale,delta,borderType)import numpy as np import cv2 as cv import matplotlib.pyplot as pltimgcv.imread(./汪学长的随堂…

无需停服!PostgreSQL数据迁移工具-NineData

PostgreSQL 是一种备受开发者和企业青睐的关系型数据库,其丰富的数据类型、地理空间负载和强大的扩展能力等特性使其备受欢迎。然而,在企业使用 PostgreSQL 承载应用的过程中,由于业务需要上云、跨云、下云、跨机房迁移、跨地域迁移、数据库版…

云上社群学习系统部分接口设计详解

目录 一、项目简介 二、技术选型 三、数据库设计 四、接口设计及思考 回复帖子部分 4.1 回复帖子 4.1.1.1 实现逻辑 4.1.1.2创建Service接⼝ 4.1.1.3 实现Service接⼝ 4.1.1.4 实现Controller 4.1.1.5 测试接口 4.1.1.6 实现前端页面 4.2 点赞帖子 4.2.1.1 参数要求…

【数据结构】堆的实现,堆排序以及TOP-K问题

目录 1.堆的概念及结构 2.堆的实现 2.1初始化堆 2.2销毁堆 2.3取堆顶元素 2.4返回堆的大小 2.5判断是否为空 2.6打印堆 2.7插入元素 2.8堆的向上调整 2.9弹出元素 2.10堆的向下调整 3. 建堆时间复杂度 4. 堆的应用 4.1 堆排序 4.2 TOP-K问题 1.堆的概念及结构 …

【Spring】统一事件的处理(拦截器、统一异常处理、统一数据格式返回)

文章目录 前言一、Spring 拦截器1.1 用户登录权限校验案例1.1.1 最初的用户登录验证1.1.2 使用 Spring AOP 实现登录验证的问题 1.2 Spring 拦截器的使用1.2.1 Spring 拦截器概念与使用步骤1.2.2 使用拦截器实现对用户登录权限的校验 1.3 拦截器实现原理1.4 Spring 拦截器和 Sp…

响应式设计是什么?怎么学习? - 易智编译EaseEditing

响应式设计是一种用于创建能够适应不同设备和屏幕尺寸的网站和应用程序的设计方法。它的目标是确保网站在各种设备上都能提供良好的用户体验,无论是在大屏幕的桌面电脑上还是在小屏幕的移动设备上。 在响应式设计中,页面的布局、字体、图像和其他元素会…

读《芯片浪潮》,学习台积电张忠谋的管理之道

大家知道,台积电一个公司就占据了全球晶圆代工市场一半的份额。 5纳米及以下最先进工艺的芯片,台积电可占到惊人的90%以上的市场。全球最新最强的智能手机、笔记本电脑的核心计算芯片都必须仰仗台积电一个企业的供应。 换一个说法,如果没有…

每天一道leetcode:剑指 Offer 12. 矩阵中的路径(中等DFS深度优先遍历)

今日份题目: 给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。 单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元…

62、华为昇腾开发板Atlas 200I DK A2配置mmpose的hrnet模型推理python/c++

基本思想:适配mmpose模型,记录一下流水帐,环境配置和模型来自,请查看参考链接。 链接: https://pan.baidu.com/s/1IkiwuZf1anyKX1sZkYmD1g?pwdi51s 提取码: i51s 一、转模型 (base) rootdavinci-mini:~/sxj731533730# atc --mo…

docker pull 设置代理 centos

On CentOS the configuration file for Docker is at: /etc/sysconfig/docker 用 root 权限打开 text editor sudo gedit 注意 加引号 Adding the below line helped me to get the Docker daemon working behind a proxy server: HTTP_PROXY“http://<proxy_host>:&…

C++ 动态规划经典案例解析之最长公共子序列(LCS)_窥探递归和动态规划的一致性

1. 前言 动态规划处理字符相关案例中&#xff0c;求最长公共子序列以及求最短编辑距离&#xff0c;算是经典中的经典案例。 讲解此类问题的算法在网上一抓应用一大把&#xff0c;即便如此&#xff0c;还是忍不住有写此文的想法。毕竟理解、看懂都不算是真正掌握&#xff0c;唯…

浅谈统一权限管理服务的设计与开发

作者 | 天地练心 导读 本文详细探讨了统一权限管理服务&#xff08;MPS&#xff09;的设计与开发&#xff0c;针对企业内部多平台权限管理混乱的问题&#xff0c;提出了一套综合RBAC、ACL、DAC权限模型的解决方案。文章从需求分析、技术选型、功能设计等方面全面介绍了MPS的构建…

阿里云ACP知识点

前言&#xff1a;记录ACP错题 1、在创建阿里云ECS时&#xff0c;每台服务器必须要包含_______用来存储操作系统和核心配置。 系统盘&#xff08;不是实例&#xff0c;实例是一个虚拟的计算环境&#xff0c;由CPU、内存、系统盘和运行的操作系统组成&#xff1b;ESC实例作为云…

2023国赛数学建模E题思路分析

文章目录 0 赛题思路1 竞赛信息2 竞赛时间3 建模常见问题类型3.1 分类问题3.2 优化问题3.3 预测问题3.4 评价问题 4 建模资料 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 1 竞赛信息 全国大学生数学建模…

纯js点击按钮切换首页部分页面

像我这种大数据的&#xff0c;不会前端的&#xff0c;懒得学框架&#xff0c;现在有gpt了&#xff0c;前端对于我来说&#xff0c;用原生的更加友好&#xff0c;毕竟算法gpt都能优化。 首页我有个页面&#xff0c;然后我现在想点击gm替换上面的统计&#xff0c;点击用户替换回…

Flask Web开发实战(狼书)| 笔记第1、2章

前言 2023-8-11 以前对网站开发萌生了想法&#xff0c;又有些急于求成&#xff0c;在B站照着视频敲了一个基于flask的博客系统。但对于程序的代码难免有些囫囵吞枣&#xff0c;存在许多模糊或不太理解的地方&#xff0c;只会照葫芦画瓢。 而当自己想开发一个什么网站的时&…

SpringCloud微服务之间如何进行用户信息传递(涉及:Gateway、OpenFeign组件)

目录 1、想达到的效果2、用户信息在微服务之间传递的两种途径3、用RuoYi-Cloud为例进行演示说明&#xff08;1&#xff09;网关将用户信息写在请求头中&#xff08;2&#xff09;业务微服务之间通过OpenFeign进行调用&#xff0c;并且将用户信息写在OpenFeign准备的请求头中&am…

Qt+C++自定义控件仪表盘动画仿真

程序示例精选 QtC自定义控件仪表盘动画仿真 如需安装运行环境或远程调试&#xff0c;见文章底部个人QQ名片&#xff0c;由专业技术人员远程协助&#xff01; 前言 这篇博客针对<<QtC自定义控件仪表盘动画仿真>>编写代码&#xff0c;代码整洁&#xff0c;规则&…