资讯专栏INFORMATION COLUMN

【中高级前端必备】手摸手教你撸一个脚手架

caoym / 3357人阅读

摘要:管理文件当前用户目录下文件的增删改查是配置文件是默认的配置发布将本脚手架发布至上。

脚手架

vue-cli, create-react-appreact-native-cli 等都是非常优秀的脚手架,通过脚手架,我们可以快速初始化一个项目,无需自己从零开始一步步配置,有效提升开发体验。尽管这些脚手架非常优秀,但是未必是符合我们的实际应用的,我们可以定制一个属于自己的脚手架(或公司通用脚手架),来提升自己的开发效率。

更多优质文章可戳: https://github.com/YvetteLau/...

</>复制代码

  1. 脚手架的作用

减少重复性的工作,不需要复制其他项目再删除无关代码,或者从零创建一个项目和文件。

可以根据交互动态生成项目结构和配置文件。

多人协作更为方便,不需要把文件传来传去。

实现的功能

在开始之前,我们需要明确自己的脚手架需要哪些功能。vue init template-name project-namecreate-react-app project-name。我们这次编写的脚手架(eos-cli)具备以下能力(脚手架的名字爱叫啥叫啥,我选用了Eos黎明女神):

eos init template-name project-name 根据远程模板,初始化一个项目(远程模板可配置)

eos config set 修改配置信息

eos config get [] 查看配置信息

eos --version 查看当前版本号

eos -h

大家可以自行扩展其它的 commander,本篇文章旨在教大家如何实现一个脚手架。

</>复制代码

  1. 效果展示

初始化一个项目

修改.eosrc文件,从 vuejs-template 下载模板

需要使用的第三方库

babel-cli/babel-env: 语法转换

commander: 命令行工具

download-git-repo: 用来下载远程模板

ini: 格式转换

inquirer: 交互式命令行工具

ora: 显示loading动画

chalk: 修改控制台输出内容样式

log-symbols: 显示出 √ 或 × 等的图标

关于这些第三方库的说明,可以直接npm上查看相应的说明,此处不一一展开。

初始化项目

创建一个空项目(eos-cli),使用 npm init 进行初始化。

安装依赖

</>复制代码

  1. npm install babel-cli babel-env chalk commander download-git-repo ini inquirer log-symbols ora
目录结构

</>复制代码

  1. ├── bin
  2. │ └── www //可执行文件
  3. ├── dist
  4. ├── ... //生成文件
  5. └── src
  6. ├── config.js //管理eos配置文件
  7. ├── index.js //主流程入口文件
  8. ├── init.js //init command
  9. ├── main.js //入口文件
  10. └── utils
  11. ├── constants.js //定义常量
  12. ├── get.js //获取模板
  13. └── rc.js //配置文件
  14. ├── .babelrc //babel配置文件
  15. ├── package.json
  16. ├── README.md
babel 配置

开发使用了ES6语法,使用 babel 进行转义,

</>复制代码

  1. .bablerc

</>复制代码

  1. {
  2. "presets": [
  3. [
  4. "env",
  5. {
  6. "targets": {
  7. "node": "current"
  8. }
  9. }
  10. ]
  11. ]
  12. }
eos 命令

node.js 内置了对命令行操作的支持,package.json 中的 bin 字段可以定义命令名和关联的执行文件。在 package.json 中添加 bin 字段

</>复制代码

  1. package.json

</>复制代码

  1. {
  2. "name": "eos-cli",
  3. "version": "1.0.0",
  4. "description": "脚手架",
  5. "main": "index.js",
  6. "bin": {
  7. "eos": "./bin/www"
  8. },
  9. "scripts": {
  10. "compile": "babel src -d dist",
  11. "watch": "npm run compile -- --watch"
  12. }
  13. }

</>复制代码

  1. www 文件

行首加入一行 #!/usr/bin/env node 指定当前脚本由node.js进行解析

</>复制代码

  1. #! /usr/bin/env node
  2. require("../dist/main.js");
链接到全局环境

开发过程中为了方便调试,在当前的 eos-cli 目录下执行 npm link,将 eos 命令链接到全局环境。

启动项目

</>复制代码

  1. npm run watch
处理命令行

利用 commander 来处理命令行。

</>复制代码

  1. main

</>复制代码

  1. import program from "commander";
  2. import { VERSION } from "./utils/constants";
  3. import apply from "./index";
  4. import chalk from "chalk";
  5. /**
  6. * eos commands
  7. * - config
  8. * - init
  9. */
  10. let actionMap = {
  11. init: {
  12. description: "generate a new project from a template",
  13. usages: [
  14. "eos init templateName projectName"
  15. ]
  16. },
  17. config: {
  18. alias: "cfg",
  19. description: "config .eosrc",
  20. usages: [
  21. "eos config set ",
  22. "eos config get ",
  23. "eos config remove "
  24. ]
  25. },
  26. //other commands
  27. }
  28. // 添加 init / config 命令
  29. Object.keys(actionMap).forEach((action) => {
  30. program.command(action)
  31. .description(actionMap[action].description)
  32. .alias(actionMap[action].alias) //别名
  33. .action(() => {
  34. switch (action) {
  35. case "config":
  36. //配置
  37. apply(action, ...process.argv.slice(3));
  38. break;
  39. case "init":
  40. apply(action, ...process.argv.slice(3));
  41. break;
  42. default:
  43. break;
  44. }
  45. });
  46. });
  47. function help() {
  48. console.log("
  49. Usage:");
  50. Object.keys(actionMap).forEach((action) => {
  51. actionMap[action].usages.forEach(usage => {
  52. console.log(" - " + usage);
  53. });
  54. });
  55. console.log("
  56. ");
  57. }
  58. program.usage(" [options]");
  59. // eos -h
  60. program.on("-h", help);
  61. program.on("--help", help);
  62. // eos -V VERSION 为 package.json 中的版本号
  63. program.version(VERSION, "-V --version").parse(process.argv);
  64. // eos 不带参数时
  65. if (!process.argv.slice(2).length) {
  66. program.outputHelp(make_green);
  67. }
  68. function make_green(txt) {
  69. return chalk.green(txt);
  70. }
下载模板

download-git-repo 支持从 Github、Gitlab 下载远程仓库到本地。

</>复制代码

  1. get.js

</>复制代码

  1. import { getAll } from "./rc";
  2. import downloadGit from "download-git-repo";
  3. export const downloadLocal = async (templateName, projectName) => {
  4. let config = await getAll();
  5. let api = `${config.registry}/${templateName}`;
  6. return new Promise((resolve, reject) => {
  7. //projectName 为下载到的本地目录
  8. downloadGit(api, projectName, (err) => {
  9. if (err) {
  10. reject(err);
  11. }
  12. resolve();
  13. });
  14. });
  15. }
init 命令 命令行交互

在用户执行 init 命令后,向用户提出问题,接收用户的输入并作出相应的处理。命令行交互利用 inquirer 来实现:

</>复制代码

  1. inquirer.prompt([
  2. {
  3. name: "description",
  4. message: "Please enter the project description: "
  5. },
  6. {
  7. name: "author",
  8. message: "Please enter the author name: "
  9. }
  10. ]).then((answer) => {
  11. //...
  12. });

视觉美化

在用户输入之后,开始下载模板,这时候使用 ora 来提示用户正在下载模板,下载结束之后,也给出提示。

</>复制代码

  1. import ora from "ora";
  2. let loading = ora("downloading template ...");
  3. loading.start();
  4. //download
  5. loading.succeed(); //或 loading.fail();

</>复制代码

  1. index.js

</>复制代码

  1. import { downloadLocal } from "./utils/get";
  2. import ora from "ora";
  3. import inquirer from "inquirer";
  4. import fs from "fs";
  5. import chalk from "chalk";
  6. import symbol from "log-symbols";
  7. let init = async (templateName, projectName) => {
  8. //项目不存在
  9. if (!fs.existsSync(projectName)) {
  10. //命令行交互
  11. inquirer.prompt([
  12. {
  13. name: "description",
  14. message: "Please enter the project description: "
  15. },
  16. {
  17. name: "author",
  18. message: "Please enter the author name: "
  19. }
  20. ]).then(async (answer) => {
  21. //下载模板 选择模板
  22. //通过配置文件,获取模板信息
  23. let loading = ora("downloading template ...");
  24. loading.start();
  25. downloadLocal(templateName, projectName).then(() => {
  26. loading.succeed();
  27. const fileName = `${projectName}/package.json`;
  28. if(fs.existsSync(fileName)){
  29. const data = fs.readFileSync(fileName).toString();
  30. let json = JSON.parse(data);
  31. json.name = projectName;
  32. json.author = answer.author;
  33. json.description = answer.description;
  34. //修改项目文件夹中 package.json 文件
  35. fs.writeFileSync(fileName, JSON.stringify(json, null, "
  36. "), "utf-8");
  37. console.log(symbol.success, chalk.green("Project initialization finished!"));
  38. }
  39. }, () => {
  40. loading.fail();
  41. });
  42. });
  43. }else {
  44. //项目已经存在
  45. console.log(symbol.error, chalk.red("The project already exists"));
  46. }
  47. }
  48. module.exports = init;
config 配置

</>复制代码

  1. eos config set registry vuejs-templates

config 配置,支持我们使用其它仓库的模板,例如,我们可以使用 vuejs-templates 中的仓库作为模板。这样有一个好处:更新模板无需重新发布脚手架,使用者无需重新安装,并且可以自由选择下载目标。

</>复制代码

  1. config.js

</>复制代码

  1. // 管理 .eosrc 文件 (当前用户目录下)
  2. import { get, set, getAll, remove } from "./utils/rc";
  3. let config = async (action, key, value) => {
  4. switch (action) {
  5. case "get":
  6. if (key) {
  7. let result = await get(key);
  8. console.log(result);
  9. } else {
  10. let obj = await getAll();
  11. Object.keys(obj).forEach(key => {
  12. console.log(`${key}=${obj[key]}`);
  13. })
  14. }
  15. break;
  16. case "set":
  17. set(key, value);
  18. break;
  19. case "remove":
  20. remove(key);
  21. break;
  22. default:
  23. break;
  24. }
  25. }
  26. module.exports = config;

</>复制代码

  1. rc.js

.eosrc 文件的增删改查

</>复制代码

  1. import { RC, DEFAULTS } from "./constants";
  2. import { decode, encode } from "ini";
  3. import { promisify } from "util";
  4. import chalk from "chalk";
  5. import fs from "fs";
  6. const exits = promisify(fs.exists);
  7. const readFile = promisify(fs.readFile);
  8. const writeFile = promisify(fs.writeFile);
  9. //RC 是配置文件
  10. //DEFAULTS 是默认的配置
  11. export const get = async (key) => {
  12. const exit = await exits(RC);
  13. let opts;
  14. if (exit) {
  15. opts = await readFile(RC, "utf8");
  16. opts = decode(opts);
  17. return opts[key];
  18. }
  19. return "";
  20. }
  21. export const getAll = async () => {
  22. const exit = await exits(RC);
  23. let opts;
  24. if (exit) {
  25. opts = await readFile(RC, "utf8");
  26. opts = decode(opts);
  27. return opts;
  28. }
  29. return {};
  30. }
  31. export const set = async (key, value) => {
  32. const exit = await exits(RC);
  33. let opts;
  34. if (exit) {
  35. opts = await readFile(RC, "utf8");
  36. opts = decode(opts);
  37. if(!key) {
  38. console.log(chalk.red(chalk.bold("Error:")), chalk.red("key is required"));
  39. return;
  40. }
  41. if(!value) {
  42. console.log(chalk.red(chalk.bold("Error:")), chalk.red("value is required"));
  43. return;
  44. }
  45. Object.assign(opts, { [key]: value });
  46. } else {
  47. opts = Object.assign(DEFAULTS, { [key]: value });
  48. }
  49. await writeFile(RC, encode(opts), "utf8");
  50. }
  51. export const remove = async (key) => {
  52. const exit = await exits(RC);
  53. let opts;
  54. if (exit) {
  55. opts = await readFile(RC, "utf8");
  56. opts = decode(opts);
  57. delete opts[key];
  58. await writeFile(RC, encode(opts), "utf8");
  59. }
  60. }
发布

npm publish 将本脚手架发布至npm上。其它用户可以通过 npm install eos-cli -g 全局安装。
即可使用 eos 命令。

项目地址

本项目完整代码请戳: https://github.com/YvetteLau/...

编写本文,虽然花费了一定时间,但是在这个过程中,我也学习到了很多知识,谢谢各位小伙伴愿意花费宝贵的时间阅读本文,如果本文给了您一点帮助或者是启发,请不要吝啬你的赞和Star,您的肯定是我前进的最大动力。
https://github.com/YvetteLau/...

参考文章:

[1] npm依赖文档(https://www.npmjs.com/package...

</>复制代码

  1. 关注公众号,加入技术交流群。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/106017.html

相关文章

  • 前端之从零开始系列

    摘要:只有动手,你才能真的理解作者的构思的巧妙只有动手,你才能真正掌握一门技术持续更新中项目地址求求求源码系列跟一起学如何写函数库中高级前端面试手写代码无敌秘籍如何用不到行代码写一款属于自己的类库原理讲解实现一个对象遵循规范实战手摸手,带你用撸 Do it yourself!!! 只有动手,你才能真的理解作者的构思的巧妙 只有动手,你才能真正掌握一门技术 持续更新中…… 项目地址 https...

    Youngdze 评论0 收藏0
  • PHP--摸手,教你撸一个会自动补全的命令行工具1

    摘要:前言一次在使用的时候,发现使用命令行的时候有些关键字会自动提示。介绍随着,等框架的流行,命令行工具越来越流行,但是很多时候命令太多,根本无法记住所有参数,或者参数太长输入太不方便。下文,我们一起来优化这个工具。备注不支持自动补全 前言 一次在使用symfony的时候,发现使用命令行的时候有些关键字会自动提示。 showImg(https://segmentfault.com/img/b...

    Chiclaim 评论0 收藏0
  • 手摸教你用canvas实现给图片添加平铺水印

    摘要:最近项目中遇到一个需求,需要把一张图片加上平铺的水印类似这样的效果首先想到的是用完成这种功能,因为我之前也没有接触过,所以做这个功能的时候,就是一步一步的摸索中学习,过程还是挺的,接下来跟我一步步来实现这个功能以及发现一些的坑吧。 最近项目中遇到一个需求,需要把一张图片加上平铺的水印 类似这样的效果showImg(https://segmentfault.com/img/remote/...

    崔晓明 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<