node.js – Meteor:如何将大文件流式传输并解析为异步节点功能?

前端之家收集整理的这篇文章主要介绍了node.js – Meteor:如何将大文件流式传输并解析为异步节点功能?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我正在使用 job-collection软件包执行以下操作:

>下载包含大量有关网页元数据的大型文件
>使用NPM事件流包从正则表达式拆分的文件元数据中创建流
>检查集合中的元数据是否匹配(我一直试图将每个网页的元数据流式传输到另一个函数来执行此操作)

文件太大而无法缓冲,因此需要流式传输. Here is a small file with a few examples of the metadata 如果你想试试这个.

作业集合包中的每个作业都已在异步函数中:

  1. var request = Npm.require('request');
  2. var zlib = Npm.require('zlib');
  3. var EventStream = Meteor.npmRequire('event-stream');
  4.  
  5. function (job,callback) {
  6.  
  7. //This download is much too long to block
  8. request({url: job.fileURL,encoding: null},function (error,response,body) {
  9. if (error) console.error('Error downloading File');
  10. if (response.statusCode !== 200) console.error(downloadResponse.statusCode,'Status not 200');
  11.  
  12. var responseEncoding = response.headers['content-type'];
  13. console.log('response encoding is %s',responseEncoding);
  14. if (responseEncoding === 'application/octet-stream' || 'binary/octet-stream') {
  15. console.log('Received binary/octet-stream');
  16. var regexSplit = /WARC\/1\./;
  17. response.pipe(zlib.createGunzip()
  18. .pipe(EventStream.split(regexSplit))
  19. .pipe(EventStream.map(function (webpageMetaData) {
  20. /* Need parse the MetaData or pass each webpageMetaData to function
  21. * This next function could block if it had to */
  22. searchPageMetaData(webpageMetaData); // pass each Metadatum to this function to update a collection - this function can be synchronous
  23. }));
  24. } else {
  25. console.error('Wrong encoding');
  26. }
  27. });
  28. }
  29.  
  30. function searchWebPageMetaData(MetaData) {
  31. // Parse JSON and search collection for match
  32. }

>有没有更好的方法来构建它?我是在正确的轨道上吗?
>在哪里放置Meteor.bindEnvironment? – 每次传递给searchWebPageMetaData()时,我都会绑定环境吗?我需要在这里明确使用纤维吗?
>如果我将其运行到process.stdout,则在运行此流时会停止.我应该将流放入Meteor的一个包裹中
>我知道Meteor.wrapAsync.我想在Meteor.wrapAsync中包装最里面的searchWebPageMetaData()函数吗? (当我输入时,想想我回答的是)
>流是否会缓慢以补偿数据库调用的缓慢?我的猜测不是,但我该怎么处理?

我花了很长时间了解Meteor的wrapAsync和bindEnvironment,但是无法将它们整合在一起并了解在哪里使用它们.

补充1

只是为了澄清,步骤是:

>下载文件;
>创建流;
>解压缩;
>将其拆分为单独的webPages – EventStream处理此问题
>将其发送给函数 – 不需要返回值;这可能是阻塞,只是一些搜索数据库调用

我试图做这样的事情,除了我需要帮助的核心代码是在一个不同文件函数中.以下代码中包含@ electric-jesus的大部分答案.

  1. processJobs('parseWatFile',{
  2. concurrency: 1,cargo: 1,pollInterval: 1000,prefetch: 1
  3. },function (job,callback) {
  4.  
  5. if (job.data.watZipFileLink) {
  6. queue.pause();
  7. console.log('queue should be paused now');
  8.  
  9.  
  10. var watFileUrl = 'https://s3.amazonaws.com/ja-common-crawl/exampleWatFile.wat.gz';
  11. function searchPageMetaData(webpageMetaData,callback) {
  12. console.log(webpageMetaData); // Would be nice to just get this function logging each webPageMetaData
  13. future.return(callback(webpageMetaData)); //I don't need this to return any value - do I have to return something?
  14. }
  15.  
  16. if (!watFile)
  17. console.error('No watFile passed to downloadAndSearchWatFileForEntity ');
  18.  
  19. var future = new Future(); // Doc Brown would be proud.
  20.  
  21. if(typeof callback !== 'function') future.throw('callbacks are supposed to be functions.');
  22.  
  23. request({url: watFile,body) {
  24.  
  25. if (error) future.throw('Error Downloading File');
  26. if (response.statusCode !== 200) future.throw('Expected status 200,got ' + response.statusCode + '.');
  27.  
  28. var responseEncoding = response.headers['content-type'];
  29.  
  30. if (responseEncoding === 'application/octet-stream' || 'binary/octet-stream') {
  31.  
  32. var regexSplit = /WARC\/1\./;
  33. response.pipe(zlib.createGunzip()
  34. .pipe(EventStream.split(regexSplit))
  35. .pipe(EventStream.map(function (webpageMetaData) {
  36. searchPageMetaData(webpageMetaData,callback);
  37. })
  38. ));
  39. } else {
  40. future.throw('Wrong encoding');
  41. }
  42. });
  43.  
  44. return future.wait();
  45.  
  46. } else {
  47. console.log('No watZipFileLink for this job');
  48. job.log('ERROR: NO watZipFileLink from commonCrawlJob collection');
  49. }
  50. queue.resume();
  51. job.done;
  52. callback();
  53. }

解决方法

有意思,看起来还不错.我从未使用过工作集,但它似乎只是一个Mongo驱动的任务队列..所以我假设它像常规队列一样工作.我总是找到回调的东西,我当然使用Future模式.例如:
  1. var request = Npm.require('request');
  2. var zlib = Npm.require('zlib');
  3. var EventStream = Meteor.npmRequire('event-stream');
  4.  
  5. var Future = Npm.require('fibers/future');
  6.  
  7.  
  8. var searchWebPageMetaData = function (MetaData) {
  9. // Parse JSON and search collection for match
  10. // make it return something
  11. var result = /droids/ig.test(MetaData);
  12. return result;
  13. }
  14.  
  15. var processJob = function (job,callback) {
  16.  
  17. var future = new Future(); // Doc Brown would be proud.
  18.  
  19. if(typeof callback !== 'function') future.throw("Oops,you forgot that callbacks are supposed to be functions.. not undefined or whatever.");
  20.  
  21. //This download is much too long to block
  22. request({url: job.fileURL,body) {
  23.  
  24. if (error) future.throw("Error Downloading File");
  25. if (response.statusCode !== 200) future.throw("Expected status 200,got " + downloadResponse.statusCode + ".");
  26.  
  27. var responseEncoding = response.headers['content-type'];
  28.  
  29.  
  30. if (responseEncoding === 'application/octet-stream' || 'binary/octet-stream') {
  31.  
  32. var regexSplit = /WARC\/1\./;
  33. response.pipe(zlib.createGunzip()
  34. .pipe(EventStream.split(regexSplit))
  35. .pipe(EventStream.map(function (webpageMetaData) {
  36. /* Need parse the MetaData or pass each webpageMetaData to function
  37. * This next function could block if it had to */
  38.  
  39. // pass each Metadatum to this function to update a collection - this function can be synchronous
  40.  
  41. future.return(callback(webpageMetaData)); // this way,processJob returns whatever we find in the completed webpage,via callback.
  42.  
  43. }));
  44. } else {
  45. future.throw('Wrong encoding');
  46. }
  47. });
  48.  
  49. return future.wait();
  50. }

用法示例:

所以每当你在这里分配变量:

  1. var currentJob = processJob(myjob,searchWebPageMetaData);

即使使用同步类型获取/变量分配,您也可以及时完成异步工作并及时传输.

要回答你的问题,

>在哪里放置Meteor.bindEnvironment? – 每次传递给searchWebPageMetaData()时,我都会绑定环境吗?我需要在这里明确使用纤维吗?

不是真的,我相信明确使用纤维/未来已经解决了这个问题.
>如果我将其运行到process.stdout,则在运行此流时会停止.我应该将流放入Meteor的一个包裹中

你是什​​么意思?我依稀记得process.stdout是阻塞的,这可能是一个原因.再次,将结果包装在未来应该照顾这一点.
>我知道Meteor.wrapAsync.我想在Meteor.wrapAsync中包装最里面的searchWebPageMetaData()函数吗? (当我输入时,想想我回答的是)

Take a look at the Meteor.wrapAsync helper code.它基本上是应用的未来分辨率,当然你可以再做一遍,你也可以自己明确地使用光纤/未来没有问题.
>流是否会缓慢以补偿数据库调用的缓慢?我的猜测不是,但我该怎么处理?

不太确定你在这里的意思..但由于我们正在尝试使用异步光纤,我的猜测也不是很好.我还没有看到使用纤维有任何缓慢.可能只有在同时启动(并同时运行)多个作业的情况下,您才会在内存使用方面遇到性能问题.保持并发队列低,因为Fibers可以在运行内容时非常强大.你只有一个核心来处理它,这是一个可悲的事实,因为节点不能多核:(

猜你在找的Node.js相关文章