Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement minimal EXIF and IPTC tag support #40

Open
wants to merge 11 commits into
base: master
Choose a base branch
from

Conversation

sebastiaan-lampo
Copy link
Contributor

Updated version of PR #32.

Rebased on the updated version of HPS. I've added tests to cover the EXIF and IPTC additions.
Minor changes to the configuration tests are required as this PR introduces 3 potential new config file entries. These have been added to the config tests.

This feature allows:

  • Back filling the copyright or caption data from an EXIF or IPTC metadata field if not defined in the album.yml
  • Prepare for allowing access to all tags in the shortcode by making (a subset of) tags available as a property of the Photo object

Use PIL's TAGS to deal with EXIF. Use
iptcinfo3 to deal with IPTC tags. Allows to
use these as fallback options for copyright
and caption of photos.

Will not change the default behaviour without
specifying the setting tag_map, exif or iptc to
the configuration file.
Copy link
Owner

@GjjvdBurg GjjvdBurg left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for adding this @sebastiaan-lampo! Very cool addition. I have some small comments from looking through the code and testing it, but hopefully these should be easy to fix. Thanks again and sorry for the delay! :)

def free(self):
"""Manually clean up the cached image"""
if hasattr(self, "_original_img") and self._original_img:
self._original_img.close()
del self._original_img
self._original_img = None

@property
def iptc(self):
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What's the rationale for not having a _load_iptc method, similar to the _load_exif method?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good point & implemented.

The only downside is that it now always attempts to load IPTC. We always needed exif for orientation but we could lazy-load IPTC on first use. Right now it will simply load both.

The benefit of the code alignment may outweigh the performance penalty for users that don't use iptc data. I haven't tested the performance penalty yet.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure I understand why you can't lazy load IPTC anyway. In _get_tag_value, you can load IPTC when needed.

Instead of:

o = getattr(self, obj.lower(), {})

you could use:

if obj.lower() == 'exif':
  return self.exif.get(t)
return self.iptc.get(t)

Then, in def iptc(self), you can have:

if self._iptc is None:
  self._load_iptc()
return self._iptc

This means you won't have _load_iptc in the same place as _load_exif, but I think that's acceptable.

Let me know if you're interested in making this change. This has been open for a long time so if you're not, we could also just merge it and I can handle it instead.

hugophotoswipe/photo.py Outdated Show resolved Hide resolved
docs/README.md Outdated Show resolved Hide resolved
hugophotoswipe/photo.py Outdated Show resolved Hide resolved
return str(self._get_tag_value(settings.tag_map.get('copyright')))
return ""

def __getattribute__(self, attr):
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not a huge fan of overloading __getattribute__ like this, especially because you're catching an AttributeError and then looking in the settings map. What if in the future I unintentionally introduce a class variable that's also an exif tag name? I'm not entirely sure where you're using this, but perhaps a get_tag method can work instead? Happy to discuss this if you'd like.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The reason for overloading __getattribute__ is so that you can access those tags from within the hugo shortcode. That would allow you to expand the shortcode with, for example, {{ .Get "aperture"}}. I couldn't figure out another way to make that data accessible when processing the shortcode, except for explicitly defining each as a property. I'm definitely open to other ideas.

I do understand your objection though and as it isn't quite immediately related to the exif/iptc feature I'm removing this from this branch until we can find out a better solution.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry for the slow response!

This makes sense, Here's one solution I can think of: require a prefix when used in a shortcode: {{ .Get "exit:ApertureValue" }} or {{ .Get "iptc:ReleaseTime" }}. Then you can overload the __getattribute__ method by first checking for the exif or iptc prefixes, and retrieving them using _get_tag_value. If they don't start with either prefix, return getattr(self, attr).

hugophotoswipe/photo.py Outdated Show resolved Hide resolved
Also add test cases for improperly formatted tags.
Need a more robust solution to extract arbitrary attributes. For now, removing this from this branch.
iptcinfo spews many warnings about missing IPTC data, reading before image or decoding error. Most of the time this just seems to mean that the desired tags aren't present.
Note: Was not detected in test suite because all test images have exif['Orientation'] = None.
Note: Was not detected in test suite because all test images have exif['Orientation'] = None.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants