Добавить информацию Exif/XMP "Дата съемки" в файл TIF с помощью Python
Используя следующий код Python в Windows 10, я пытаюсь отредактировать информацию EXIF для "Date Taken" в файле изображения tif, созданном с помощью 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)
Это отлично работает в Python 3 для файлов jpeg, но когда я пытаюсь использовать тот же код для редактирования своего файла 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
Я до сих пор искал другие пакеты, которые могли бы поддерживать мой файл, но безрезультатно.
Кто-нибудь знает, как редактирование может быть выполнено в Python без удаления существующей информации exif и без изменения формата изображения?
Чтобы воспроизвести или получить образец неисправных файлов tif, клонируйте мой проект (ссылка ниже).
Детали:
После сканирования тысяч изображений в файлы tif я хотел бы указать значение EXIF для "Дата съемки". Я пишу сценарий Python для этого в Windows (BitBucket), который также будет редактировать "Дата создания" и "Дата изменения" из предопределенного соглашения об именах папок, начиная с ГГГГ-ММ-ДД *. Последние две задачи работают как с файлами tif, так и с jpeg, но EXIF не работает с файлами tif.
Обновить:
Запуская инструмент exif, я получаю вывод без узла даты создания, но после установки даты в Windows с помощью свойств файла появляются поля "Дата создания" и "Исходная дата / время". Кроме того, распечатка необработанного текста с мета-значениями XMP дает добавленный узел под названием xmp:createdate после установки даты создания в Windows. Тем не менее, я не могу понять, как создать эти поля в файле впервые.
Обновление 2:
Похоже, Exif не работает с файлами из Nikon Scan (2005). Единственный вариант - добавить узел xmp:createdate к информации XMP в файле. Если кто-нибудь может показать мне, как это делается, либо на чистом Python, либо вызвав отдельный инструмент из python в Windows, он заслуживает полной награды.
2 ответа
Этот вопрос был более сложным, чем я думал изначально. Во время исследования я посмотрел на следующие модули Python:
- exif
- exifread
- Piexif
- подушка
- pyexiv2
Некоторые модули приблизились к изменению даты, которую вы хотите изменить. Но в итоге я не смог заставить работать ни один из модулей правильно. Правильно означает изменение поля даты без повреждения файла. В конце, я собираюсь порекомендовать другой подход, который использует подпроцесс и внешний инструмент, который работает для 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, потому что он требует меньше кода и не теряет некоторые детали из исходного файла.
Согласно документации Piexif,
piexif.insert
метод работает только для файлов JPEG или WebP. Альтернативой было бы сохранение текущего
exif_bytes
в замещающий файл изображения с помощью PIL:
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)