web-dev-qa-db-ja.com

ES8 async / awaitをストリームで使用する方法は?

https://stackoverflow.com/a/18658613/779159 は、組み込みの暗号化ライブラリとストリームを使用してファイルのmd5を計算する方法の例です。

var fs = require('fs');
var crypto = require('crypto');

// the file you want to get the hash    
var fd = fs.createReadStream('/some/file/name.txt');
var hash = crypto.createHash('sha1');
hash.setEncoding('hex');

fd.on('end', function() {
    hash.end();
    console.log(hash.read()); // the desired sha1sum
});

// read all file and pipe it (write it) to the hash object
fd.pipe(hash);

しかし、上記のようにコールバックを使用する代わりにES8 async/awaitを使用するようにこれを変換することは可能ですが、それでもストリームを使用する効率は維持されますか?

38
user779159

async/awaitはプロミスでのみ機能し、ストリームでは機能しません。独自の構文を取得する追加のストリームのようなデータ型を作成するためのアイデアがありますが、それらは非常に実験的であり、詳細については説明しません。

とにかく、コールバックはストリームの終わりを待っているだけです。これは約束にぴったりです。ストリームをラップするだけです。

var fd = fs.createReadStream('/some/file/name.txt');
var hash = crypto.createHash('sha1');
hash.setEncoding('hex');
// read all file and pipe it (write it) to the hash object
fd.pipe(hash);

var end = new Promise(function(resolve, reject) {
    hash.on('end', () => resolve(hash.read()));
    fd.on('error', reject); // or something like that. might need to close `hash`
});

今、あなたはその約束を待つことができます:

(async function() {
    let sha1sum = await end;
    console.log(sha1sum);
}());
70
Bergi

ノードバージョン> = v10.0.0を使用している場合は、 stream.pipeline および til.promisify を使用できます。

const fs = require('fs');
const crypto = require('crypto');
const util = require('util');
const stream = require('stream');

const pipeline = util.promisify(stream.pipeline);

const hash = crypto.createHash('sha1');
hash.setEncoding('hex');

async function run() {
  await pipeline(
    fs.createReadStream('/some/file/name.txt'),
    hash
  );
  console.log('Pipeline succeeded');
}

run().catch(console.error);
13
shusson

このような何かが動作します:

for (var res of fetchResponses){ //node-fetch package responses
    const dest = fs.createWriteStream(filePath,{flags:'a'});
    totalBytes += Number(res.headers.get('content-length'));
    await new Promise((resolve, reject) => {
        res.body.pipe(dest);
        res.body.on("error", (err) => {
            reject(err);
        });
        dest.on("finish", function() {
            resolve();
        });
    });         
}
3
Ron Royston