mirror of
https://github.com/nextcloud/server.git
synced 2026-02-20 00:12:30 -05:00
Allow scanning for metadata with occ scan:file --generate-metadata
Signed-off-by: Carl Schwan <carl@carlschwan.eu> Signed-off-by: Louis Chemineau <louis@chmn.me>
This commit is contained in:
parent
92d065b6eb
commit
54b6d0708b
4 changed files with 68 additions and 61 deletions
|
|
@ -37,8 +37,11 @@ use OC\Core\Command\Base;
|
|||
use OC\Core\Command\InterruptedException;
|
||||
use OC\DB\Connection;
|
||||
use OC\DB\ConnectionAdapter;
|
||||
use OCP\Files\File;
|
||||
use OC\ForbiddenException;
|
||||
use OC\Metadata\MetadataManager;
|
||||
use OCP\EventDispatcher\IEventDispatcher;
|
||||
use OCP\Files\IRootFolder;
|
||||
use OCP\Files\Mount\IMountPoint;
|
||||
use OCP\Files\NotFoundException;
|
||||
use OCP\Files\StorageNotAvailableException;
|
||||
|
|
@ -51,19 +54,22 @@ use Symfony\Component\Console\Input\InputOption;
|
|||
use Symfony\Component\Console\Output\OutputInterface;
|
||||
|
||||
class Scan extends Base {
|
||||
private IUserManager $userManager;
|
||||
protected float $execTime = 0;
|
||||
protected int $foldersCounter = 0;
|
||||
protected int $filesCounter = 0;
|
||||
private IRootFolder $root;
|
||||
private MetadataManager $metadataManager;
|
||||
|
||||
/** @var IUserManager $userManager */
|
||||
private $userManager;
|
||||
/** @var float */
|
||||
protected $execTime = 0;
|
||||
/** @var int */
|
||||
protected $foldersCounter = 0;
|
||||
/** @var int */
|
||||
protected $filesCounter = 0;
|
||||
|
||||
public function __construct(IUserManager $userManager) {
|
||||
public function __construct(
|
||||
IUserManager $userManager,
|
||||
IRootFolder $rootFolder,
|
||||
MetadataManager $metadataManager
|
||||
) {
|
||||
$this->userManager = $userManager;
|
||||
parent::__construct();
|
||||
$this->root = $rootFolder;
|
||||
$this->metadataManager = $metadataManager;
|
||||
}
|
||||
|
||||
protected function configure() {
|
||||
|
|
@ -83,6 +89,12 @@ class Scan extends Base {
|
|||
InputArgument::OPTIONAL,
|
||||
'limit rescan to this path, eg. --path="/alice/files/Music", the user_id is determined by the path and the user_id parameter and --all are ignored'
|
||||
)
|
||||
->addOption(
|
||||
'generate-metadata',
|
||||
null,
|
||||
InputOption::VALUE_NONE,
|
||||
'Generate metadata for all scanned files'
|
||||
)
|
||||
->addOption(
|
||||
'all',
|
||||
null,
|
||||
|
|
@ -106,21 +118,26 @@ class Scan extends Base {
|
|||
);
|
||||
}
|
||||
|
||||
protected function scanFiles($user, $path, OutputInterface $output, $backgroundScan = false, $recursive = true, $homeOnly = false) {
|
||||
protected function scanFiles(string $user, string $path, bool $scanMetadata, OutputInterface $output, bool $backgroundScan = false, bool $recursive = true, bool $homeOnly = false): void {
|
||||
$connection = $this->reconnectToDatabase($output);
|
||||
$scanner = new \OC\Files\Utils\Scanner(
|
||||
$user,
|
||||
new ConnectionAdapter($connection),
|
||||
\OC::$server->query(IEventDispatcher::class),
|
||||
\OC::$server->get(IEventDispatcher::class),
|
||||
\OC::$server->get(LoggerInterface::class)
|
||||
);
|
||||
|
||||
# check on each file/folder if there was a user interrupt (ctrl-c) and throw an exception
|
||||
|
||||
$scanner->listen('\OC\Files\Utils\Scanner', 'scanFile', function ($path) use ($output) {
|
||||
$scanner->listen('\OC\Files\Utils\Scanner', 'scanFile', function (string $path) use ($output, $scanMetadata) {
|
||||
$output->writeln("\tFile\t<info>$path</info>", OutputInterface::VERBOSITY_VERBOSE);
|
||||
++$this->filesCounter;
|
||||
$this->abortIfInterrupted();
|
||||
if ($scanMetadata) {
|
||||
$node = $this->root->get($path);
|
||||
if ($node instanceof File) {
|
||||
$this->metadataManager->generateMetadata($node, false);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
$scanner->listen('\OC\Files\Utils\Scanner', 'scanFolder', function ($path) use ($output) {
|
||||
|
|
@ -197,7 +214,7 @@ class Scan extends Base {
|
|||
++$user_count;
|
||||
if ($this->userManager->userExists($user)) {
|
||||
$output->writeln("Starting scan for user $user_count out of $users_total ($user)");
|
||||
$this->scanFiles($user, $path, $output, $input->getOption('unscanned'), !$input->getOption('shallow'), $input->getOption('home-only'));
|
||||
$this->scanFiles($user, $path, $input->getOption('generate-metadata'), $output, $input->getOption('unscanned'), !$input->getOption('shallow'), $input->getOption('home-only'));
|
||||
$output->writeln('', OutputInterface::VERBOSITY_VERBOSE);
|
||||
} else {
|
||||
$output->writeln("<error>Unknown user $user_count $user</error>");
|
||||
|
|
@ -291,7 +308,7 @@ class Scan extends Base {
|
|||
protected function formatExecTime() {
|
||||
$secs = round($this->execTime);
|
||||
# convert seconds into HH:MM:SS form
|
||||
return sprintf('%02d:%02d:%02d', (int)($secs / 3600), ( (int)($secs / 60) % 60), $secs % 60);
|
||||
return sprintf('%02d:%02d:%02d', (int)($secs / 3600), ((int)($secs / 60) % 60), $secs % 60);
|
||||
}
|
||||
|
||||
protected function reconnectToDatabase(OutputInterface $output): Connection {
|
||||
|
|
|
|||
|
|
@ -112,59 +112,36 @@ class FileMetadataMapper extends QBMapper {
|
|||
* @return Entity the saved entity with the set id
|
||||
* @throws Exception
|
||||
* @throws \InvalidArgumentException if entity has no id
|
||||
* @since 14.0.0
|
||||
*/
|
||||
public function update(Entity $entity): Entity {
|
||||
// if entity wasn't changed it makes no sense to run a db query
|
||||
$properties = $entity->getUpdatedFields();
|
||||
if (\count($properties) === 0) {
|
||||
return $entity;
|
||||
if (!($entity instanceof FileMetadata)) {
|
||||
throw new \Exception("Entity should be a FileMetadata entity");
|
||||
}
|
||||
|
||||
// entity needs an id
|
||||
$id = $entity->getId();
|
||||
if ($id === null) {
|
||||
throw new \InvalidArgumentException(
|
||||
'Entity which should be updated has no id');
|
||||
}
|
||||
|
||||
if (!($entity instanceof FileMetadata)) {
|
||||
throw new \Exception("Entity should be a FileMetadata entity");
|
||||
throw new \InvalidArgumentException('Entity which should be updated has no id');
|
||||
}
|
||||
|
||||
// entity needs an group_name
|
||||
$groupName = $entity->getGroupName();
|
||||
if ($id === null) {
|
||||
throw new \InvalidArgumentException(
|
||||
'Entity which should be updated has no group_name');
|
||||
}
|
||||
|
||||
// get updated fields to save, fields have to be set using a setter to
|
||||
// be saved
|
||||
// do not update the id and group_name field
|
||||
unset($properties['id']);
|
||||
unset($properties['group_name']);
|
||||
|
||||
$qb = $this->db->getQueryBuilder();
|
||||
$qb->update($this->tableName);
|
||||
|
||||
// build the fields
|
||||
foreach ($properties as $property => $updated) {
|
||||
$column = $entity->propertyToColumn($property);
|
||||
$getter = 'get' . ucfirst($property);
|
||||
$value = $entity->$getter();
|
||||
|
||||
$type = $this->getParameterTypeForProperty($entity, $property);
|
||||
$qb->set($column, $qb->createNamedParameter($value, $type));
|
||||
if ($groupName === null) {
|
||||
throw new \InvalidArgumentException('Entity which should be updated has no group_name');
|
||||
}
|
||||
|
||||
$idType = $this->getParameterTypeForProperty($entity, 'id');
|
||||
$groupNameType = $this->getParameterTypeForProperty($entity, 'groupName');
|
||||
$metadataValue = $entity->getMetadata();
|
||||
$metadataType = $this->getParameterTypeForProperty($entity, 'metadata');
|
||||
|
||||
$qb->where($qb->expr()->eq('id', $qb->createNamedParameter($id, $idType)))
|
||||
->andWhere($qb->expr()->eq('group_name', $qb->createNamedParameter($groupName, $groupNameType)));
|
||||
$qb = $this->db->getQueryBuilder();
|
||||
|
||||
$qb->executeStatement();
|
||||
$qb->update($this->tableName)
|
||||
->set('metadata', $qb->createNamedParameter($metadataValue, $metadataType))
|
||||
->where($qb->expr()->eq('id', $qb->createNamedParameter($id, $idType)))
|
||||
->andWhere($qb->expr()->eq('group_name', $qb->createNamedParameter($groupName, $groupNameType)))
|
||||
->executeStatement();
|
||||
|
||||
return $entity;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -21,27 +21,19 @@ namespace OC\Metadata;
|
|||
|
||||
use OC\Metadata\Provider\ExifProvider;
|
||||
use OCP\Files\File;
|
||||
use OCP\IConfig;
|
||||
use Psr\Log\LoggerInterface;
|
||||
|
||||
class MetadataManager implements IMetadataManager {
|
||||
/** @var array<string, IMetadataProvider> */
|
||||
private array $providers;
|
||||
private array $providerClasses;
|
||||
private FileMetadataMapper $fileMetadataMapper;
|
||||
private IConfig $config;
|
||||
private LoggerInterface $logger;
|
||||
|
||||
public function __construct(
|
||||
FileMetadataMapper $fileMetadataMapper,
|
||||
IConfig $config,
|
||||
LoggerInterface $logger
|
||||
FileMetadataMapper $fileMetadataMapper
|
||||
) {
|
||||
$this->providers = [];
|
||||
$this->providerClasses = [];
|
||||
$this->fileMetadataMapper = $fileMetadataMapper;
|
||||
$this->config = $config;
|
||||
$this->logger = $logger;
|
||||
|
||||
// TODO move to another place, where?
|
||||
$this->registerProvider(ExifProvider::class);
|
||||
|
|
|
|||
|
|
@ -1,5 +1,25 @@
|
|||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
/**
|
||||
* @copyright Copyright 2022 Carl Schwan <carl@carlschwan.eu>
|
||||
* @copyright Copyright 2022 Louis Chmn <louis@chmn.me>
|
||||
* @license AGPL-3.0-or-later
|
||||
*
|
||||
* This code is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU Affero General Public License, version 3,
|
||||
* as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU Affero General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Affero General Public License, version 3,
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
||||
*
|
||||
*/
|
||||
|
||||
namespace OC\Metadata\Provider;
|
||||
|
||||
use OC\Metadata\FileMetadata;
|
||||
|
|
@ -24,6 +44,7 @@ class ExifProvider implements IMetadataProvider {
|
|||
return extension_loaded('exif');
|
||||
}
|
||||
|
||||
/** @return array{'gps': FileMetadata, 'size': FileMetadata} */
|
||||
public function execute(File $file): array {
|
||||
$exifData = [];
|
||||
$fileDescriptor = $file->fopen('rb');
|
||||
|
|
|
|||
Loading…
Reference in a new issue