Python Tutorial: Cool EXIF Captions for Photographers with Pillow and Beem

As a photographer, I like to provide my readers with some EXIF-Information about my photos. Besides the informative value of EXIF photo captions, it also adds whitespace between the photos which I find much more pleasant to look at than the default way steemit.com displays photo galleries. However, especially when writing longer posts, nobody would want to add EXIF Captions for 20+ photos manually.


1/640s; 140mm; f/5.6; ISO 640

I didn't have a photo of a python, so here are some monkeys that I captured at Mt. Popa in Myanmar three days ago. Note the short exposure time that I needed because they were moving so fast and that you can now see in the cool EXIF caption - awesome, isn't it?

Repository

https://github.com/holgern/beem
https://github.com/python-pillow/Pillow

What Will I Learn?

  • How to read and format EXIF-Data with Pillow
  • How to upload photos to steemitimages using beem.imageuploader
  • How to create cool EXIF captions
  • How to add a watermark with Pillow

Requirements

  • Working Python installation
  • Beem Python library by @holger80 with a valid posting key added to the CLI wallet and the environment variable UNLOCK set to the CLI wallet passphrase
  • Pillow Python library
  • fractions library (comes with Python) for EXIF calculations

Difficulty

  • Intermediate

Tutorial

In this tutorial you will learn how to write a Python script that will read the EXIF-Data of all images in a folder, upload them to steemitimages and output a convenient HTML-snippet for each photo including the Exif caption.

Before
After

Reading EXIF with Pillow

Before we start, we need to define a default function for Pillow for reading the EXIF of a photo:

def get_exif (exif,field) :
  for (k,v) in exif.items():
     if TAGS.get(k) == field:
        return v

Now, we open the image and call our get_exif() functions with the parameters we want to know:

exif = img._getexif()
tmp_exp = get_exif(exif,'ExposureTime')
tmp_focal = get_exif(exif,'FocalLength')
tmp_fstop = get_exif(exif,'FNumber')
ISO = "ISO "+str(get_exif(exif,'ISOSpeedRatings'))

Formatting the EXIF outputs

You may have wondered why we called the variables tmp_. Let's quickly print out the results with print(tmp_exp, tmp_focal, tmp_fstop, ISO) to see what is wrong with them:

(1, 50) (280, 10) (18, 10) ISO 560

This is not what we want to get! The numbers are correct though, we just have to format them correctly. To prevent empty values from crashing the script, we use proper exception handling that returns the default value "None" if the EXIF field in question is empty or could not be processed:

try:
    ExposureTime = str(Fraction(tmp_exp[0]/tmp_exp[1]).limit_denominator())+"s"
except:
    ExposureTime = "None"
try:
    FocalLength = str(int(tmp_focal[0]/tmp_focal[1]))+"mm"
except: 
    FocalLength = "None"
try: 
    FNumber = "f/"+str(tmp_fstop[0]/tmp_fstop[1])
except:
    FNumber = "None"

Let's now create a string out of all the values. If the value of any of the EXIF data is "None", we return an empty string.

settings = "" if ExposureTime == "None" or FocalLength == "None" or FNumber == "None" or ISO == "None" else ExposureTime+"; "+FocalLength+"; "+FNumber+"; "+ISO

With our example values from above, we now get the following output:

1/50s; 28mm; f/1.8; ISO 560

That looks much better!

Uploading photos with beem.imageuploader

Uploading the photos to steemitimages is very easy thanks to beem:

steem = Steem()
iu = ImageUploader(steem_instance=steem)
steem.wallet.unlock(walletpw)
postimage = iu.upload(path+"/"+imgurl, postaccount)

Now, we only need one line to print out a HTML snippet for each image containing the image that we just uploaded to steemitimages and its EXIF data that we previously saved into the string settings as a caption.

print("

"+settings+"")

Finishing the function get_image()

Now, let's wrap this code in the function process_image():

def process_image(imgurl):
    try:
        """ 
        Reading the Exif-Data with Pillow
        """
        img = Image.open(path+"/"+imgurl)
        exif = img._getexif()
        tmp_exp = get_exif(exif,'ExposureTime')
        tmp_focal = get_exif(exif,'FocalLength')
        tmp_fstop = get_exif(exif,'FNumber')
        ISO = "ISO "+str(get_exif(exif,'ISOSpeedRatings'))
        """
        Formatting the EXIF-Data
        """
        try:
                ExposureTime = str(Fraction(tmp_exp[0]/tmp_exp[1]).limit_denominator())+"s"
        except:
                ExposureTime = "None"
        try:
                FocalLength = str(int(tmp_focal[0]/tmp_focal[1]))+"mm"
        except: 
                FocalLength = "None"
        try: 
                FNumber = "f/"+str(tmp_fstop[0]/tmp_fstop[1])
        except:
                FNumber = "None"
        settings = "" if ExposureTime == "None" or FocalLength == "None" or FNumber == "None" or ISO == "None" else ExposureTime+"; "+FocalLength+"; "+FNumber+"; "+ISO
        """
        Uploading to steemitimages using beem
        """
        steem = Steem()
        iu = ImageUploader(steem_instance=steem)
        steem.wallet.unlock(walletpw)
        postimage = iu.upload(path+"/"+imgurl, postaccount)
        """ 
        Printing out the result
        """
        print("

"+settings+"") except Exception as error: print("Error during image processing "+repr(error))

Queuing Images

Finally, let's write the main function. It calls our function process_image() for each file in our folder path with the filename as parameter. We could also add a filter to only queue files with image filetypes, but this would go too far for this tutorial.

if __name__ == '__main__':
    for image in os.listdir(path):
        process_image(image)
    print("Finished")

Remember to also set your own values on the top of the script for path (path to your folder that contains the image files) and postaccount (your steem account previously added to the beem wallet). Remember to set the environment variable UNLOCK or alternatively define walletpw as your beem CLI wallet password.

path = '00steemit'
postaccount = 'yoursteemaccount'
walletpw = os.environ.get('UNLOCK')

Now we can run the script and get an output similar to this:

<center><img src=' />
1/
50s; 28mm; f/1.8; ISO 560</sup>center> <center><img src=' />
1/
30s; 20mm; f/1.8; ISO 5000</sup>center> <center><img src=' />
1/
500s; 19mm; f/1.8; ISO 900</sup>center> Finished

Bonus: More fun with Pillow

Pillow can do much more than just reading EXIF-Data, it is also capable of editing images.
I use it to add a watermark to my photos: Instead of simply uploading the photo with postimage = iu.upload(path+"/"+imgurl, postaccount), we make a watermarked copy of the photo in a second folder:

watermarkedpath = '00steemit'
watermarkpath = 'watermark.png'
watermark = Image.open(watermarkpath)
destination = img.copy()
position = ((destination.width - watermark.width - 32), (destination.height - watermark.height - 30))
destination.paste(watermark, position, watermark)
waterimgurl = watermarkedpath+'/'+imgurl
destination.save(waterimgurl)
postimage = iu.upload(waterimgurl, postaccount)

Of course, you could also read more EXIF Data such as the lens used with Pillow, add the title as an ALT-Attribute to the image for SEO optimation or link each image to its steemitimages URL for a fullscreen preview.

I run the script from a folder containing the script itself, the folder 00steemit where I put my photo files, the optional watermark file watermark.png and the optional folder for watermarked images 01steemit. I run the script with python exifcaptions.py > output.txt which saves the HTML snippets generated into the file output.txt which contents I then copy into my steemit post.

Enjoy the cool EXIF captions!

Proof of Work Done

You can find the complete code on Github:
https://github.com/tiotdev/steem-tutorials/blob/master/cool-exif-captions/exifcaptions.py


I would love to see this functionality implemented in some Steem frontends such as @steempeak or @busy.org!

H2
H3
H4
3 columns
2 columns
1 column
10 Comments