php 支持 WEBP 图像元数据

php support for WEBP image metadata

php是否支持 webp 图像元数据?

具体来说,我希望能够在 php 代码中本地读取和写入 webp 图像的 XMP 和 EXIF 元数据。

我一直在试验下面的代码,它给我一个“文件不受支持”的警告。

<?php

$photoSourceThumbnail = "publicAssets/images/att_galleryWebP/A0001_LSF-PHOTOS-WM-TM-WEBP/A0001-EWF-LSF-01.webp";
$photoSourceFull = "assets/images/att_galleryWebP/A0001_LSF-PHOTOS-WM-FULL-WEBP/A0001-EWF-LSF-01.webp";

echo "$photoSourceFull:<br />\n";
$exif = exif_read_data($photoSourceFull, 'IFD0');
echo $exif===false ? "No header data found.<br />\n" : "Image contains headers<br />\n";

$exif = exif_read_data($photoSourceFull, 0, true);
echo "test2.jpg:<br />\n";
foreach ($exif as $key => $section) {
    foreach ($section as $name => $val) {
        echo "$key.$name: $val<br />\n";
    }
} 

最好使用ExifTool。

安装 ExifTool

https://exiftool.org/

PHP 例子

class ExifToolException extends RuntimeException{}

function getInfo(string $file) : object 
{
    $info = shell_exec('exiftool -json ' . escapeshellarg($file) . ' 2>&1');
    if(strpos($info, 'Error:') > -1) {
        throw new ExifToolException(rtrim($info, PHP_EOL));
    }
    return json_decode($info)[0];
}

try {
    var_dump(getInfo('abc.webp')->Megapixels);
} catch(ExifToolException $e) {
    var_dump($e->getMessage());
}

更新:ExifTool不支持写webp

您可以从 Google 查看 webpmux: https://developers.google.com/speed/webp/docs/webpmux

WebP supports both Exif and XMP for years already. The format is based on RIFF 并至少确认块 EXIFXMP ICCP。好的软件也会读取 IPTC 块。

仅仅因为一些随机软件不考虑携带元数据并不意味着目标格式不支持它 - 大多数软件都相当草率而不是雄心勃勃。由于 RIFFormat 相当简单,因此修改现有文件以包含更多块应该很容易:

  • 每个块包含 4 个字节的标识符/FourCC (i.e. the ASCII characters EXIF), then 4 bytes of its size in little Endian,然后是有效负载。
  • The file's first chunk is similar:开始是4字节RIFF,然后是它整个文件大小的4字节减8,然后是内容标识的4字节,这里是WEBP(为了区别这个来自其他也使用 RIFF 的格式,例如 WAV、AVI、Maya、AIF、MIDI...)。
  • 这意味着您将新块附加到文件末尾,然后使用新文件大小修补文件的第 5 到第 8 字节。示例:
$sExif= '...the binary data...';  // You read that off the other file, of course
$iLenExif= strlen( $sExif );  // Payload length in bytes
if( $iLenExif% 2== 1 ) $sExif.= "[=10=]";  // RIFF needs 16bit alignment

$hFile= fopen( 'TARGET.WEBP', 'r+' );   // Read and write access
fseek( $hFile, 0, SEEK_END );  // Go to end of file

fwrite( $hFile, 'EXIF' );  // 4 bytes chunk ID
fwrite( $hFile, pack( 'V', $iLenExif ) );  // 4 bytes of payload length
fwrite( $hFile, $sExif );  // Actual data

$iFilesize= ftell( $hFile );  // Should be bigger
fseek( $hFile, 4, SEEK_SET );  // Go to 5th byte of file
fwrite( $hFile, pack( 'V', $iFilesize- 8 ) );  // Write 4 bytes, patching old filesize

fclose( $hFile );  // Store everything.

Google 最近更改了网站的图像要求,表示它们应包含 IPTC 图像版权元数据。

https://developers.google.com/search/docs/advanced/appearance/image-rights-metadata

我已经成功地使用 EXIFTOOL 从我的 SQL table 更新了 jpg 图像的图像 XMP、EXIF 和 IPTC 元数据以满足这些新的 google 要求。 我已经检查了 IPTC 工具上的测试图像,它确实包含所需的 XMP、EXIF 和 IPTC 元数据。

https://getpmd.iptc.org/getiptcpmd.html

不过…………

广泛阅读 WEBP 格式和 exiv2、dwebp 和 webpmux 等工具,只能将 XMP 和 EXIF 元数据写入 webp 图像。

https://www.exiv2.org/manpage.html
https://developers.google.com/speed/webp/docs/cwebp
https://developers.google.com/speed/webp/docs/webpmux
https://image.online-convert.com/convert-to-webp (drag and drop web tool that converts Jpg to webp with XMP & Exif metadata)

看来 webp 不支持 IPTC 元数据因此不满足这些要求:

https://developers.google.com/search/docs/advanced/appearance/image-rights-metadata

结论:

  1. Webp 作为我想要 google 索引的图像格式,是一个死鸭子 unless/until google 修改 RIFF header 以包含 IPTC。
  2. 下一步是将我网站上的图片恢复为 jpg。

对于任何对我如何从 php 修改图像 XMP、EXIF 和 IPTC 元数据感兴趣的人。

a) 这些是我最终决定使用 exiftool 命令更新它们的元数据字段。

exiftool -iptc:by-line="image creator" A0000-01.jpg
exiftool -xmp:creator="image creator" A0000-01.jpg
exiftool -exif:Artist="image creator" A0000-01.jpg

exiftool -iptc:CopyrightNotice="Copywrite 2022 websiteName.com" A0000-01.jpg
exiftool -xmp:rights="Copywrite 2022 websiteName.com" A0000-01.jpg
exiftool -exif:Copyright="Copywrite 2022 websiteName.com" A0000-01.jpg

exiftool -iptc:keywords="keyword1,keyword2,keyword3" A0000-01.jpg
exiftool -xmp:Subject="keyword1,keyword2,keyword3" A0000-01.jpg
exiftool -exif:UserComment="keyword1,keyword2,keyword3" A0000-01.jpg

exiftool -iptc:credit="image reproduced with permission from" A0000-01.jpg
exiftool -xmp:credit="image reproduced with permission from" A0000-01.jpg

exiftool -iptc:ObjectName="Image title" A0000-01.jpg
exiftool -xmp:Title="Image title" A0000-01.jpg

exiftool -iptc:Caption-Abstract="Image description" A0000-01.jpg
exiftool -xmp:description="Image description" A0000-01.jpg
exiftool -exif:ImageDescription="Image description" A0000-01.jpg

exiftool -iptc:Source="Url where image can be found" A0000-01.jpg
exiftool -xmp:Source="Url where image can be found" A0000-01.jpg

exiftool -exif:gpslatitude="44.081102" -exif:gpslatituderef=N A0000-01.jpg
exiftool -exif:gpslongitude="-35.489600" -exif:gpslongituderef=W A0000-01.jpg
exiftool -xmp:gpslatitude="44.081102 N" A0000-01.jpg
exiftool -gpslongitude="-35.489600 E" A0000-01.jpg

exiftool -xmp:AuthorsPosition="website owner & coder" A0000-01.jpg
exiftool -iptc:By-lineTitle="website owner & coder" A0000-01.jpg

exiftool -xmp:CaptionWriter="websiteName.com.com" A0000-01.jpg
exiftool -iptc:Writer-Editor="websiteName.com.com" A0000-01.jpg

b) 在 php 中,我使用以下代码生成包含 exiftool 命令的文本字符串:

$F_XmpDescriptionSt = 'exiftool -xmp:description="'.$iptcDescription.'" "'.$absoluteImgJpgFullSizePath.'"';

其中 $iptcDescription 和 $absoluteImgJpgFullSizePath 是 mySQL table

中所需的元数据值

c) 然后我使用以下代码通过命令提示符终端启动命令:

$FullsizeExiftoolXmpDescriptionExecute = exec("$F_XmpDescriptionSt");

我对要更新的所有元数据字段重复上述操作,将 php 代码调整为 exiftool 命令。

我确定有更优雅的方式来执行此操作,例如批处理脚本?但我更喜欢将每个 exiftool 命令作为一行代码逐行启动,这样我就可以为我更新的每个元数据字段收到响应消息“1 个文件已更新”。完成所有元数据字段的代码大约需要 15 秒,所以不是什么大问题。

这行代码输出包含在变量 FullsizeExiftoolXmpDescriptionExecute 中的 exiftool 响应。

<h4>Fullsize XMP Description $FullsizeExiftoolXmpDescriptionExecute to $iptcDescription</h4>

本视频教程介绍了如何在 windows10 上安装 exiftool。 https://www.youtube.com/watch?v=Ku1Nx-kl7RM