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