gpt4 book ai didi

javascript - 使用 Promises 处理来自 ssh2 的成功/错误响应

转载 作者:行者123 更新时间:2023-11-29 23:28:28 25 4
gpt4 key购买 nike

我正在构建一个 node.js 应用程序,它在生产中将充当许多服务器的 SSH 客户端,其中一些服务器可能在任何给定时间都无法访问。我正在尝试编写一个函数,该函数在启动时尝试在其配置中的每个客户端运行 SSH 命令,但我无法同时处理成功的 session 和以错误结束的 session 。我在 promise 中包装了一个 ssh2 客户端。如果我删除了第三个(垃圾)服务器并且只成功了结果,这工作正常!查看输出:

STDOUT: Hello World

STDOUT: Hello World

Session closed
Session closed
Successful session: Hello World,Hello World

但是,如果其中一个连接超时,即使我处理了错误,我也无法保留任何数据。看起来错误消息覆盖了所有已解决的 promise

Successful session: Error: Timed out while waiting for handshake,Error: 
Timed out while waiting for handshake,Error: Timed out while waiting
for handshake

这是我的代码,如果这有点分散,请原谅我,因为为了这个问题,我已经组合了一些我的模块。我的目标是保留成功 session 的数据并优雅地处理失败。

var Client = require('ssh2').Client;

const labs = {
"ny1": "192.168.1.2",
"ny2": "192.168.1.3",
"ny3": "1.1.1.1"
};

function checkLabs() {
let numLabs = Object.keys(labs).length;
let promises = [];

for(i=0;i<numLabs;i++){
let labName = Object.keys(labs)[i];
promises.push(asyncSSH("echo 'Hello World'", labs[labName]));
}

Promise.all(promises.map(p => p.catch(e => e)))
.then(results => console.log("Successful session: " + results))
.catch(e => console.log("Error! " + e));
}

var sendSSH = function (command, dest, callback) {
var conn = new Client();

conn.on('ready', function() {
return conn.exec(command, function(err, stream) {
if (err) throw err;
stream.on('data', function(data) {
callback(null, data);
console.log('STDOUT: ' + data);
}).stderr.on('data', function(data){
callback(err);
console.log('STDERR: ' + data);
}).on('close', function(err) {
if(err) {
console.log('Session closed due to error');
} else {
console.log('Session closed');
}
});
stream.end('ls -l\nexit\n');
});
}).on('error', function(err){
callback(err);
}).connect({
host: dest,
port: 22,
username: 'root',
readyTimeout: 10000,
privateKey: require('fs').readFileSync('link-to-my-key')
});
};

function asyncSSH(command, dest) {
return new Promise(function(resolve, reject){
sendSSH(command, dest, function(err,data) {
if (!err) {
resolve(data);
} else {
reject(err);
}
});
});
}

checklabs();

我怎样才能更好地使用这个 promise 包装器来处理来自 ssh2 客户端的任何错误?任何提示表示赞赏。

最佳答案

为了充分利用每个连接,您可以(并且可以说应该)单独 promise :

  • 每个 Client() 实例的实例化
  • 每个实例的 conn.exec() 方法(以及任何其他需要的异步方法)

这将允许 Client() 的每个实例被不同的命令重用(虽然在这个例子中不是必需的)。

您还应该确保通过调用 client_.end() 断开每个套接字的连接。为此,建议使用“处置器模式”。

考虑到这些要点和一些假设,这就是我的最终结果:

var Client = require('ssh2').Client;

const labs = {
"ny1": "192.168.1.2",
"ny2": "192.168.1.3",
"ny3": "1.1.1.1"
};

function checkLabs() {
let promises = Object.keys(labs).map((key) => {
return withConn(labs[key], (conn) => {
return conn.execAsync("echo 'Hello World'")
.catch((e) => "Error: " + e.message); // catch in order to immunise the whole process against any single failure.
// and inject an error message into the success path.
});
});
Promise.all(promises)
.then(results => console.log("Successful session: " + results))
.catch(e => console.log("Error! " + e.message)); // with individual errors caught above, you should not end up here.
}

// disposer pattern, based on https://stackoverflow.com/a/28915678/3478010
function withConn(dest, work) {
var conn_;
return getConnection(dest).then((conn) => {
conn_ = conn;
return work(conn);
}).then(() => {
if(conn_) {
conn_.end(); // on success, disconnect the socket (ie dispose of conn_).
}
}, () => {
if(conn_) {
conn_.end(); // on error, disconnect the socket (ie dispose of conn_).
}
});
// Note: with Bluebird promises, simplify .then(fn,fn) to .finally(fn).
}

function getConnection(dest) {
return new Promise((resolve, reject) => {
let conn = promisifyConnection(new Client());
conn.on('ready', () => {
resolve(conn);
})
.on('error', reject)
.connect({
host: dest,
port: 22,
username: 'root',
readyTimeout: 10000,
privateKey: require('fs').readFileSync('link-to-my-key')
});
});
}

function promisifyConnection(conn) {
conn.execAsync = (command) => { // promisify conn.exec()
return new Promise((resolve, reject) => {
conn.exec(command, (err, stream) => {
if(err) {
reject(err);
} else {
let streamSegments = []; // array in which to accumulate streamed data
stream.on('close', (err) => {
if(err) {
reject(err);
} else {
resolve(streamSegments.join('')); // or whatever is necessary to combine the accumulated stream segments
}
}).on('data', (data) => {
streamSegments.push(data);
}).stderr.on('data', function(data) {
reject(new Error(data)); // assuming `data` to be String
});
stream.end('ls -l\nexit\n'); // not sure what this does?
}
});
});
};
// ... promisify any further Client methods here ...
return conn;
}

注意事项:

  • conn.exec() 的 promise 包括一个假设,即可以在一系列段(例如数据包)中接收数据。如果此假设不成立,则不再需要 streamSegments 数组。
  • getConnection()promisifyConnection() 可以写成一个函数,但使用单独的函数更容易看出发生了什么。
  • getConnection()promisifyConnection() 让所有困惑的东西远离应用程序代码。

关于javascript - 使用 Promises 处理来自 ssh2 的成功/错误响应,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48199109/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com