javascript - 使用 Node.js 和 SSH2 从 SFTP 服务器读取文件

标签 javascript node.js buffer sftp readable

我在使用 Node.js 中的读取流时遇到了一个非常奇怪的问题。我正在使用 SSH2 在我和 sftp 服务器之间创建 sftp 连接。然后我尝试从 sftp 流创建一个读取流。从读取流发出的“数据”事件中,我将数据附加到一个数组。当读取流的“关闭”事件发生时,我调用 Buffer.concat 以将我检索到的所有数据 block 连接到一个缓冲区中。这与此处在堆栈溢出时提出的其他问题中描述的技术相同。例如here .但是,我无法使用我检索到的数据。看起来缓冲区的大小比我要检索的文件小 32 个字节(通过计算检索到的数据的长度)。这可能与我的 SFTP 连接有关吗?或者我如何创建我的阅读流?

如果重要的话,文件是 zip 类型。当我在读取文件以进行缓冲后尝试解压缩文件(在 node.js 中和手动)时,它不起作用。

经过调查我发现:

  1. 当我对文件使用 readdir 时,文件的大小是正确的。
  2. 对我的开发 FTP 服务器使用 FTP (JSFTP) 使用上述相同的技术工作得很好。

感谢任何建议!

这是我的代码:

        var Client = require('ssh2').Client;
        var m_ssh2Credentials = {
           host: config.ftpHostName,
           port: config.ftpPort,
           username: config.ftpUser,
           password: config.ftpPassword,
           readyTimeout: 20000,
           algorithms: { cipher: ["3des-cbc", "aes256-cbc", "aes192-cbc","aes128-cbc"]}
        };
        ...
        var conn = new Client();
        var dataLength = 0;
        conn.on('ready', function() {
            conn.sftp(function(err, sftp) {
                if (err) {
                    writeToErrorLog("downloadFile(): Failed to open SFTP connection.");
                } else {
                    writeToLog("downloadFile(): Opened SFTP connection.");
                }

                var streamErr = "";
                var dataLength = 0;
                var stream = sftp.createReadStream(config.ftpPath + "/" + m_fileName)
                stream.on('data', function(d){
                    data.push(d);
                    dataLength += d.length;
                });
                .on('error', function(e){
                    streamErr = e;
                })
                .on('close', function(){
                    if(streamErr) {
                        writeToErrorLog("downloadFile(): Error retrieving the file: " + streamErr);
                    } else {
                        writeToLog("downloadFile(): No error using read stream.");
                        m_fileBuffer = Buffer.concat(data, dataLength);
                        writeToLog("Data length: " + dataLength);

                        writeToLog("downloadFile(): File saved to buffer.");
                    }
                    conn.end();
                });
            })
        })
        .on('error', function(err) {
            writeToErrorLog("downloadFile(): Error connecting: " + err);
        }).connect(m_ssh2Credentials);

最佳答案

经过大量调查,我终于意识到在“数据”事件中传输的最后几位数据有问题。据我了解,这似乎是读取流实现中的错误。通过使用 SSH2 库中更简单的函数(open、fstat、read),我能够解决这个问题。这个解决方案对我有用。如果其他人遇到同样的问题,希望分享解决方案。

工作代码:

sftp.open(config.ftpPath + "/" + m_fileName, "r", function(err, fd) {
sftp.fstat(fd, function(err, stats) {
    var bufferSize = stats.size,
        chunkSize = 16384,
        buffer = new Buffer(bufferSize),
        bytesRead = 0,
        errorOccured = false;

    while (bytesRead < bufferSize && !errorOccured) {
        if ((bytesRead + chunkSize) > bufferSize) {
            chunkSize = (bufferSize - bytesRead);
        }
        sftp.read(fd, buffer, bytesRead, chunkSize, bytesRead, callbackFunc);
        bytesRead += chunkSize;
    }

    var totalBytesRead = 0;
    function callbackFunc(err, bytesRead, buf, pos) {
        if(err) {
            writeToErrorLog("downloadFile(): Error retrieving the file.");
            errorOccured = true;
            sftp.close(fd);
        }
        totalBytesRead += bytesRead;
        data.push(buf);
        if(totalBytesRead === bufferSize) {
            m_fileBuffer = Buffer.concat(data);
            writeToLog("downloadFile(): File saved to buffer.");
            sftp.close(fd);
            m_eventEmitter.emit(' downloadFile_Completed ');
        }
    }
})
});   

关于javascript - 使用 Node.js 和 SSH2 从 SFTP 服务器读取文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37375001/

相关文章:

javascript - 如何缓慢迭代 jquery 循环,并每次替换占位符文本

javascript - AngularJS 在 ng-repeat 元素中附加 HTML

node.js - 从另一个文件调用的 Sinon stub 方法

performance - 面试解决方案 - 修剪间隙缓冲区

ios - 在 MTLBuffer 的一部分上运行计算内核?

javascript - RxJS 如何返回带有默认错误处理函数的 Observable

JavaScript 无法从选择下拉列表中检索选项值

node.js - 使用 NodeJS (wo Angular) 进行 SEO 怎么样?

node.js - 在ubuntu或centos上安装node-expat

c++ - 控制准备从 QT 中的声卡读取的字节数