在Nodejs中使用archiver-zip-encrypted库加密压缩文件出现报错如何解决

在Nodejs中使用archiver-zip-encrypted库加密压缩文件出现报错如何解决?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。

在windows系统上测试时,发现会概率的出现以下异常:

在Nodejs中使用archiver-zip-encrypted库加密压缩文件出现报错如何解决

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库加密压缩文件出现报错如何解决问题的解答就分享到这里了,希望以上内容可以对大家有一定的帮助,如果你还有很多疑惑没有解开,可以关注恰卡编程网行业资讯频道了解更多相关知识。

发布于 2021-03-24 01:21:44
收藏
分享
海报
0 条评论
173
上一篇:怎么在Python中使用turtle库绘制同心圆 下一篇:如何在Python中使用Pandas读写Excel文件
目录

    0 条评论

    本站已关闭游客评论,请登录或者注册后再评论吧~

    忘记密码?

    图形验证码