前端自动化部署之ssh2和ssh2-sftp-client

ssh2-sftp-client 本身是一个专门用于处理 SFTP文件操作的库,它不直接提供执行远程命令的功能。但是可以通过它的底层依赖库 ssh2 实现执行命令的功能。

以下是实现方法和示例代码:


方法一:使用 ssh2 执行远程命令

ssh2ssh2-sftp-client 的底层库,它支持执行远程命令。

安装依赖

如果尚未安装 ssh2,可以通过以下命令安装:

bash 复制代码
npm install ssh2
代码示例
javascript 复制代码
const { Client } = require('ssh2');

const serverOptions = {
  host: 'your-server-ip',
  port: 22,
  username: 'your-username',
  password: 'your-password', // 或者使用 privateKey
};

const command = 'ls -l /home/cuijiying/log_tools';

const conn = new Client();

conn.on('ready', () => {
  console.log('SSH 连接成功!');
  conn.exec(command, (err, stream) => {
    if (err) throw err;

    stream
      .on('close', (code, signal) => {
        console.log(`命令执行完成,退出代码:${code}`);
        conn.end();
      })
      .on('data', (data) => {
        console.log('输出:\n' + data.toString());
      })
      .stderr.on('data', (data) => {
        console.error('错误:\n' + data.toString());
      });
  });
}).connect(serverOptions);
输出解释
  • 标准输出通过 stream.on('data') 接收。
  • 错误输出通过 stream.stderr.on('data') 接收。

方法二:结合 ssh2-sftp-clientssh2

如果你同时需要执行命令和进行 SFTP 文件操作,可以同时使用 ssh2ssh2-sftp-client

代码示例
javascript 复制代码
const { Client } = require('ssh2');
const SftpClient = require('ssh2-sftp-client');

const serverOptions = {
  host: 'your-server-ip',
  port: 22,
  username: 'your-username',
  password: 'your-password',
};

const command = 'df -h'; // 示例命令:查看磁盘使用情况
const sftp = new SftpClient();
const conn = new Client();

async function executeCommand() {
  return new Promise((resolve, reject) => {
    conn.on('ready', () => {
      console.log('SSH 连接成功!');
      conn.exec(command, (err, stream) => {
        if (err) return reject(err);

        let output = '';
        stream
          .on('close', (code, signal) => {
            console.log(`命令执行完成,退出代码:${code}`);
            conn.end();
            resolve(output);
          })
          .on('data', (data) => {
            output += data.toString();
          })
          .stderr.on('data', (data) => {
            console.error('错误:\n' + data.toString());
          });
      });
    }).connect(serverOptions);
  });
}

async function main() {
  try {
    // 执行远程命令
    const commandOutput = await executeCommand();
    console.log('命令输出:\n', commandOutput);

    // 使用 SFTP 上传文件
    await sftp.connect(serverOptions);
    await sftp.put('./example.zip', '/home/cuijiying/example.zip');
    console.log('文件上传成功!');
  } catch (err) {
    console.error('操作失败:', err);
  } finally {
    sftp.end();
  }
}

main();

方法三:使用 ssh2 的高级特性(Shell 模式)

如果需要执行多个命令,可以使用 Shell 模式。

代码示例
javascript 复制代码
const { Client } = require('ssh2');

const serverOptions = {
  host: 'your-server-ip',
  port: 22,
  username: 'your-username',
  password: 'your-password',
};

const conn = new Client();

conn.on('ready', () => {
  console.log('SSH 连接成功!');
  conn.shell((err, stream) => {
    if (err) throw err;

    stream
      .on('close', () => {
        console.log('Shell 会话结束');
        conn.end();
      })
      .on('data', (data) => {
        console.log('输出:\n' + data.toString());
      });

    // 执行多个命令
    stream.write('pwd\n');
    stream.write('ls -l\n');
    stream.write('exit\n');
  });
}).connect(serverOptions);

注意事项

  1. 权限问题

    • 确保 SSH 用户具有执行命令的权限。如果是限制用户(如 chroot 环境),可能无法正常执行某些命令。
  2. 命令路径

    • 某些命令可能需要完整路径,例如 /usr/bin/ls 而非 ls
  3. 错误处理

    • 捕获所有异常,并合理处理连接超时或命令失败等问题。
  4. 谨慎使用rm 相关命令

    • 哈哈,这是一个极其危险的命令,不懂得可以自行搜索。

通过以上方法,你可以在 Node.js 环境中使用 ssh2 执行远程命令,同时结合 ssh2-sftp-client 进行文件传输,满足大多数 SSH 操作需求。

相关推荐
nujnewnehc5 小时前
ps, ai, ae插件都可以用html和js开发了
前端·javascript
Jagger_8 小时前
整洁架构三连问:是什么,怎么做,为什么要用
前端
一个处女座的程序猿O(∩_∩)O8 小时前
React 完全入门指南:从基础概念到组件协作
前端·react.js·前端框架
前端摸鱼匠8 小时前
Vue 3 的defineEmits编译器宏:详解<script setup>中defineEmits的使用
前端·javascript·vue.js·前端框架·ecmascript
里欧跑得慢9 小时前
Flutter 测试全攻略:从单元测试到集成测试的完整实践
前端·css·flutter·web
Jagger_9 小时前
前端整洁架构详解
前端
徐小夕9 小时前
我花一天时间Vibe Coding的开源AI工具,一键检测你的电脑能跑哪些AI大模型
前端·javascript·github
英俊潇洒美少年9 小时前
Vue3 企业级封装:useEventListener + 终极版 BaseEcharts 组件
前端·javascript·vue.js
嵌入式×边缘AI:打怪升级日志9 小时前
使用JsonRPC实现前后台
前端·后端
小码哥_常10 小时前
深度剖析:为什么Android选择了Binder
前端