问题描述
在Windows 10上使用以下Python代码,我试图在使用Nikon Scan 4.0.3(2008年左右的软件)创建的tif图像文件上编辑“拍摄日期”的EXIF信息。
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文件的样本,请克隆我的项目(链接如下)。
详细信息:
将数千张图片扫描到tif文件后,我想为“拍摄日期”指定EXIF值。我正在编写Python脚本以在Windows (BitBucket)中执行此操作,该脚本还将从以YYYY-MM-DD *开头的预定义文件夹命名约定中编辑“创建日期”和“修改日期”。最后两个任务适用于tif文件和jpeg,但EXIF不适用于tif。
更新:
运行exif工具,我得到的输出没有创建日期节点,但是在Windows中使用文件属性设置日期后,出现了“创建日期”和“原始日期/时间”字段。同样,在Windows中设置创建日期后,XMP元值的原始文本打印输出会添加一个名为xmp:createdate的节点。仍然我不知道如何第一次在文件中创建这些字段。
更新2:
似乎Exif不适用于Nikon Scan(2005)中的文件。唯一的选择是将xmp:createdate节点添加到文件中的XMP信息中。 如果有人可以用纯Python或通过在Windows上从python调用单独的工具向我展示如何做到这一点,那将是徒劳的。
解决方法
根据Piexif documentation,piexif.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。我注意到,当您创建此副本时,某些元数据会丢失,根据您的用例,这可能是不可行的。
import piexif
from PIL import Image
from datatime import datetime
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.width
width = img.width
# resize the image ad same it to a new file,which is a JPEG
img.resize((height,width),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,因为它需要更少的代码并且不会丢失原始文件中的某些细节。