Добавить информацию 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)
Другие вопросы по тегам