exif.js 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. /*jslint browser: true, devel: true, bitwise: false, debug: true, eqeq: false, es5: true, evil: false, forin: false, newcap: false, nomen: true, plusplus: true, regexp: false, unparam: false, sloppy: true, stupid: false, sub: false, todo: true, vars: true, white: true */
  2. function readExifValue(format, stream) {
  3. switch(format) {
  4. case 1: return stream.nextUInt8();
  5. case 3: return stream.nextUInt16();
  6. case 4: return stream.nextUInt32();
  7. case 5: return [stream.nextUInt32(), stream.nextUInt32()];
  8. case 6: return stream.nextInt8();
  9. case 8: return stream.nextUInt16();
  10. case 9: return stream.nextUInt32();
  11. case 10: return [stream.nextInt32(), stream.nextInt32()];
  12. case 11: return stream.nextFloat();
  13. case 12: return stream.nextDouble();
  14. default: throw new Error('Invalid format while decoding: ' + format);
  15. }
  16. }
  17. function getBytesPerComponent(format) {
  18. switch(format) {
  19. case 1:
  20. case 2:
  21. case 6:
  22. case 7:
  23. return 1;
  24. case 3:
  25. case 8:
  26. return 2;
  27. case 4:
  28. case 9:
  29. case 11:
  30. return 4;
  31. case 5:
  32. case 10:
  33. case 12:
  34. return 8;
  35. default:
  36. return 0;
  37. }
  38. }
  39. function readExifTag(tiffMarker, stream) {
  40. var tagType = stream.nextUInt16(),
  41. format = stream.nextUInt16(),
  42. bytesPerComponent = getBytesPerComponent(format),
  43. components = stream.nextUInt32(),
  44. valueBytes = bytesPerComponent * components,
  45. values,
  46. value,
  47. c;
  48. /* if the value is bigger then 4 bytes, the value is in the data section of the IFD
  49. and the value present in the tag is the offset starting from the tiff header. So we replace the stream
  50. with a stream that is located at the given offset in the data section. s*/
  51. if(valueBytes > 4) {
  52. stream = tiffMarker.openWithOffset(stream.nextUInt32());
  53. }
  54. //we don't want to read strings as arrays
  55. if(format === 2) {
  56. values = stream.nextString(components);
  57. //cut off \0 characters
  58. var lastNull = values.indexOf('\0');
  59. if(lastNull !== -1) {
  60. values = values.substr(0, lastNull);
  61. }
  62. }
  63. else if(format === 7) {
  64. values = stream.nextBuffer(components);
  65. }
  66. else if(format !== 0) {
  67. values = [];
  68. for(c = 0; c < components; ++c) {
  69. values.push(readExifValue(format, stream));
  70. }
  71. }
  72. //since our stream is a stateful object, we need to skip remaining bytes
  73. //so our offset stays correct
  74. if(valueBytes < 4) {
  75. stream.skip(4 - valueBytes);
  76. }
  77. return [tagType, values, format];
  78. }
  79. function readIFDSection(tiffMarker, stream, iterator) {
  80. var numberOfEntries = stream.nextUInt16(), tag, i;
  81. for(i = 0; i < numberOfEntries; ++i) {
  82. tag = readExifTag(tiffMarker, stream);
  83. iterator(tag[0], tag[1], tag[2]);
  84. }
  85. }
  86. function readHeader(stream) {
  87. var exifHeader = stream.nextString(6);
  88. if(exifHeader !== 'Exif\0\0') {
  89. throw new Error('Invalid EXIF header');
  90. }
  91. var tiffMarker = stream.mark();
  92. var tiffHeader = stream.nextUInt16();
  93. if(tiffHeader === 0x4949) {
  94. stream.setBigEndian(false);
  95. } else if(tiffHeader === 0x4D4D) {
  96. stream.setBigEndian(true);
  97. } else {
  98. throw new Error('Invalid TIFF header');
  99. }
  100. if(stream.nextUInt16() !== 0x002A) {
  101. throw new Error('Invalid TIFF data');
  102. }
  103. return tiffMarker;
  104. }
  105. module.exports = {
  106. IFD0: 1,
  107. IFD1: 2,
  108. GPSIFD: 3,
  109. SubIFD: 4,
  110. InteropIFD: 5,
  111. parseTags: function(stream, iterator) {
  112. var tiffMarker;
  113. try {
  114. tiffMarker = readHeader(stream);
  115. } catch(e) {
  116. return false; //ignore APP1 sections with invalid headers
  117. }
  118. var subIfdOffset, gpsOffset, interopOffset;
  119. var ifd0Stream = tiffMarker.openWithOffset(stream.nextUInt32()),
  120. IFD0 = this.IFD0;
  121. readIFDSection(tiffMarker, ifd0Stream, function(tagType, value, format) {
  122. switch(tagType) {
  123. case 0x8825: gpsOffset = value[0]; break;
  124. case 0x8769: subIfdOffset = value[0]; break;
  125. default: iterator(IFD0, tagType, value, format); break;
  126. }
  127. });
  128. var ifd1Offset = ifd0Stream.nextUInt32();
  129. if(ifd1Offset !== 0) {
  130. var ifd1Stream = tiffMarker.openWithOffset(ifd1Offset);
  131. readIFDSection(tiffMarker, ifd1Stream, iterator.bind(null, this.IFD1));
  132. }
  133. if(gpsOffset) {
  134. var gpsStream = tiffMarker.openWithOffset(gpsOffset);
  135. readIFDSection(tiffMarker, gpsStream, iterator.bind(null, this.GPSIFD));
  136. }
  137. if(subIfdOffset) {
  138. var subIfdStream = tiffMarker.openWithOffset(subIfdOffset), InteropIFD = this.InteropIFD;
  139. readIFDSection(tiffMarker, subIfdStream, function(tagType, value, format) {
  140. if(tagType === 0xA005) {
  141. interopOffset = value[0];
  142. } else {
  143. iterator(InteropIFD, tagType, value, format);
  144. }
  145. });
  146. }
  147. if(interopOffset) {
  148. var interopStream = tiffMarker.openWithOffset(interopOffset);
  149. readIFDSection(tiffMarker, interopStream, iterator.bind(null, this.InteropIFD));
  150. }
  151. return true;
  152. }
  153. };