在Nodejs中使用archiver-zip-encrypted库加密压缩文件出现报错如何解决
在Nodejs中使用archiver-zip-encrypted库加密压缩文件出现报错如何解决?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。
在windows系统上测试时,发现会概率的出现以下异常:
events.js:174 thrower;//Unhandled'error'event ^ Error:filedatastreamhasunexpectednumberofbytes atByteCounter.( xxx\node_modules\yazl\index.js:162:99) atByteCounter.emit(events.js:194:15) atendReadableNT(_stream_readable.js:1103:12) atprocess._tickCallback(internal/process/next_tick.js:63:19) Emitted'error'eventat: atByteCounter.(xxx\node_modules\yazl\index.js:162:85) atByteCounter.emit(events.js:194:15) atendReadableNT(_stream_readable.js:1103:12) atprocess._tickCallback(internal/process/next_t
我的本机环境是:
npm:6.9.0node: v10.16.3
在另外一个同事的windows系统上测试,他那边是上面异常必现,对应的node版本是v10.15。
具体使用的代码不贴了,基本上是参照官方demo 来写的,压缩完成最后调用代码如下所示:
archive.finalize().then(()=>{ //到这里认为是压缩完成,进行后续处理,实际并没有,参照后面分析 anotherProcess(); }).catch(err=>{ //压缩出现异常处理... });
出现异常后一一检查代码和官方demo不一样的地方,并没有发现什么异常之处,网上搜索也没有发现这种异常记录。由于刚接触JS,不是很熟,就从问题开始下手,找到出现问题的代码,开始调试。
错误日志中提示是在yzal/index.js 文件中发生异常,找到出现异常的代码如下所示:
functionpumpFileDataReadStream(self,entry,readStream){ varcrc32Watcher=newCrc32Watcher(); varuncompressedSizeCounter=newByteCounter(); varcompressor=entry.compress?newzlib.DeflateRaw():newPassThrough(); varcompressedSizeCounter=newByteCounter(); readStream.pipe(crc32Watcher) .pipe(uncompressedSizeCounter) .pipe(compressor) .pipe(compressedSizeCounter) .pipe(self.outputStream,{end:false}); compressedSizeCounter.on("end",function(){ entry.crc32=crc32Watcher.crc32; if(entry.uncompressedSize==null){ entry.uncompressedSize=uncompressedSizeCounter.byteCount; }else{ //异常从这里抛出来的 if(entry.uncompressedSize!==uncompressedSizeCounter.byteCount)returnself.emit("error",newError("filedatastreamhasunexpectednumberofbytes")); } entry.compressedSize=compressedSizeCounter.byteCount; self.outputStreamCursor+=entry.compressedSize; writeToOutputStream(self,entry.getDataDescriptor()); entry.state=Entry.FILE_DATA_DONE; pumpEntries(self); }); }
从上面代码可以看出来:uncompressedSizeCounter.byteCount
是从pumpFileDataReadStream()
函数readStream 参数中获取的属性值,而entry.uncompressedSize
也是函数的entry 参数中获取的属性。接着找pumpFileDataReadStream()
函数是从哪里调用的。
通过输出日志得出pumpFileDataReadStream()
函数是在以下面的代码中被调用的:
ZipFile.prototype.addFile=function(realPath,metadataPath,options){ varself=this; metadataPath=validateMetadataPath(metadataPath,false); if(options==null)options={}; varentry=newEntry(metadataPath,false,options); self.entries.push(entry); fs.stat(realPath,function(err,stats){ if(err)returnself.emit("error",err); if(!stats.isFile())returnself.emit("error",newError("notafile:"+realPath)); //这里是文件的大小 entry.uncompressedSize=stats.size; if(options.mtime==null)entry.setLastModDate(stats.mtime); if(options.mode==null)entry.setFileAttributesMode(stats.mode); entry.setFileDataPumpFunction(function(){ //readStream在这里创建的 varreadStream=fs.createReadStream(realPath); entry.state=Entry.FILE_DATA_IN_PROGRESS; readStream.on("error",function(err){ self.emit("error",err); }); //在这里被调用 pumpFileDataReadStream(self,entry,readStream); }); pumpEntries(self); }); };
从上面代码可以看出来entry.uncompressedSize
是stats.size,即文件的大小,readStream 是创建的文件流。但是在什么情况下两者会不一样呢?感觉只可能在文件还没有读取完,但是是什么原因导致这种情况发生?由于对JS接触的时间不长,没有进行深入分析。最后在抛出异常的上面一行用console.log 将两个属性的大小值都输出,代码如下所示:
if(entry.uncompressedSize==null){ entry.uncompressedSize=uncompressedSizeCounter.byteCount; }else{ //增加日志输出 console.log("entrysize:"+entry.uncompressedSize+",uncompressedSize:"+uncompressedSizeCounter.byteCount); if(entry.uncompressedSize!==uncompressedSizeCounter.byteCount)returnself.emit("error",newError("filedatastreamhasunexpectednumberofbytes")); }
在archive.finalize()
时和output 写入流的close 事件时(详细参照官方的示例代码),分别加上日志输出,代码如下所示:
archive.finalize().then(()=>{ //到这里认为是压缩完成,进行后续处理,实际并没有,参照后面分析 console.log("finalize"); //anotherProcess(); }).catch(err=>{ //压缩出现异常 }); output.on('close',function(){ console.log('close'); //这个业务函数与上面finalize函数中的是互斥,不会同时存在 anotherProcess(); });
最后分别将anotherProcess()
函数加到两个异步回调中执行,发现在close 事件执行时,两个size输出的大小一致,都是文件的大小。而在finalize 场景测试发现uncompressedSize
要小于文件的大小。最后将anotherProcess()
函数放在close 事件回调函数中执行,问题解决。
关于在Nodejs中使用archiver-zip-encrypted库加密压缩文件出现报错如何解决问题的解答就分享到这里了,希望以上内容可以对大家有一定的帮助,如果你还有很多疑惑没有解开,可以关注恰卡编程网行业资讯频道了解更多相关知识。
推荐阅读
-
NodeJS怎么实现单点登录
-
怎么用Vue+NodeJS实现大文件上传
-
node中的buffer有什么用
node中的buffer有什么用本文小编为大家详细介绍“node中...
-
node中multer的概念是什么
node中multer的概念是什么这篇文章主要介绍“node中mu...
-
nodejs怎么读取文件夹目录的内容
nodejs怎么读取文件夹目录的内容小编给大家分享一下nodejs...
-
nodejs如何结合socket.io实现websocket通信功能
这篇文章主要介绍nodejs如何结合socket.io实现websocket通信功能,文中介绍的非常详细,具有一定的参考价值,感兴...
-
nodejs如何结合Socket.IO实现的即时通讯功能
-
nodejs如何实现TCP服务器端和客户端聊天功能
这篇文章主要介绍了nodejs如何实现TCP服务器端和客户端聊天功能,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇...
-
Nodejs能够应用于什么场景
这篇文章给大家分享的是有关Nodejs能够应用于什么场景的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。...
-
Nodejs +Websocket如何实现指定发送及群聊