使用 Python 添加 "Date Taken" Exif/XMP 信息到 TIF 文件

Add "Date Taken" Exif/XMP information to TIF file using Python

在 Windows10 上使用以下 Python 代码,我正在尝试编辑使用 Nikon Scan 4.0.3(软件来自2008 年左右)。

import piexif

def setImageDateTakenAttribute(filename, date_taken):
exif_dict = piexif.load(filename)
exif_dict['Exif'] = { 
    piexif.ExifIFD.DateTimeOriginal: datetime.datetime(*date_taken[:6]).strftime("%Y:%m:%d %H:%M:%S") 
} 
exif_bytes = piexif.dump(exif_dict)
piexif.insert(exif_bytes, filename)

对于 jpeg 文件,这在 Python 3 中工作正常,但是当我尝试使用相同的代码编辑我的 tif 文件时,出现以下错误:

    setImageDateTakenAttribute(full_path, folder_date)
  File "image-and-movie-bulk-creation-dates.py", line 78, in setImageDateTakenAttribute
    piexif.insert(exif_bytes, filename)
  File "C:\Python37\lib\site-packages\piexif\_insert.py", line 39, in insert
    raise InvalidImageDataError
piexif._exceptions.InvalidImageDataError

到目前为止,我一直在寻找可能支持我的文件但无济于事的其他软件包。

有谁知道如何在不删除现有 exif 信息、不更改图像格式的情况下在 Python 中完成编辑?

要复制或获取失败的 tif 文件的样本,请克隆我的项目(link 如下)。

详情:

将数千张图片扫描成 tif 文件后,我想为“拍摄日期”指定 EXIF 值。我正在编写一个 Python 脚本来在 Windows (BitBucket) 中执行此操作,该脚本还将根据以 YYYY-MM 开头的预定义文件夹命名约定编辑“创建日期”和“修改日期” -DD *。最后两个任务适用于 tif 文件和 jpeg,但 EXIF 不适用于 tif。

更新:

运行 exif 工具,我得到没有创建日期节点的输出,但是在 Windows 使用文件属性设置日期后,字段“创建日期”和“Date/Time原创”出现。此外,XMP 元值的原始文本打印输出在 Windows 中设置创建日期后提供了一个名为 xmp:createdate 的添加节点。我仍然不知道如何第一次在文件中创建这些字段。

更新 2:

Exif 似乎无法处理来自 Nikon Scan (2005) 的文件。唯一的选择是在文件中的 XMP 信息中添加 xmp:createdate 节点。 如果有人能告诉我这是如何完成的,无论是在 Python 中还是通过在 Windows 上调用 python 中的单独工具,它都值得赏金。

根据 Piexif documentationpiexif.insert 方法仅适用于 JPEG 或 WebP 文件。另一种方法是使用 PIL:

将当前 exif_bytes 保存到替换图像文件中
import piexif
from PIL import Image

def setImageDateTakenAttribute(filename, date_taken):
    img = Image.open(filename)
    exif_dict = piexif.load(filename)
    exif_dict['Exif'] = { 
        piexif.ExifIFD.DateTimeOriginal: datetime.datetime(*date_taken[:6]).strftime("%Y:%m:%d %H:%M:%S") 
    } 
    exif_bytes = piexif.dump(exif_dict)
    img.save(filename, 'tiff', exif=exif_bytes)

这个问题比我原先想象的要复杂。我在研究期间查看了以下 Python 个模块:

  • exif
  • exifread
  • piexif
  • 枕头
  • pyexiv2

一些模块接近于修改您想要更改的日期。但最后,我无法让任何模块正常工作。正确意味着在不损坏文件的情况下更改日期字段。最后,我将推荐一种不同的方法,它使用 subprocess 和适用于 Unix 和 Windows 的外部工具。这个工具就是exiftool,我已经用了很多年了。

import subprocess
from subprocess import check_output
from datatime import datetime

filename = 'Nikon.NEF'
rtn_data = check_output(['exiftool', filename])
print(rtn_data.decode("utf-8"))
# output 
...
Create Date : 2008:10:24 09:12:12.61
...

today = datetime.today()
new_date = today.strftime("%Y:%m:%d %H:%M:%S")
subprocess.call(['exiftool', f'-CreateDate={new_date}', filename])

changed_data = check_output(['exiftool', filename])
print(changed.decode("utf-8"))
# output 
...
Create Date : 2020:11:02 18:43:13
...

exiftool 允许您一次性更改任何设置和所有日期。

不使用 exiftool 更新:

您可以使用 piexif, 执行此操作,但您必须创建 TIFF 的副本并将其转换为 JPEG。我注意到,当您创建此副本时,一些元数据丢失了,根据您的用例,这可能是 no-go。

import piexif
from PIL import Image
from datetime import datetime
from PIL.ExifTags import TAGS

img = Image.open('test.tiff')

# get metadata
meta_dict = {TAGS[key]: img.tag[key] for key in img.tag.keys()}
exif_bytes = piexif.dump(meta_dict)

# get image height and width 
height = img.height
width = img.width

# resize the image and save it to a new file, which is a JPEG
img.resize((width, height), Image.ANTIALIAS).save('test2.jpeg', "JPEG", exif=exif_bytes, quality="web_high", optimize=True)

today = datetime.today()
new_date = today.strftime("%Y:%m:%d %H:%M:%S")

# load the metadata from the original file
exif_dict = piexif.load("test.tiff")

# change various dates
exif_dict['0th'][piexif.ImageIFD.DateTime] = bytes(new_date, 'utf-8')
exif_dict['Exif'][piexif.ExifIFD.DateTimeOriginal] = bytes(new_date, 'utf-8')
exif_dict['Exif'][piexif.ExifIFD.DateTimeDigitized] = bytes(new_date, 'utf-8')

# dump the changes
exif_bytes = piexif.dump(exif_dict)

# write the changes the the JPEG file
piexif.insert(exif_bytes, 'test2.jpeg')

我仍然更喜欢使用 exiftool,因为它需要更少的代码,并且不会丢失原始文件中的一些细节。