/bruce-cli

React应用自动化构建脚手架

Primary LanguageJavaScriptMIT LicenseMIT

Bruce Cli

bruce-cli是一个React/Vue应用自动化构建脚手架,其零配置开箱即用的优点非常适合入门级、初中级、快速开发项目的前端同学使用,还可通过创建brucerc.js文件来覆盖其默认配置,只需专注业务代码的编写无需关注构建代码的编写,让项目结构更简洁。详情请戳这里,使用时记得查看文档哟,喜欢的可以给个Star

🎥背景

在前端技术的日益壮大下,从以前简单的几个文件到现在的一堆文件,各种扩展和工具植入到项目里,使得项目越来越庞大越来越难管理,前端项目也因此而趋于工程化和一体化。日新月异的前端技术已经让前端代码的业务逻辑和交互效果越来越复杂,项目会一直维护和迭代,令开发者更加不易于管理。模块化开发和各种框架把项目分成若干个小模块,增加了最后发布的困难,没有一个统一的标准,让前端项目结构千奇百怪。通常的项目都是团队开发,每个人的代码编写习惯和逻辑编写风格也很难一致。工作围绕着开发效率运行性能的工程化问题是我们作为前端开发者必须得处理的问题。因此前端项目自动化构建在整个项目开发中越来越重要。

🔗依赖

本项目是基于Webpack4.x.x进行开发的极速零配置开箱即用的Web应用打包工具(每次更新都会保持最新的依赖),集成各种常用工具(HandlebarsPostcssSassLessBabelTypeScriptStylelintEslintTslint等)扩展构建功能,用于构建和管理React/Vue技术栈的项目应用

📦安装

  • npm i -g bruce-cli
  • yarn global add bruce-cli

安装失败

  • 将npm源镜像设置为淘宝镜像:npm config set registry https://registry.npm.taobao.org/
  • 重新执行命令安装:npm i -g bruce-cliyarn add bruce-cli

💻使用

~ 命令 缩写
构建项目 bruce build bruce b
初始项目 bruce init bruce i
切换语言 bruce locale bruce l
创建组件 bruce new bruce n
删除依赖 bruce remove bruce r

☎️语言

  • zh-chs 简体中文 默认
  • zh-cht 繁體中文
  • en English

💡功能

命令功能

  • 构建项目:根据终端交互式问答选择所需的配置进行项目调试和项目打包
  • 初始项目:根据终端交互式问答选择所需的配置生成项目的基本文件和入口文件
  • 切换语言:切换在终端面板中显示的文本语言,可选简体中文、繁体中文、英文
  • 创建组件:输入组件路径后在项目根目录对应的路径下创建模板文件
  • 删除依赖:快速删除项目依赖文件和依赖锁定文件

内置功能

  • 模式选择:提供开发环境测试环境生产环境三中模式,每种模式对应不同的构建配置和优化方案
  • 端口监听:使用开发模式时,启动本地服务器,并监听指定端口,可自动打开浏览器来访问页面
  • 局部刷新:启用Webpack内置的HMR,配合react-hot-loader,增量更新css文件js文件,修哪更哪,无需刷新页面即可实时看见修改结果,并保存当前状态管理
  • 入口判断:快速扫描项目中指定的入口文件路径,判断其是否存在,项目构建中入口文件作为根节点,必须得保证其存在和路径正确
  • 垫片插入:根据项目兼容性自动插入垫片,兼容低版本浏览器
    • 插入动态polyfill,根据浏览器请求时的UserAgent返回垫片文件,babel编译JS代码时就不带上垫片进行编译,起到减包作用
    • 插入静态polyfill,根据browserslist和编写代码中的ES6语法自动插入所需的垫片代码
  • 动态导入:可使用动态导入的语法(import().then()),处理代码时会单独分离此模块,执行页面对应操作时才加载此模块,使用才加载不使用则不加载(代码懒加载),减少首屏加载代码大小和渲染时间
  • 代码编译:内置CSS编译器(PostcssSassLess)和JS编译器(BabelTypeScript)编译样式和脚本,开发时可使用最新或草案规范中的语法,使得代码更简洁,提高代码的可读性
    • 使用raw-loader处理和TXT,把文件内容以字符串的形式导入
    • 使用handlebars-loader处理内联HTML和HBS,把HTML元素以内联的形式插入页面当中
    • 使用style-loadercss-loader处理CSS,把CSS从JS中单独抽离出来
    • 使用postcss-loader处理CSS新特性和草案规范,根据browserslist增加CSS前缀
    • 使用sass-loader处理Sass,通过node-sass编译sass文件css文件
    • 使用less-loader处理Less,通过less编译less文件css文件
    • 使用babel-loader根据预设环境和browserslist处理编写的ES6代码TS代码,并生成浏览器可识别的ES5代码
  • 代码校验:确保编写的语法没有错误,统一规范团队协作中每位同事的代码编写风格,减少代码冗余,在保证代码语法正确的前提下提高代码的可读性
    • CSS校验:内置stylelint,配置标准的CSS语法规则,对开发过程中出现的语法错误进行检查和纠正
    • JS校验:内置eslinttslint,配置标准的JS和TS语法规则,对开发过程中出现的语法错误进行检查和纠正
  • 友好提示:当遇到警告和错误时输出语法高亮的代码片段和解决提示,帮助开发者定位问题
  • 代码分割:对业务代码进行打包,分割成WebpackRuntime代码块第三方依赖代码块公共业务代码块单个业务代码块
  • 代码合并:通过对相同模块、相同功能或复用多次的代码进行整体合并,起到减包作用
  • 压缩合并
    • CSS压缩:内置cssnano,对抽离出来的CSS进行压缩去重
    • JS压缩:内置uglifyjsterser,对抽离出来的JS进行压缩去重,uglifyjs用于压缩ES5terser用于压缩ES6
    • 图片压缩:内置imagemin,对pngjpg等图像进行无损压缩
  • 接口代理:使用proxy反向代理服务端接口,解决接口跨域问题
  • 资源处理:使用file-loaderurl-loader来处理字体、图像、音频、视频资源
  • 摇树优化:启用Webpack内置的Tree Shaking,禁止babel把代码转换成Commonjs规范,使用ESM规范的静态声明特点来去除不被引用或不被执行的代码块,起到减包作用
  • 缓存优化:在开启文件哈希化后,根据文件哈希值是否发生变化来执行打包操作,哈希没有变化的文件直接从缓存中获取,减少打包生成文件的时间
  • 文件哈希:可对生成文件设置哈希值,只有文件内容修改了才会更改哈希值,用于长缓存优化
  • 文件缓存:第一次构建速度可能慢一些,构建完成后会生成本地缓存文件,可提高后续再次构建的速度
  • 目录时化:可对输出的项目根目录进行时间序列化命名,增加时间戳来区分版本
  • 打包分析:可在打包完成后展示打包依赖的关系,根据关系图对模块的编写进行合理的分配和修改
  • 文件上传:暴露出构建成功的钩子,可在钩子函数上编写上传到服务器的代码用于打包后将文件上传到服务器,还可进行其他操作
  • 配置定制:当部分配置不符合项目需求时,可通过项目根目录下的配置文件brucerc.js来修改默认配置,构建启动时就会使用此配置文件来覆盖默认构建配置

⚙️配置

  • alias:模块导入快捷方式,配置详情请参考webpack resolve alias
  • browserList:目标浏览器配置列表,配置详情请参考browserslist
  • errorCb(err):构建失败回调函数,可自定其他操作
    • err:错误信息
  • eslintIgnores:Eslint忽略路径列表,配置详情请参考eslint ignores
  • eslintRules:Eslint校验规则列表,配置详情请参考eslint rulesrules
  • frame:开发框架(default表示普通应用,react表示React应用,vue表示Vue应用)
  • openPath:开发环境下浏览器打开后显示URL路径
  • proxy:接口代理,配置详情请参考webpack-dev-server proxy
  • publicPathProd:生产环境公共路径
  • publicPathTest:测试环境公共路径
  • style:样式格式(scss、less)
  • stylelintIgnores:Stylelint忽略路径列表,配置详情请参考stylelint ignores
  • stylelintRules:Stylelint校验规则列表,配置详情请参考stylelint rulesrules
  • successCb(mode, dir):构建成功回调函数,可自定义上传文件操作或其他操作
    • mode:运行环境(test表示测试环境,prod表示生产环境)
    • dir:输出路径
  • uploadOpts(mode):上传配置函数(必须返回一个Object,并包含以下字段),配置详情请参考scp2
    • 回调参数
      • mode:运行环境(test表示测试环境,prod表示生产环境)
    • 返回对象
      • host:服务器IP
      • password:密码(不可与privateKey共存)
      • path:目标文件路径
      • privateKey:秘钥(不可与password共存)
      • username:账号
  • useTs:集成TypeScript

默认配置

module.exports = {
	alias: {},
	browserList: [
		"last 20 Chrome versions",
		"last 20 Firefox versions",
		"last 20 Opera versions",
		"Explorer >= 11",
		"Safari >= 8",
		"Android >= 4.4",
		"iOS >= 8"
	],
	errorCb: null,
	eslintIgnores: [],
	eslintRules: {
		"eol-last": ["error", "never"],
		"handle-callback-err": ["error", "error"],
		"indent": ["error", "tab"],
		"jsx-quotes": ["error", "prefer-double"],
		"no-console": ["off"],
		"no-new": ["off"],
		"no-tabs": ["off"],
		"no-unreachable": ["off"],
		"object-curly-spacing": ["error", "always"],
		"operator-linebreak": ["error", "before"],
		"quotes": ["error", "double"],
		"semi": ["error", "always"],
		"sort-keys": ["error", "asc", { "caseSensitive": false, "natural": true }],
		"space-before-function-paren": ["error", "never"]
	},
	frame: "default",
	openPath: "",
	proxy: {},
	publicPathProd: "",
	publicPathTest: "",
	style: "scss",
	stylelintIgnores: [],
	stylelintRules: {
		"at-rule-empty-line-before": "never",
		"at-rule-no-unknown": [true, { "ignoreAtRules": ["content", "extend", "include", "mixin"] }],
		"color-hex-case": "lower",
		"comment-empty-line-before": "never",
		"declaration-empty-line-before": "never",
		"indentation": "tab",
		"no-descending-specificity": null,
		"no-missing-end-of-source-newline": null,
		"number-leading-zero": "never",
		"rule-empty-line-before": "never"
	},
	successCb: null,
	uploadOpts: null,
	useTs: false
};

覆盖配置

  • 目前只提供.js的形式编写文件,可参考以下配置编写
  • 因为本项目使用CommonJS规范来开发,所以必须使用module.exports = { ... };导出以下配置属性
  • 如需自定义上传操作,必须把uploadOpts设置为null或删除此字段,并使用successCb来定义上传操作
module.exports = {
	alias: {
		swiper: "swiper/dist/js/swiper.js"
	},
	browserList: [
		"> 0.5%",
		"last 2 versions"
	],
	errorCb(err) {
		console.log("错误信息", err);
	}
	eslintIgnores: [
		"src/components/*"
	],
	eslintRules: {
		"space-before-function-paren": ["error", "always"]
	},
	frame: "react",
	openPath: "?abc=123",
	proxy: [{
		changeOrigin: true,
		context: [
			"/cain/article/detail",
			"/cain/vote/result",
			"/cain/vote/post"
		],
		secure: false,
		target: "https://cain-dev.gameyw.netease.com"
	}],
	publicPathProd: "https://www.baidu.com",
	publicPathTest: "https://www.baidu.com",
	style: "less",
	stylelintIgnores: [
		"src/assets/css/*"
	],
	stylelintRules: {
		"color-hex-case": "upper"
	},
	successCb(mode, dir) {
		console.log("运行环境", mode);
		console.log("输出路径", dir);
	},
	uploadOpts(mode) {
		return {
			host: "your server ip",
			password: "your server password",
			path: "/root/www",
			username: "root"
		};
	},
	useTs: true
};

📋细节

CLI相关

  • 默认显示语言为简体中文,如需切换繁体中文英文请执行bruce l进行切换
  • 目前只装备了ReactVue的构建配置,请勿用来构建Angular或其他MVVM项目
  • 当前应用只能是React应用或Vue应用才能使用bruce n命令
  • 配置文件brucerc.js的属性是null[]{}时,会使用内置配置默认值
  • 请务必遵循构建错误提示进行修正,不要随意改动构建源代码和生成配置,有可能导致项目构建进程无法运行

IDE相关

  • 推荐使用VSCode进行项目开发,以下配置也是基于VSCode进行
  • 如果启用StylelintEslintTslint,需要在IDE上安装Stylelint插件Eslint插件Tslint插件才能配合本项目代码校验高亮显示警告和错误
  • 如需Stylelint高亮显示语法错误和保存时自动修复语法错误,可在首选项>设置中设定(目前失效)
  • 如需Eslint高亮显示语法错误和保存时自动修复语法错误,可在首选项>设置中设定(参考插件文档)
  • 如需Tslint高亮显示语法错误和保存时自动修复语法错误,可在首选项>设置中设定(参考插件文档)

文件相关

  • 项目只能单独存在JS或TS,JS项目下脚本文件只能是.js.jsx,TS项目下脚本文件只能是tstsx
  • 应用类型为SPA时,入口文件必须为src/index.js|ts|jsx|tsx
  • 应用类型为MPA时,入口文件必须为src/pages/pageName/index.js|ts|jsx|tsx
  • src/pages目录存在且包含子文件夹,则自动识别为MPA项目
  • 使用CSS精灵图时,必须把图标统一放到src/assets/icon下,且文件格式为png
  • 暴露出全局变量RUN_ENV用于获取当前运行环境,在使用EslintTslint的情况下会报语法错误,在后面加上eslint-disable-linetslint:disable-line即可
    • dev:开发环境
    • test:测试环境
    • prod:生产环境
  • 文件导入快捷方式
    • #:根目录
    • @src目录

垫片相关

  • @babel/polyfill7.4.0后被弃用,因此本项目使用的垫片为core-jsregenerator-runtime
  • 如无特殊兼容,入口文件最顶处无需增加import "core-js";import "regenerator-runtime";
  • 如需兼容低版本浏览器,在入口文件最顶处加入import "core-js";import "regenerator-runtime";(core-js版本必须为2.x.y)

TS相关

  • 当使用TS时,会在项目根目录下自动生成配置文件tsconfig.json
  • 如需修改TS配置,只需修改tsconfig.json
  • 如需修改Tslint配置,在项目根目录下创建配置文件tslint.json,手动编写配置覆盖默认Tslint配置

Default项目相关(bruce i初始项目时选择default)

  • 可使用内置handlebars模板
  • 入口文件必须为src/index.jssrc/pages/pageName/index.js
  • 初始时的应用类型为SPA(不提供MPA形式的初始化),如需转换成MPA,必须按照基础项目规定的入口文件形式重新分配文件路径
  • 可用来开发原生JS项目、Jquery项目、Zepto项目等
  • 公共函数需放置src/templates/helpers目录下,在模板内使用{{> fileName}}进行引用
  • 公共模板需放置src/templates/partials目录下,在模板内使用{{fileName param}}进行引用
  • 公共函数和公共模板的用法和例子请参考handlebars-loader

⚖️对比

  • 传统构建方案
  • 本构建方案

传统构建方案

基于GulpWebpack构建的React项目,项目代码分为构建代码业务代码,项目目录和文件配置是比较传统和多人使用的项目搭建方案。整个项目中除去业务代码后,构建代码的文件比较多,配置比较分散,比较难集中管理,无法做到开箱即用,通用性比较低,前期搭建项目构建方案可能花费的时间比较多,项目构建时需依赖本项目存在的依赖模块才能驱动。对于增删改构建功能和新同事入门,可能需要花比较多的时间去查找代码和熟悉构建逻辑

本构建方案

基于Webpack构建的React项目,代码只有业务代码,构建代码集中在一起做成一个npm模块并安装到全局环境中,通过命令调用本方案来驱动此项目,实现构建代码和业务代码完全分离。开发时无需关注如何写好构建代码和使用何种工具扩展构建功能,只需专注于业务代码的编写,整个项目只存在业务代码,可通过配置文件修改默认构建配置,大大节省项目前期准备时间,保证项目的简洁性独立性高效性维护性。省去项目前期搭建准备,直接开箱即用,使开发者集中精力写好业务代码

方案对比

~ 传统构建方案 本构建方案
构建文件 build文件夹 config文件夹 tools文件夹
.babelrc .eslintignore .eslintrc
.stylelintignore stylelintrc
业务文件 src文件夹 src文件夹
配置文件 无(已集成在构建文件中) brucerc.js
基础文件 package.json readme.md package.json readme.md
依赖模块 Webpack技术栈(构建) React技术栈(业务) React技术栈(业务)
安装时间 较慢
安装构建和业务代码依赖的模块
较快
只安装业务代码依赖的模块
开发准备 编写webpack和多种工具搭配的构建代码 开箱即用
全局使用 不可以 可以
构建复用 新建文件夹,复制粘贴构建代码,修改配置文件等 执行命令行初始项目和构建项目
新手构建 需了解构建代码逻辑和配置文件 执行命令行
后期扩展 在原有构建代码中进行增删改构建功能 通过配置文件brucerc.js进行增删改构建功能
配置管理 分散到不同的构建配置文件中 集成构建的基础配置
可通过配置文件brucerc.js进行覆盖

🔖版权

MIT © Joway Young

本项目由本人独自开发,经过2年多的时间沉淀出来,过程中进行了大量的项目测试和应用,目前上线的项目多达20多个,足可以支撑本项目的可行性和稳定性

由于自己项目开发经验和技术积累有限,不能保证本项目不存在任何Bug,如果在后续使用本项目时发现Bug或产生疑问,可以随时Issues或通过WeChat联系我(推荐,下方有二维码),我会立马反馈和修复相关Bug

🔨示例

  • 查看帮助

    brucebruce -h

  • 查看版本

    bruce -v

  • 切换语言

    bruce l

  • 进入文件夹

    cd projectList

  • 初始项目

    bruce i

  • 进入项目根目录

    cd myProject

  • 构建项目

    bruce b

  • 创建组件(构建项目时需另起一个cmd窗口执行)

    bruce n

  • 删除依赖

    bruce r

📝待做

  • 增加Vue的构建配置
  • 增加bruce eject命令,用于开发者自定义bruce-cli的配置
文档中虽然有提及Vue,但是目前还没完全集成进去本项目中,故Vue相关的构建无法使用,待后续更新

⏳后记

本项目源于我2017年3月份一个Angular项目里的构建代码,从最开始的Webpack2一直迭代到今天的Webpack4,话说Webpack5过段时间就要出了。

最初我的构思是写一份构建代码模板存放在Github上,然后通过脚本把构建代码拉下来。可是这样构建代码和业务代码还是同时存放在一个文件夹里,不容易管理,文件又多又杂。下次新开项目时又要把构建代码复制过去,有时升级构建功能,为了保持构建功能的统一,需要同时修改几个项目里的构建代码,妈的真烦。既然这样,为什么不把那些通用的构建代码抽出来做成一个npm模块呢,这样一次安装全局运行,多爽呀!

2018年3月我就开始对这个项目进行升级改造,做成一个npm模块,只不过一直在自己的电脑里跑。因为还没怎样应用到项目里,所以也不敢开源。经过差不多1年大大小小20多个项目的应用,终于稳定了这个项目的功能,所以我也决定对bruce-cli进行开源。对于所有通过bruce-cli创建的项目都可以开箱即用所有构建功能,如无特殊需求甚至是零配置即可运行项目。

开发这个项目经历了很多,挖的坑很多,填的坑也很多,很苦很累,有段时间还经常熬夜就是为了把它做得更好。不过收获也很大,学习了很多新知识新技能,把常用的Node知识都用上了,也为自己后期做Node服务端开发打下了巩固的基础。有付出就有收获,我还是一直深信这句话,因为自己确实进步了很多。

本项目是基于Node10Node11进行开发的,为了兼容Node8+,所以使用了babel对源码进行了编译,生成现在线上版本的代码,待更多的项目测试完成和应用起来后会开放源码供大家一起学习和完善。如果在后续使用本项目时发现Bug或产生疑问,可以随时Issues或通过WeChat联系我。使用时记得查看文档哟,详情请戳这里,喜欢的可以给个Star。