js检查图像是否被截断/损坏数据

时间:2017-01-04 10:55:08

标签: javascript image

我正在寻找如何检测图像数据被截断\损坏。例如这张图片: enter image description here

数据图像不完整(它在IE上更有形,并且在firefox控制台中显示为警告),但img.onerror未触发,img.completed为真。

演示:https://jsfiddle.net/7dd0ybb4/

var img = document.getElementById('MyPicture');

img.onerror = () => alert('error img');  
img.onload = () =>  console.log(img.complete); //true

img.src = "https://i.stack.imgur.com/nGkok.jpg";

我想知道这一点。如果图像有无效数据。

3 个答案:

答案 0 :(得分:4)

您可以通过画布检查损坏的像素,如此代码



var ctx = document.getElementById('canvas').getContext('2d');
var img = new Image();
img.onload = function(){
    ctx.drawImage(img,0,0);
    var height=this.height;
    var width=this.width;
    var hCenter=Math.round(width/2);
    var corruptedRowsCount=0;
    for(var row=height;row>0;row--){
      var c = ctx.getImageData(hCenter, row, 1, 1).data;
      if(c[0]==0 && c[1]==0 && c[2]==0 && c[3]==0)
          corruptedRowsCount++;
      else 
          break;
    }
    console.log(Math.round(corruptedRowsCount*100/height)+" precent of image is corrupted");
};

img.src="";

<canvas id="canvas">
&#13;
&#13;
&#13;

答案 1 :(得分:4)

您可以将图像解码为字节数组:

var src = 'here comes your base64 data'
var imageData = Uint8Array.from(atob(src.replace('data:image/jpeg;base64,', '')), c => c.charCodeAt(0))

JPEG必须start with the bytes FF D8 and end with FF D9,因此我们检查创建的数组缓冲区的最后两个元素是否为255和217.请参阅live example

var imageCorrupted = ((imageData[imageData.length - 1] === 217) && (imageData[imageData.length - 2] === 255));

我们可以对PNG(live example)使用类似的检查,end with an IEND chunk包含这个字节序列:

// in hex: 00 00 00 00 49 45 4e 44 ae 42 60 82
var sequence = [0, 0, 0, 0, 73, 69, 78, 68, 174, 66, 96, 130];

答案 2 :(得分:1)

对于png文件,我决定我想要一个解决方案,该解决方案不仅仅检查文件的开头和结尾字节以查看它们是否有效。我想要一个解决方案,该解决方案实际上利用png文件内每个块的嵌入式crc代码来检查文件内的所有数据是否损坏。

幸运的是,我找到了一个成熟的库来执行此操作: https://github.com/lukeapage/pngjs

不幸的是,它很大。浏览器版本几乎为500 kb。这是一个功能齐全的png文件操作库,这使它易于理解。如果您有一个需要所有这些功能的用例,那么您可以为大小付费。但是对于仅想验证png文件的简单用例来说,这太繁琐了。

因此我从中切出了验证代码,并留下了一个不错的2.5 kb npm软件包: https://www.npmjs.com/package/png-validator