memories/lib/Db/LivePhoto.php

165 lines
6.4 KiB
PHP
Raw Normal View History

2022-11-22 11:10:25 +00:00
<?php
declare(strict_types=1);
namespace OCA\Memories\Db;
use OCA\Memories\Exif;
2022-11-22 11:10:25 +00:00
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\Files\File;
use OCP\IDBConnection;
class LivePhoto
{
public function __construct(private IDBConnection $connection) {}
2022-11-22 11:10:25 +00:00
/**
* Check if a given Exif data is the video part of a Live Photo.
*/
public function isVideoPart(array $exif): bool
2022-11-22 11:10:25 +00:00
{
return 'video/quicktime' === ($exif['MIMEType'] ?? null)
&& !empty($exif['ContentIdentifier'] ?? null);
2022-11-22 11:10:25 +00:00
}
/** Get liveid from photo part */
public function getLivePhotoId(File $file, array $exif): string
2022-11-22 11:10:25 +00:00
{
// Apple JPEG (MOV has ContentIdentifier)
if ($uuid = ($exif['ContentIdentifier'] ?? $exif['MediaGroupUUID'] ?? null)) {
return (string) $uuid;
2022-11-22 11:10:25 +00:00
}
// Google MVIMG and Samsung JPEG
if (($offset = ($exif['MicroVideoOffset'] ?? null)) && ($offset > 0)) {
// As explained in the following issue,
// https://github.com/pulsejet/memories/issues/468
//
// MicroVideoOffset is the length of the video in bytes
// and the video is located at the end of the file.
//
// Note that we could have just used "self__trailer" here,
// since exiftool can extract the video from the trailer,
// but explicitly specifying the offset is much faster because
// we don't need to spawn exiftool to read the video file.
//
// For Samsung JPEG, we can also check for EmbeddedVideoType
// and subsequently extract the video file using the
// EmbeddedVideoFile binary prop, but setting the offset
// is faster for the same reason mentioned above.
$videoOffset = $file->getSize() - $offset;
return "self__traileroffset={$videoOffset}";
}
// Google JPEG and Samsung HEIC / JPEG (Apple?)
if ($exif['MotionPhoto'] ?? null) {
if ('image/jpeg' === ($exif['MIMEType'] ?? null)) {
// Google Motion Photo JPEG
// We need to read the DirectoryItemLength key to get the length of the video
// These keys are duplicate, one for the image and one for the video
// With exiftool -G4, we get the following:
//
// "Unknown:DirectoryItemSemantic": "Primary"
// "Unknown:DirectoryItemLength": 0
// "Copy1:DirectoryItemSemantic": "MotionPhoto"
// "Copy1:DirectoryItemLength": 3011435 // <-- this is the length of the video
//
// The video is then located at the end of the file, so we can get the offset.
// Match each DirectoryItemSemantic to find MotionPhoto, then get the length.
$path = $file->getStorage()->getLocalFile($file->getInternalPath())
?: throw new \Exception('[BUG][LivePhoto] Failed to get local file path');
$extExif = Exif::getExifWithDuplicates($path);
foreach ($extExif as $key => $value) {
if (str_ends_with($key, ':DirectoryItemSemantic')) {
if ('MotionPhoto' === $value) {
$videoLength = $extExif[str_replace('Semantic', 'Length', $key)];
if (\is_int($videoLength) && $videoLength > 0) {
$videoOffset = $file->getSize() - $videoLength;
return "self__traileroffset={$videoOffset}";
}
}
}
}
// Fallback: video should hopefully be in trailer
return 'self__trailer';
}
if ('image/heic' === ($exif['MIMEType'] ?? null)) {
// Samsung HEIC -- no way to get this out yet (DirectoryItemLength is senseless)
// The reason this is above the MotionPhotoVideo check is because extracting binary
// EXIF fields on the fly is extremely expensive compared to trailer extraction.
}
}
// Samsung HEIC (at least S21)
if (!empty($exif['MotionPhotoVideo'] ?? null)) {
// It's a binary exif field, decode when the user requests it
return 'self__exifbin=MotionPhotoVideo';
}
2022-11-22 11:10:25 +00:00
return '';
}
/**
* Process video part of Live Photo.
*/
public function processVideoPart(File $file, array $exif): bool
2022-11-22 11:10:25 +00:00
{
$fileId = $file->getId();
$mtime = $file->getMTime();
$liveid = $exif['ContentIdentifier'] ?? null;
2022-11-22 11:10:25 +00:00
if (empty($liveid)) {
return false;
2022-11-22 11:10:25 +00:00
}
// Check if entry already exists
2022-11-22 11:10:25 +00:00
$query = $this->connection->getQueryBuilder();
$exists = $query->select('fileid')
2022-11-22 11:10:25 +00:00
->from('memories_livephoto')
->where($query->expr()->eq('fileid', $query->createNamedParameter($fileId, IQueryBuilder::PARAM_INT)))
->executeQuery()
->fetch()
2022-11-22 11:10:25 +00:00
;
// Construct query parameters
$query = $this->connection->getQueryBuilder();
$params = [
'liveid' => $query->createNamedParameter($liveid, IQueryBuilder::PARAM_STR),
'mtime' => $query->createNamedParameter($mtime, IQueryBuilder::PARAM_INT),
'fileid' => $query->createNamedParameter($fileId, IQueryBuilder::PARAM_INT),
'orphan' => $query->createNamedParameter(false, IQueryBuilder::PARAM_BOOL),
];
// Insert or update
if ($exists) {
$query->update('memories_livephoto')
->where($query->expr()->eq('fileid', $query->createNamedParameter($fileId, IQueryBuilder::PARAM_INT)))
;
foreach ($params as $key => $value) {
$query->set($key, $value);
2022-11-22 11:10:25 +00:00
}
} else {
$query->insert('memories_livephoto')->values($params);
2022-11-22 11:10:25 +00:00
}
return $query->executeStatement() > 0;
2022-11-22 11:10:25 +00:00
}
/**
* Delete entry from memories_livephoto table.
*/
public function deleteVideoPart(File $file): void
{
$query = $this->connection->getQueryBuilder();
$query->delete('memories_livephoto')
->where($query->expr()->eq('fileid', $query->createNamedParameter($file->getId(), IQueryBuilder::PARAM_INT)))
;
$query->executeStatement();
}
2022-11-22 11:10:25 +00:00
}