Stopped flattening EXIF IFD into getexif()

This commit is contained in:
Andrew Murray 2020-10-05 20:35:45 +11:00
parent 058b8d3d12
commit faf8fad76d
6 changed files with 106 additions and 94 deletions

View File

@ -663,43 +663,43 @@ class TestImage:
exif = im.getexif() exif = im.getexif()
assert 258 not in exif assert 258 not in exif
assert 274 in exif assert 274 in exif
assert 40960 in exif assert 282 in exif
assert exif[40963] == 450 assert exif[296] == 2
assert exif[11] == "gThumb 3.0.1" assert exif[11] == "gThumb 3.0.1"
out = str(tmp_path / "temp.jpg") out = str(tmp_path / "temp.jpg")
exif[258] = 8 exif[258] = 8
del exif[274] del exif[274]
del exif[40960] del exif[282]
exif[40963] = 455 exif[296] = 455
exif[11] = "Pillow test" exif[11] = "Pillow test"
im.save(out, exif=exif) im.save(out, exif=exif)
with Image.open(out) as reloaded: with Image.open(out) as reloaded:
reloaded_exif = reloaded.getexif() reloaded_exif = reloaded.getexif()
assert reloaded_exif[258] == 8 assert reloaded_exif[258] == 8
assert 274 not in reloaded_exif assert 274 not in reloaded_exif
assert 40960 not in reloaded_exif assert 282 not in reloaded_exif
assert reloaded_exif[40963] == 455 assert reloaded_exif[296] == 455
assert reloaded_exif[11] == "Pillow test" assert reloaded_exif[11] == "Pillow test"
with Image.open("Tests/images/no-dpi-in-exif.jpg") as im: # Big endian with Image.open("Tests/images/no-dpi-in-exif.jpg") as im: # Big endian
exif = im.getexif() exif = im.getexif()
assert 258 not in exif assert 258 not in exif
assert 40962 in exif assert 306 in exif
assert exif[40963] == 200 assert exif[274] == 1
assert exif[305] == "Adobe Photoshop CC 2017 (Macintosh)" assert exif[305] == "Adobe Photoshop CC 2017 (Macintosh)"
out = str(tmp_path / "temp.jpg") out = str(tmp_path / "temp.jpg")
exif[258] = 8 exif[258] = 8
del exif[34665] del exif[306]
exif[40963] = 455 exif[274] = 455
exif[305] = "Pillow test" exif[305] = "Pillow test"
im.save(out, exif=exif) im.save(out, exif=exif)
with Image.open(out) as reloaded: with Image.open(out) as reloaded:
reloaded_exif = reloaded.getexif() reloaded_exif = reloaded.getexif()
assert reloaded_exif[258] == 8 assert reloaded_exif[258] == 8
assert 34665 not in reloaded_exif assert 306 not in reloaded_exif
assert reloaded_exif[40963] == 455 assert reloaded_exif[274] == 455
assert reloaded_exif[305] == "Pillow test" assert reloaded_exif[305] == "Pillow test"
@skip_unless_feature("webp") @skip_unless_feature("webp")

View File

@ -3309,11 +3309,11 @@ class Exif(MutableMapping):
# returns a dict with any single item tuples/lists as individual values # returns a dict with any single item tuples/lists as individual values
return {k: self._fixup(v) for k, v in src_dict.items()} return {k: self._fixup(v) for k, v in src_dict.items()}
def _get_ifd_dict(self, tag): def _get_ifd_dict(self, offset):
try: try:
# an offset pointer to the location of the nested embedded IFD. # an offset pointer to the location of the nested embedded IFD.
# It should be a long, but may be corrupted. # It should be a long, but may be corrupted.
self.fp.seek(self[tag]) self.fp.seek(offset)
except (KeyError, TypeError): except (KeyError, TypeError):
pass pass
else: else:
@ -3351,11 +3351,16 @@ class Exif(MutableMapping):
self.fp.seek(self._info.next) self.fp.seek(self._info.next)
self._info.load(self.fp) self._info.load(self.fp)
def _get_merged_dict(self):
merged_dict = dict(self)
# get EXIF extension # get EXIF extension
ifd = self._get_ifd_dict(0x8769) if 0x8769 in self:
ifd = self._get_ifd_dict(self[0x8769])
if ifd: if ifd:
self._data.update(ifd) merged_dict.update(ifd)
self._ifds[0x8769] = ifd
return merged_dict
def tobytes(self, offset=8): def tobytes(self, offset=8):
from . import TiffImagePlugin from . import TiffImagePlugin
@ -3370,17 +3375,22 @@ class Exif(MutableMapping):
return b"Exif\x00\x00" + head + ifd.tobytes(offset) return b"Exif\x00\x00" + head + ifd.tobytes(offset)
def get_ifd(self, tag): def get_ifd(self, tag):
if tag not in self._ifds and tag in self: if tag not in self._ifds:
if tag in [0x8825, 0xA005]: if tag in [0x8769, 0x8825]:
# gpsinfo, interop # exif, gpsinfo
self._ifds[tag] = self._get_ifd_dict(tag) if tag in self:
elif tag == 0x927C: # makernote self._ifds[tag] = self._get_ifd_dict(self[tag])
elif tag in [0xA005, 0x927C]:
# interop, makernote
if 0x8769 not in self._ifds:
self.get_ifd(0x8769)
tag_data = self._ifds[0x8769][tag]
if tag == 0x927C:
from .TiffImagePlugin import ImageFileDirectory_v2 from .TiffImagePlugin import ImageFileDirectory_v2
if self[0x927C][:8] == b"FUJIFILM": if self._ifds[0x8769][tag][:8] == b"FUJIFILM":
exif_data = self[0x927C] ifd_offset = i32le(tag_data, 8)
ifd_offset = i32le(exif_data, 8) ifd_data = tag_data[ifd_offset:]
ifd_data = exif_data[ifd_offset:]
makernote = {} makernote = {}
for i in range(0, struct.unpack("<H", ifd_data[:2])[0]): for i in range(0, struct.unpack("<H", ifd_data[:2])[0]):
@ -3388,9 +3398,10 @@ class Exif(MutableMapping):
"<HHL4s", ifd_data[i * 12 + 2 : (i + 1) * 12 + 2] "<HHL4s", ifd_data[i * 12 + 2 : (i + 1) * 12 + 2]
) )
try: try:
unit_size, handler = ImageFileDirectory_v2._load_dispatch[ (
typ unit_size,
] handler,
) = ImageFileDirectory_v2._load_dispatch[typ]
except KeyError: except KeyError:
continue continue
size = count * unit_size size = count * unit_size
@ -3414,14 +3425,12 @@ class Exif(MutableMapping):
makernote[ifd_tag] = handler( makernote[ifd_tag] = handler(
ImageFileDirectory_v2(), data, False ImageFileDirectory_v2(), data, False
) )
self._ifds[0x927C] = dict(self._fixup_dict(makernote)) self._ifds[tag] = dict(self._fixup_dict(makernote))
elif self.get(0x010F) == "Nintendo": elif self.get(0x010F) == "Nintendo":
ifd_data = self[0x927C]
makernote = {} makernote = {}
for i in range(0, struct.unpack(">H", ifd_data[:2])[0]): for i in range(0, struct.unpack(">H", tag_data[:2])[0]):
ifd_tag, typ, count, data = struct.unpack( ifd_tag, typ, count, data = struct.unpack(
">HHL4s", ifd_data[i * 12 + 2 : (i + 1) * 12 + 2] ">HHL4s", tag_data[i * 12 + 2 : (i + 1) * 12 + 2]
) )
if ifd_tag == 0x1101: if ifd_tag == 0x1101:
# CameraInfo # CameraInfo
@ -3450,7 +3459,10 @@ class Exif(MutableMapping):
camerainfo["Category"] = self.fp.read(2) camerainfo["Category"] = self.fp.read(2)
makernote = {0x1101: dict(self._fixup_dict(camerainfo))} makernote = {0x1101: dict(self._fixup_dict(camerainfo))}
self._ifds[0x927C] = makernote self._ifds[tag] = makernote
else:
# gpsinfo, interop
self._ifds[tag] = self._get_ifd_dict(tag_data)
return self._ifds.get(tag, {}) return self._ifds.get(tag, {})
def __str__(self): def __str__(self):

View File

@ -478,7 +478,7 @@ class JpegImageFile(ImageFile.ImageFile):
def _getexif(self): def _getexif(self):
if "exif" not in self.info: if "exif" not in self.info:
return None return None
return dict(self.getexif()) return self.getexif()._get_merged_dict()
def _getmp(self): def _getmp(self):

View File

@ -82,7 +82,7 @@ class MpoImageFile(JpegImagePlugin.JpegImageFile):
n = i16(self.fp.read(2)) - 2 n = i16(self.fp.read(2)) - 2
self.info["exif"] = ImageFile._safe_read(self.fp, n) self.info["exif"] = ImageFile._safe_read(self.fp, n)
exif = self.getexif() exif = self.getexif().get_ifd(0x8769)
if 40962 in exif and 40963 in exif: if 40962 in exif and 40963 in exif:
self._size = (exif[40962], exif[40963]) self._size = (exif[40962], exif[40963])
elif "exif" in self.info: elif "exif" in self.info:

View File

@ -968,7 +968,7 @@ class PngImageFile(ImageFile.ImageFile):
self.load() self.load()
if "exif" not in self.info and "Raw profile type exif" not in self.info: if "exif" not in self.info and "Raw profile type exif" not in self.info:
return None return None
return dict(self.getexif()) return self.getexif()._get_merged_dict()
def getexif(self): def getexif(self):
if "exif" not in self.info: if "exif" not in self.info:

View File

@ -96,7 +96,7 @@ class WebPImageFile(ImageFile.ImageFile):
def _getexif(self): def _getexif(self):
if "exif" not in self.info: if "exif" not in self.info:
return None return None
return dict(self.getexif()) return self.getexif()._get_merged_dict()
def seek(self, frame): def seek(self, frame):
if not self._seek_check(frame): if not self._seek_check(frame):