likes
comments
collection
share

基于 node-ssh 实现前后台项目自动部署本文主要介绍了基于 node-ssh 实现前后台项目自动发布的工具实现,其

作者站长头像
站长
· 阅读数 46

前言

为了方便自己的前后台项目的部署,而无需每次都通过 xshell 或者其他工具来手动上传文件,过程相对复杂,且容易出错。因此决定自己开发一个自动化部署工具。同时为了方便之后每个项目都能方便的使用,而不是将这个项目上实现的部署代码拷贝到另一个项目上使用,所以需要将该项目作为脚手架的形式做成一个 npm 包,方便其他项目使用。

自动化部署方案

当今市面上自动部署的方案有 Jenkins、GitLab 等等。其中 Jenkins 适合需要高度定制和复杂集成的场景,而 GitLab 则适合希望在一个平台上完成整个开发、测试和部署流程的团队。

而我们项目,只需要将前端代码,即 dist 文件,丢到 nginx 上,同时将服务代码丢到服务器上就行了,因此不需要搞一套复杂的发布流程,因此决定选择使用以 node-ssh 这个库为基础,实现自动化部署。

什么是 node-ssh

node-ssh 是一个基于 Node.js 的模块,它提供了通过 SSH 连接到远程服务器并执行命令的功能。这使得我们可以在 Node.js 环境下轻松地实现对远程服务器的操作,如文件传输、命令执行等。

基于 node-ssh 连接服务器及操作服务器资源

建立 SSH 连接

const { NodeSSH } = require("node-ssh");
const ssh = new NodeSSH();

const config = {
  host: "remote-server-ip",
  username: "your-username",
  password: "your-password",
};

ssh
  .connect(config)
  .then(() => {
    console.log("SSH 连接成功");
    // 连接成功后可以执行远程命令或文件操作
  })
  .catch((err) => {
    console.error("SSH 连接失败", err);
  });

执行远程命令

ssh
  .execCommand("pm2 delete 0")
  .then((result) => {
    console.log("命令输出:", result.stdout);
  })
  .catch((err) => {
    console.error("命令执行失败", err);
  });

上传文件到远程服务器

ssh
  .putFile("本地文件路径", "需要上传到服务器的路径")
  .then(() => {
    console.log("文件上传成功");
  })
  .catch((err) => {
    console.error("文件上传失败", err);
  });

下载文件到本地

ssh
  .getFile("remote-path/to/file", "local-path/to/save")
  .then(() => {
    console.log("文件下载成功");
  })
  .catch((err) => {
    console.error("文件下载失败", err);
  });

读取远程文件内容到本地

ssh
  .execCommand("cat 服务器文件路径")
  .then((result) => {
    console.log("文件内容:", result.stdout);
  })
  .catch((err) => {
    console.error("执行 cat 命令时出错:", err);
  });

关闭 SSH 连接

ssh
  .dispose()
  .then(() => {
    console.log("SSH 连接关闭");
  })
  .catch((err) => {
    console.error("关闭 SSH 连接时出错", err);
  });

实现自动化部署具体流程

自动化部署实现流程图

基于 node-ssh 实现前后台项目自动部署本文主要介绍了基于 node-ssh 实现前后台项目自动发布的工具实现,其

收集服务器及项目相关信息

通过 commander 来解析命令行参数

用户可以通过 dnhyxc-ci publish projectName 命令行携带了 -h-p-u-m-l-r-s-i 等参数,表明服务器的 host、端口号、用户名、密码、本地文件路径、远程文件路径、是否是服务端项目、是否需要安装依赖等信息。通过这些携带的参数可以减少用户的输入,提高用户的操作效率。如果没有携带这些参数,那么就需要用户手动输入相关信息。通过如下 prompts 进行收集。

#!/usr/bin/env node

import { program } from "commander"; // 解析命令行参
import chalk from "chalk"; // 终端标题美化
import { updateVersion } from "@ci/utils";
import { publish, Options } from "@ci/publish";
import pkg from "./package.json";

program.version(updateVersion(pkg.version), "-v, --version");

program
  .name('dnhyxc-ci')
  .description('自动部署工具')
  .usage('<command> [options]')
  .on('--help', () => {
    console.log(`\r\nRun ${chalk.cyan('dnhyxc-ci <command> --help')} for detailed usage of given command\r\n`);
  });

const publishCallback = async (name: string, options: Options) => {
  await publish(name, options);
};

program
  .command('publish <name>')
  .description('项目部署')
  .option('-h, --host [host]', '输入host')
  .option('-p, --port [port]', '输入端口号')
  .option('-u, --username [username]', '输入用户名')
  .option('-m, --password [password]', '输入密码')
  .option('-l, --localFilePath [localFilePath]', '输入本地文件路径,必须以 / 开头')
  .option('-r, --remoteFilePath [remoteFilePath]', '输入服务器目标文件路径,必须以 / 开头')
  .option('-s, --isServer [isServer]', '是否是 node 服务端项目,只允许输入 true 或 false')
  .option('-i, --install', '是否需要安装依赖')
  .action((name, option) => {
    if (option?.localFilePath && !isValidFilePath(option?.localFilePath)) {
      console.log(`\n${chalk.redBright('Error: 本地文件路径必须以 / 开头')}\n`);
      process.exit(1);
    }
    if (option?.remoteFilePath && !isValidFilePath(option?.remoteFilePath)) {
      console.log(`\n${chalk.redBright('Error: 服务器目标文件路径必须以 / 开头')}\n`);
      process.exit(1);
    }
    if (option?.isServer && !['true', 'false'].includes(option.isServer)) {
      console.log(`\n${chalk.redBright('Error: -s 只能携带 true 或 false,如 -s true')}\n`);
      process.exit(1);
    }
    publishCallback(name, option);
  });

// 必须写在所有的 program 语句之后,否则上述 program 语句不会执行
program.parse(process.argv);

通过 require 导入用户需要发布的项目根目录下的 publish.config.json 中的发布配置,如果没有配置,那么就需要用户手动输入配置信息,这样可能会增加用户的操作复杂度,同时容错率也会降低,因此建议提前在项目根目录下配置好:

export const getPublishConfig = () => {
  try {
    const config = JSON.parse(fs.readFileSync(`${ompatiblePath(process.cwd(), 'publish.config.json')}`, 'utf8'));
    return config;
  } catch (error) {
    console.log(beautyLog.warning, chalk.redBright(`未找到 ${chalk.cyan('publish.config.json')} 相关发布配置`));
    return null;
  }
};

publish.config.json 发布配置示例:

{
  "serverInfo": {
    "host": "106.69.29.11",
    "username": "root",
    "port": "22"
  },
  "nginxInfo": {
    "remoteFilePath": "/usr/local/nginx/conf",
    "restartPath": "/usr/local/nginx/sbin"
  },
  "serviceInfo": {
    "restartPath": "/usr/local/server"
  },
  "dnhyxc": {
    "name": "dnhyxc",
    "localFilePath": "/Users/dnhyxc/Documents/code/dnhyxc",
    "remoteFilePath": "/usr/local/nginx/dnhyxc",
    "isServer": false
  },
  "example": {
    "name": "dnhyxc",
    "localFilePath": "/Users/dnhyxc/Documents/code/dnhyxc",
    "remoteFilePath": "/usr/local/nginx/dnhyxc",
    "isServer": false
  },
  "blogClientWeb": {
    "name": "html",
    "localFilePath": "/Users/dnhyxc/Documents/code/blog-client-web",
    "remoteFilePath": "/usr/local/nginx/html",
    "isServer": false
  },
  "blogAdminWeb": {
    "name": "admin_html",
    "localFilePath": "/Users/dnhyxc/Documents/code/blog-admin-web",
    "remoteFilePath": "/usr/local/nginx/html_admin",
    "isServer": false
  },
  "blogServerWeb": {
    "name": "server",
    "localFilePath": "/Users/dnhyxc/Documents/code/blog-server-web",
    "remoteFilePath": "/usr/local/server",
    "isServer": true
  }
}
通过 prompts 收集用户输入的服务器地址、用户名、密码等信息

当用户在需要发布的项目根目录下配置了 publish.config.json 文件时,用户只需要输入密码连接服务器后,就能直接发布项目了,如果没有配置,或者携带相关的的参数,那么对应的参数就需要用户通过 prompts 手动输入,在收集到用户输入的信息之后,才能完成发布,这种方式相对繁琐,因此,建议提前在项目根目录下配置好发布项目相关的配置。

import prompts from "prompts";
import chalk from "chalk";
import { beautyLog } from "./utils";

export const publish = async (projectName: string, options: Omit<Options, 'isServer'> & { isServer: string }) => {
  const {
    host: _host,
    port: _port,
    username: _username,
    password: _password,
    localFilePath: _localFilePath,
    remoteFilePath: _remoteFilePath,
    install: _install,
    isServer: _isServer
  } = options;

  // 标识是否已经校验
  let isVerified = false;

  const publishConfig: PublishConfigParams = getPublishConfig();

  const isService = getPublishConfigInfo(publishConfig, projectName, 'isServer');

  const localPath =
    _localFilePath ||
    (getPublishConfigInfo(publishConfig, projectName, 'localFilePath') as string) ||
    `${process.cwd()}`;

  // 发布配置中 isServer 配置存在时,直接校验
  if (localPath && (isService !== undefined || _isServer !== undefined)) {
    onVerifyFile(localPath, _isServer === 'true' || !!isService);
    isVerified = true;
  }

  try {
    const verifyIsServer = (options: Options) => {
      /**
       * 判断是否输入了 isServer 选项,并且 isServer 选项的值为 true 时,则显示安装依赖选项
       * 判断是否携带 -i 参数,如果未携带,则显示安装依赖选项
       * 如果 publish.config.json 中配置了 isServer 为 true 时,则显示安装依赖选项
       */
      const needInstall = (_isServer === 'true' || options.isServer || isService) && _install === undefined;

      !isVerified &&
        onVerifyFile(
          _localFilePath ||
            options.localFilePath ||
            (getPublishConfigInfo(publishConfig, projectName, 'localFilePath') as string) ||
            process.cwd(),
          _isServer === 'true' || options.isServer || !!isService
        );

      return needInstall;
    };

    result = await prompts(
      [
        {
          name: 'host',
          type: _host || getPublishConfigInfo(publishConfig, 'serverInfo', 'host', true) ? null : 'text',
          message: 'host:',
          initial: getPublishConfigInfo(publishConfig, 'serverInfo', 'host') || '',
          validate: (value) => (value ? true : '请输入host')
        },
        {
          name: 'port',
          type: _port || getPublishConfigInfo(publishConfig, 'serverInfo', 'port', true) ? null : 'text',
          message: '端口号:',
          initial: getPublishConfigInfo(publishConfig, 'serverInfo', 'port') || '',
          validate: (value) => (value ? true : '请输入端口号')
        },
        {
          name: 'localFilePath',
          type:
            _localFilePath || getPublishConfigInfo(publishConfig, projectName, 'localFilePath', true) ? null : 'text',
          message: '本地项目文件路径:',
          initial: process.cwd(),
          validate: (value) => (value ? true : '请输入本地项目文件路径')
        },
        {
          name: 'remoteFilePath',
          type:
            _remoteFilePath || getPublishConfigInfo(publishConfig, projectName, 'remoteFilePath', true) ? null : 'text',
          message: '目标服务器项目文件路径:',
          initial: getPublishConfigInfo(publishConfig, projectName, 'remoteFilePath') || '',
          validate: (value) => (value ? true : '请输入目标服务器项目文件路径')
        },
        {
          name: 'isServer',
          type:
            _isServer !== undefined ||
            _install ||
            getPublishConfigInfo(publishConfig, projectName, 'isServer', true) !== undefined
              ? null
              : 'toggle',
          message: '是否是后台服务:',
          initial: false,
          active: 'yes',
          inactive: 'no'
        },
        {
          name: 'install',
          type: (_, values) => {
            // isServer 为 true 时,或者 publish.config.json 中没有配置 isServer 为 true 时,或者 _install 没有传入了值时,才显示安装依赖选项
            return !verifyIsServer(values) ? null : 'toggle';
          },
          message: '是否安装依赖:',
          initial: false,
          active: 'yes',
          inactive: 'no'
        },
        {
          name: 'username',
          type: _username || getPublishConfigInfo(publishConfig, 'serverInfo', 'username', true) ? null : 'text',
          message: '用户名称:',
          initial: getPublishConfigInfo(publishConfig, 'serverInfo', 'username') || '',
          validate: (value) => (value ? true : '请输入用户名称')
        },
        {
          name: 'password',
          type: _password ? null : 'password',
          message: '密码:',
          validate: (value) => (value ? true : '请输入密码')
        }
      ],
      {
        onCancel: () => {
          console.log(`\n${beautyLog.error}`, chalk.red('已取消输入配置信息\n'));
          process.exit(1);
        }
      }
    );
  } catch (err) {
    console.log(`\n${beautyLog.error}`, `${chalk.red('请检查 publish.config.json 发布配置或者输入信息是否有误!')}\n`);
    console.log(beautyLog.error, chalk.red(`${err}!\n`));
    process.exit(1);
  }

  const { host, port, username, password, localFilePath, remoteFilePath, install, isServer } = result;

  await onPublish({
    host: host || _host || (getPublishConfigInfo(publishConfig, 'serverInfo', 'host') as string),
    port: port || _port || (getPublishConfigInfo(publishConfig, 'serverInfo', 'port') as string),
    username: username || _username || (getPublishConfigInfo(publishConfig, 'serverInfo', 'username') as string),
    password: password || _password,
    localFilePath:
      localFilePath ||
      _localFilePath ||
      (getPublishConfigInfo(publishConfig, projectName, 'localFilePath') as string) ||
      process.cwd(),
    remoteFilePath:
      remoteFilePath ||
      _remoteFilePath ||
      (getPublishConfigInfo(publishConfig, projectName, 'remoteFilePath') as string),
    install: install || _install,
    isServer: _isServer ? _isServer === 'true' : isServer || !!isService
  });
};

连接服务器

通过 node-ssh 实现服务器的连接,并在连接成功后,执行相关的命令或操作。具体实现如下:

export const onConnectServer = async ({
  host,
  port,
  username,
  password,
  ssh
}: Pick<Options, 'host' | 'port' | 'username' | 'password'> & { ssh: NodeSSH }) => {
  const spinner = ora({
    text: chalk.yellowBright(chalk.cyan(`正在连接服务器: ${username}@${host}:${port} ...`))
  }).start();
  try {
    // 连接到服务器
    await ssh.connect({
      host,
      username,
      port,
      password,
      tryKeyboard: true
    });
    spinner.succeed(chalk.greenBright('服务器连接成功!!!'));
  } catch (err) {
    spinner.fail(chalk.redBright(`服务器连接失败: ${err}`));
    process.exit(1);
  }
};

压缩本地文件

服务器连接成功之后,再通过 archiver 插件实现对本地项目打包好的 dist 文件进行压缩。以便上传到服务器上。其中需要区分是前台项目还是后台 node 服务端项目,前台只需要将项目打包出的 dist 文件进行打包即可,而后台项目是将项目目录下的 src、package.json 等文件打包成一个压缩包,然后上传到服务器上。具体实现如下:

  • 打包前台项目:
// localFilePath:本地项目的文件路径 /Users/dnhyxc/Documents/code/blog-client-web
const onCompressFile = async (localFilePath: string) => {
  return new Promise((resolve, reject) => {
    const spinner = ora({
      text: chalk.yellowBright(`正在压缩本地文件: ${chalk.cyan(ompatiblePath(localFilePath, 'dist'))} ...`)
    }).start();
    const archive = archiver('zip', {
      zlib: { level: 9 }
    }).on('error', (err: Error) => {
      console.log(beautyLog.error, chalk.red(`压缩本地文件失败: ${err}`));
    });
    const output = fs.createWriteStream(ompatiblePath(localFilePath, 'dist.zip'));
    output.on('close', (err: Error) => {
      if (err) {
        spinner.fail(chalk.redBright(`压缩文件: ${chalk.cyan(ompatiblePath(localFilePath, 'dist'))} 失败`));
        console.log(beautyLog.error, chalk.red(`压缩本地文件失败: ${err}`));
        reject(err);
        process.exit(1);
      }
      spinner.succeed(chalk.greenBright(`压缩本地文件: ${chalk.cyan(ompatiblePath(localFilePath, 'dist'))} 成功`));
      resolve(1);
    });
    archive.pipe(output);
    // 第二参数表示在压缩包中创建 dist 目录,将压缩内容放在 dist 目录下,而不是散列到压缩包的根目录
    archive.directory(ompatiblePath(localFilePath, 'dist'), '/dist');
    archive.finalize();
  });
};
  • 压缩后台服务项目:
// localFilePath:本地项目的文件路径 /Users/dnhyxc/Documents/code/blog-client-web
const onPutFile = async (localFilePath: string, remoteFilePath: string) => {
  try {
    const progressBar = new cliProgress.SingleBar({
      format: '文件上传中: {bar} | {percentage}% | ETA: {eta}s | {value}MB / {total}MB',
      barCompleteChar: '\u2588',
      barIncompleteChar: '\u2591',
      hideCursor: true
    });
    const localFile = path.resolve(__dirname, `${localFilePath}/dist.zip`);
    const remotePath = path.join(remoteFilePath, path.basename(localFile));
    const stats = fs.statSync(localFile);
    const fileSize = stats.size;
    progressBar.start(Math.ceil(fileSize / 1024 / 1024), 0);
    await ssh.putFile(localFile, remotePath, null, {
      concurrency: 10, // 控制上传的并发数
      chunkSize: 16384, // 指定每个数据块的大小,适应慢速连接 16kb
      step: (totalTransferred: number) => {
        progressBar.update(Math.ceil(totalTransferred / 1024 / 1024));
      }
    });
    progressBar.stop();
  } catch (error) {
    console.log(beautyLog.error, chalk.red(`上传文件失败: ${error}`));
    process.exit(1);
  }
};

上传本地文件到服务器

文件压缩完成之后,再通过 ssh.putFile 上传打包好的文件到远程服务器的指定目录。

// localFilePath:本地项目的文件路径 /Users/dnhyxc/Documents/code/blog-client-web
// remoteFilePath:远程服务器上的项目路径 /usr/local/nginx/html
const onPutFile = async (localFilePath: string, remoteFilePath: string) => {
  try {
    const progressBar = new cliProgress.SingleBar({
      format: '文件上传中: {bar} | {percentage}% | ETA: {eta}s | {value}MB / {total}MB',
      barCompleteChar: '\u2588',
      barIncompleteChar: '\u2591',
      hideCursor: true
    });
    const localFile = path.resolve(__dirname, `${localFilePath}/dist.zip`);
    const remotePath = path.join(remoteFilePath, path.basename(localFile));
    const stats = fs.statSync(localFile);
    const fileSize = stats.size;
    progressBar.start(Math.ceil(fileSize / 1024 / 1024), 0);
    await ssh.putFile(localFile, remotePath, null, {
      concurrency: 10, // 控制上传的并发数
      chunkSize: 16384, // 指定每个数据块的大小,适应慢速连接 16kb
      step: (totalTransferred: number) => {
        progressBar.update(Math.ceil(totalTransferred / 1024 / 1024));
      }
    });
    progressBar.stop();
  } catch (error) {
    console.log(beautyLog.error, chalk.red(`上传文件失败: ${error}`));
    process.exit(1);
  }
};

删除服务器上的 dist 文件

当文件上传完毕之后,删除服务器指定目录上的 dist 文件,以便解压服务器上刚上传的 dist.zip 文件。

// localFile:本地项目的 dist 文件路径 /Users/dnhyxc/Documents/code/blog-client-web/dist
const onDeleteFile = async (localFile: string) => {
    const spinner = ora({
        text: chalk.yellowBright(`正在删除服务器文件: ${chalk.cyan(localFile)}`),
    }).start();
    try {
        await ssh.execCommand(`rm -rf ${localFile}`);
        spinner.succeed(chalk.greenBright(`删除服务器文件:${chalk.cyan(`${localFile}`)} 成功`));
    } catch (err) {
      spinner.fail(chalk.redBright(`删除服务器文件: ${chalk.cyan(`${localFile}`)} 失败,${err}`));
      process.exit(1);
    }
};

解压服务器上刚上传的 dist.zip 文件

当文件上传成功之后,解压服务器上刚上传的 dist.zip 文件到指定的目录下,解压完成之后,删除服务器上的 dist.zip 文件。

// remotePath:远程服务器上的项目路径 /usr/local/server
const onUnzipZip = async (remotePath: string, isServer: boolean) => {
  remotePath = ompatiblePath(remotePath);
  const spinner = ora({
    text: chalk.yellowBright(`正在解压服务器文件: ${chalk.cyan(`${remotePath}/dist.zip`)} ...`)
  }).start();
  try {
    await ssh.execCommand(`unzip -o ${`${remotePath}/dist.zip`} -d ${remotePath}`);
    spinner.succeed(chalk.greenBright(`解压服务器文件: ${chalk.cyan(`${remotePath}/dist.zip`)} 成功`));
    await onRemoveServerFile(`${remotePath}/dist.zip`, ssh);
    !isServer &&
      console.log(
        `\n${beautyLog.success}`,
        chalk.greenBright(`${chalk.bold(`🎉 🎉 🎉 前端资源部署成功: ${chalk.cyan(`${remotePath}`)} 🎉 🎉 🎉`)}\n`)
      );
  } catch (err) {
    console.log(beautyLog.error, chalk.red(`Failed to unzip dist.zip: ${err}`));
    spinner.fail(chalk.redBright(`解压服务器文件: ${chalk.cyan(`${remotePath}/dist.zip`)} 失败`));
    process.exit(1);
  }
};

如果发布的是前端资源,到这一步,如果控制台没有发现错误,那么说明前端资源已经发布成功了,就可以到浏览器验证资源是否生效了。如果是 node 服务端项目,则可能还需要经过下文中的依赖安装和服务重启等步骤。

删除本地 dist.zip 文件

当文件解压成功之后,删除本地的 dist.zip 文件,当然也可以不删除,看个人意愿。

export const onRemoveFile = async (localFile: string) => {
  const fullPath = ompatiblePath(localFile);
  const spinner = ora({
    text: chalk.yellowBright(`正在删除本地文件: ${chalk.cyan(fullPath)} ...`)
  }).start();
  return new Promise((resolve) => {
    try {
      // 删除文件
      fs.unlink(fullPath, (err) => {
        if (err === null) {
          spinner.succeed(chalk.greenBright(`删除本地文件: ${chalk.cyan(fullPath)} 成功`));
          resolve(1);
        }
      });
    } catch (err) {
      spinner.fail(chalk.redBright(`删除本地文件: ${chalk.cyan(fullPath)} 失败,${err}`));
      process.exit(1);
    }
  });
};

服务端项目安装依赖

如果发布的是 node 服务端项目,需要根据收集到的信息,判断是否需要安装依赖,如果需要则需要通过如下方式进行安装。

const onInstall = async (remotePath: string) => {
  remotePath = ompatiblePath(remotePath);
  const spinner = ora({
    text: chalk.yellowBright(chalk.cyan('正在安装依赖...'))
  }).start();
  try {
    const { code, stdout, stderr } = await ssh.execCommand(`cd ${remotePath} && yarn install`);
    if (code === 0) {
      spinner.succeed(chalk.greenBright(`依赖安装成功: \n ${stdout} \n`));
    } else {
      spinner.fail(chalk.redBright(`依赖安装失败: ${stderr}`));
      process.exit(1);
    }
  } catch (error) {
    spinner.fail(chalk.redBright(`依赖安装失败: ${error}`));
    process.exit(1);
  }
};

重启 node 服务

发布的是 node 服务端项目时,在文件解压完成及依赖安装完成之后,需要重启 node 服务,以便让新发布的功能生效,此时我们可以利用 node-ssh 及服务器上的 pm2 实现对服务的重启。

export const onRestartServer = async (remotePath: string, ssh: NodeSSH) => {
  remotePath = ompatiblePath(remotePath);
  const spinner = ora({
    text: chalk.yellowBright(chalk.cyan('正在重启服务...'))
  }).start();
  try {
    const { code: restartCode, stderr: restartStderr } = await ssh.execCommand('pm2 restart 0');
    const { code: listCode, stdout } = await ssh.execCommand('pm2 list');
    if (restartCode === 0 && listCode === 0) {
      spinner.succeed(chalk.greenBright(`服务启动成功: \n${stdout}`));
      console.log(
        `\n${beautyLog.success}`,
        chalk.greenBright(`${chalk.bold(`🎉 🎉 🎉 node 服务重启成功: ${chalk.cyan(`${remotePath}`)}!!! 🎉 🎉 🎉 \n`)}`)
      );
    } else {
      spinner.fail(chalk.redBright(`服务启动失败: ${restartStderr}`));
      process.exit(1);
    }
  } catch (error) {
    spinner.fail(chalk.redBright(`服务启动失败: ${error}`));
    process.exit(1);
  }
};

至此,如果控制台没有报错,那么就预示着项目已经成功发布了,此时就可以去浏览器上查看前台项目发布的内容是否生效了。

实现服务器 nginx 配置文件的修改和发布

在修改 nginx 配置文件时,每次都需要通过 xshell 连接服务器,或者其他方式连接服务器,然后修改配置文件,而且修改的时候也不太方便,同时不能像在编辑器中修改文件这么的简洁明了。因此,就利用 node-ssh 模块实现对 nginx 配置文件的修改和发布。

解析命令行参数

通过 commander 解析用户执行 dnhyxc-ci pull 时携带的参数,包括 -h-p-u-m-ncp 等,从而获取用户需要发布的服务器 host、端口、用户名、密码及远程 nginx 配置文件路径等相关信息。如果用户没有携带这些信息,则需要通过 prompts 进行交互式的输入。

import { program } from "commander"; // 解析命令行参
import chalk from "chalk"; // 终端标题美化

const pullNginxConfCallback = async (
	name: string,
	option: Options & CollectInfoParams
) => {
	await pull(name, option);
};

program
  .command('pull [configName]')
  .description('获取 nginx.conf 配置文件到本地')
  .option('-h, --host [host]', '输入host')
  .option('-p, --port [port]', '输入端口号')
  .option('-u, --username [username]', '输入用户名')
  .option('-m, --password [password]', '输入密码')
  .option('-ncp, --nginxRemoteFilePath [nginxRemoteFilePath]', '输入服务器 nginx.conf 文件路径,必须以 / 开头')
  .action((configName, option) => {
    if (option?.nginxRemoteFilePath && !isValidFilePath(option?.nginxRemoteFilePath)) {
      console.log(`\n${chalk.redBright('Error: nginx.conf 文件路径必须以 / 开头')}\n`);
      process.exit(1);
    }
    pullNginxConfCallback(configName, option);
  });

// 必须写在所有的 program 语句之后,否则上述 program 语句不会执行
program.parse(process.argv);

通过 prompts 收集用户输入信息

如果用户在需要发布的项目根目录下的 publish.config.json 文件中配置了项目发布的相关信息,则只需要输入密码后就可完成 nginx.conf 文件的拉取操作,否则需要输入 host、端口、用户名、密码、远程 nginx 配置文件路径等相关信息。

export const onCollectServerInfo = async ({
  host,
  port,
  username,
  password,
  projectName,
  publishConfig,
  command,
  nginxRemoteFilePath,
  nginxRestartPath,
  serviceRestartPath
}: CollectInfoParams) => {
  try {
    return await prompts(
      [
        {
          name: 'host',
          type: host || getPublishConfigInfo(publishConfig, 'serverInfo', 'host', true) ? null : 'text',
          message: 'host:',
          initial: getPublishConfigInfo(publishConfig, 'serverInfo', 'host') || '',
          validate: (value) => (value ? true : '请输入host')
        },
        {
          name: 'port',
          type: port || getPublishConfigInfo(publishConfig, 'serverInfo', 'port', true) ? null : 'text',
          message: '端口号:',
          initial: getPublishConfigInfo(publishConfig, 'serverInfo', 'port') || '',
          validate: (value) => (value ? true : '请输入端口号')
        },
        {
          name: 'username',
          type: username || getPublishConfigInfo(publishConfig, 'serverInfo', 'username', true) ? null : 'text',
          message: '用户名称:',
          initial: getPublishConfigInfo(publishConfig, 'serverInfo', 'username') || '',
          validate: (value) => (value ? true : '请输入用户名称')
        },
        {
          name: 'nginxRemoteFilePath',
          type:
            nginxRemoteFilePath ||
            getPublishConfigInfo(publishConfig, 'nginxInfo', 'remoteFilePath', projectName !== 'node') ||
            projectName === 'node'
              ? null
              : 'text',
          message: '服务器 nginx.conf 文件路径:',
          initial: getPublishConfigInfo(publishConfig, 'nginxInfo', 'remoteFilePath') || '',
          validate: (value) => (isValidFilePath(value) ? true : '输入的服务器 nginx.conf 文件路径必须以 / 开头')
        },
        /**
         * 当输入了 nginxRestartPath 时、
         * 或配置文件中有 restartPath 时、
         * 或前两者都没有,并且 command 为 pull 时、
         * 或 projectName 等于 nodo 时,不显示 serviceRestartPath 字段
         */
        {
          name: 'nginxRestartPath',
          type:
            nginxRestartPath ||
            getPublishConfigInfo(
              publishConfig,
              'nginxInfo',
              'restartPath',
              (command !== 'pull' && projectName === 'nginx') || command === 'push' // 判断是否需要提示
            ) ||
            (!nginxRestartPath &&
              !getPublishConfigInfo(publishConfig, 'nginxInfo', 'restartPath') &&
              command === 'pull') ||
            projectName === 'node'
              ? null
              : 'text',
          message: '服务器 nginx 重启路径:',
          initial: getPublishConfigInfo(publishConfig, 'nginxInfo', 'restartPath') || '',
          validate: (value) => (isValidFilePath(value) ? true : '输入的服务器 nginx 重启路径必须以 / 开头')
        },
        /**
         * 当输入了 serviceRestartPath 时、
         * 或配置文件中有 restartPath 时、
         * 或前两者都没有,并且 command 为 pull 及 push 时、
         * 或 projectName 等于 nginx 时,不显示 serviceRestartPath 字段
         */
        {
          name: 'serviceRestartPath',
          type:
            serviceRestartPath ||
            getPublishConfigInfo(
              publishConfig,
              'serviceInfo',
              'restartPath',
              command === 'restart' && projectName === 'node' // 判断是否需要提示
            ) ||
            (!serviceRestartPath &&
              !getPublishConfigInfo(publishConfig, 'serviceInfo', 'restartPath') &&
              command !== 'restart') ||
            projectName === 'nginx'
              ? null
              : 'text',
          message: '服务器 node 重启路径:',
          initial: getPublishConfigInfo(publishConfig, 'serviceInfo', 'restartPath') || '',
          validate: (value) => (isValidFilePath(value) ? true : '输入的服务器 node 重启路径必须以 / 开头')
        },
        {
          name: 'password',
          type: password ? null : 'password',
          message: '密码:',
          initial: '',
          validate: (value) => (value ? true : '请输入密码')
        }
      ],
      {
        onCancel: () => {
          console.log(`\n${beautyLog.error}`, chalk.red('已取消输入配置信息\n'));
          process.exit(1);
        }
      }
    );
  } catch (err) {
    console.log(beautyLog.error, chalk.red(err));
    process.exit(1);
  }
};

将 nginx.conf 文件内容写入到本地

通过 ssh.execCommand('cat 服务器 nginx.conf 文件路径') 可以将服务器上的 nginx.conf 内容读取到,并写入到当前需要发布项目根目录下的 nginx.conf 文件中,以便与后续内容的修改,之后再重新发布到服务器上。nginx.conf 发布成功过后,该 nginx.conf 文件将会被删除。

const onReadNginxConfig = async (remotePath: string, localFileName: string) => {
  remotePath = ompatiblePath(remotePath);
  localFileName = ompatiblePath(localFileName);
  const spinner = ora({
    text: chalk.yellowBright(`正在读取远程 ${chalk.cyan(`${remotePath}`)} 文件...`)
  }).start();
  try {
    const result = await ssh.execCommand(`cat ${remotePath}`);
    const nginxConfigContent = result.stdout;
    if (nginxConfigContent) {
      // 写入到本地文件
      await fs.writeFile(localFileName, nginxConfigContent);
      spinner.succeed(
        chalk.greenBright(`读取 nginx.conf 成功,内容已写入到本地 ${chalk.cyan(`${localFileName}`)} 文件中`)
      );
    } else {
      spinner.fail(chalk.redBright(`读取 nginx.conf 失败,远程文件 ${chalk.cyan(`${remotePath}`)} 内容为空`));
      process.exit(1);
    }
  } catch (err) {
    spinner.fail(chalk.redBright(`读取: ${chalk.cyan(`${remotePath}`)} 文件失败,${err}`));
  }
};

将本地 nginx.conf 文件内容上传到服务器

通过 ssh.putFile 将本地 nginx 配置内容上传到服务器的指定目录下。

const onPutNginxConfig = async (localFilePath: string, remoteFilePath: string) => {
  localFilePath = ompatiblePath(localFilePath);
  remoteFilePath = ompatiblePath(remoteFilePath);
  const spinner = ora({
    text: chalk.yellowBright('正在推送 nginx.conf 文件到远程服务器...')
  }).start();
  try {
    // 推送本地文件到远程服务器
    await ssh.putFile(localFilePath, remoteFilePath);
    spinner.succeed(chalk.greenBright(`服务器 ${chalk.cyan(`${remoteFilePath}`)} 内容更新成功`));
  } catch (error) {
    spinner.fail(chalk.redBright(`推送 nginx.conf 文件到服务器失败: ${error}`));
    process.exit(0);
  }
};

重启 nginx 服务

nginx 配置文件上传成功之后,为了使配置能够生效,就需要重启一下 nginx 服务,这可以通过 ssh.execCommand('cd /usr/local/nginx/sbin && ./nginx -s reload') 命令实现。如果你在服务器上设置了 nginx 的环境变量,即可直接通过 ssh.execCommand('nginx -s reload') 命令重启 nginx 服务。

export const onRestartNginx = async (remoteFilePath: string, restartPath: string, ssh: NodeSSH) => {
  await onCheckNginxConfig(remoteFilePath, restartPath, ssh);
  const spinner = ora({
    text: chalk.yellowBright('正在重启 nginx 服务...')
  }).start();
  try {
    await ssh.execCommand(`cd ${ompatiblePath(restartPath)} && ./nginx -s reload`);
    spinner.succeed(chalk.greenBright(`nginx 服务已重启: ${ompatiblePath(restartPath)}`));
    if (verifyFile(`${process.cwd()}/nginx.conf`)) {
      await onRemoveFile(`${process.cwd()}/nginx.conf`);
    }
    console.log(
      `\n${beautyLog.success}`,
      chalk.greenBright(`${chalk.bold(`🎉 🎉 🎉 nginx 服务重启成功 ${ompatiblePath(restartPath)} 🎉 🎉 🎉`)}\n`)
    );
  } catch (error) {
    spinner.fail(chalk.redBright(`重启 nginx 服务失败: ${error}`));
    process.exit(0);
  }
};

重启 nginx 及 node 服务

在某些情况下,我们希望能重启一下 nginx 服务,或者 node 服务,此时也可以通过 node-ssh 模块实现。

解析命令行参数

通过 commander 解析命令行参数,获得用户执行 dnhyxc-ci restart 时携带的参数,包括 -h-p-u-m-ncp-nrp-srp 等,从而获取用户需要重启的服务器 host、端口、用户名、密码、nginx 配置文件路径、nginx 重启路径、node 服务重启路径等相关信息。

import { program } from "commander"; // 解析命令行参
import chalk from "chalk"; // 终端标题美

const restartCallback = async (
	name: string,
	option: Options & CollectInfoParams
) => {
	await restart(name, option);
};

program
  .command('restart <serviceName>')
  .description('重启 nginx 或者 node 服务')
  .option('-h, --host [host]', '输入host')
  .option('-p, --port [port]', '输入端口号')
  .option('-u, --username [username]', '输入用户名')
  .option('-m, --password [password]', '输入密码')
  .option('-ncp, --nginxRemoteFilePath [nginxRemoteFilePath]', '输入服务器 nginx.conf 文件路径,必须以 / 开头')
  .option('-nrp, --nginxRestartPath [nginxRestartPath]', '输入服务器 nginx 重启路径,必须以 / 开头')
  .option('-srp, --serviceRestartPath [serviceRestartPath]', '输入服务器 node 重启路径,必须以 / 开头')
  .action((serviceName, option) => {
    const validatedServiceName = validateServiceName(serviceName);
    if (option?.nginxRemoteFilePath && !isValidFilePath(option?.nginxRemoteFilePath)) {
      console.log(`\n${chalk.redBright('Error: nginx.conf 文件路径必须以 / 开头')}\n`);
      process.exit(1);
    }
    if (option?.nginxRestartPath && !isValidFilePath(option?.nginxRestartPath)) {
      console.log(`\n${chalk.redBright('Error: nginx 重启路径必须以 / 开头')}\n`);
      process.exit(1);
    }
    if (option?.serviceRestartPath && !isValidFilePath(option?.serviceRestartPath)) {
      console.log(`\n${chalk.redBright('Error: node 重启路径必须以 / 开头')}\n`);
      process.exit(1);
    }
    restartCallback(validatedServiceName, option);
  });

// 必须写在所有的 program 语句之后,否则上述 program 语句不会执行
program.parse(process.argv);

通过 prompts 收集用户输入信息

如果用户在需要发布的项目根目录下的 publish.config.json 文件中配置了项目发布的相关信息,则只需要输入密码后就可完成服务的重启操作,否则需要输入服务器 host、端口、用户名、密码、远程 nginx 配置文件路径、远程 nginx 重启路径、远程 node 服务重启路径等相关信息。至于 prompts 具体收集方式在上述 onCollectServerInfo 方法中已经实现,这里不再赘述。

重启 nginx 服务

重启 nginx 服务,可以看上述 onRestartNginx 方法,通过 ssh.execCommand('cd /usr/local/nginx/sbin && ./nginx -s reload') 命令实现。

重启 node 服务

重启 node 服务,可以看上述 onRestartServer 方法,通过 ssh.execCommand('pm2 delete 0')ssh.execCommand('pm2 start 服务器 node 服务路径') 等命令实现。

总结

本文主要介绍了发布前台项目及后台 node 服务端项目的流程,主要涉及到收集用户输入信息、连接服务器、压缩文件、上传文件、删除服务器上的 dist 文件、解压服务器上刚上传的 dist.zip 文件、删除本地 dist.zip 文件、服务端项目安装依赖、重启 node 服务等操作。通过这些操作,实现了一个在项目打包之后可直接自动发布的工具。同时还介绍了,如何通过 node-ssh 模块实现对 nginx 配置文件的修改和发布,以及如何通过 node-ssh 模块实现对 nginx 服务的重启等。

转载自:https://juejin.cn/post/7389651690305765417
评论
请登录