Perform periodic fixity checks on selected files.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

290 lines
10 KiB

<?php
namespace Drupal\dgi_fixity;
use Drupal\Core\Batch\BatchBuilder;
use Drupal\Core\StringTranslation\PluralTranslatableMarkup;
use Drupal\dgi_fixity\Form\SettingsForm;
/**
* Performs fixity checks.
*/
class FixityCheckBatchCheck {
/**
* Creates a batch for performing fixity checks.
*
* @param int[] $fids
* A list of file identifiers, if not specified files with periodic checks
* enabled will be selected.
* @param bool $force
* A flag to indicate if the check should be performed even if the time
* elapsed since the last check has not exceed the required threshold.
* @param int $batch_size
* The number of of files to process at a time.
* If not specified it will default to the modules configuration.
*/
public static function build(array $fids = NULL, bool $force = FALSE, int $batch_size = NULL) {
$batch_size = is_null($batch_size) ? \Drupal::config(SettingsForm::CONFIG_NAME)->get(SettingsForm::BATCH_SIZE) : $batch_size;
return is_null($fids) ?
static::buildPeriodic($force, $batch_size) :
static::buildFixed($fids, $force, $batch_size);
}
/**
* Creates a batch for processing a fixed list of file identifiers.
*/
protected static function buildFixed(array $fids, bool $force, int $batch_size) {
$builder = new BatchBuilder();
return $builder
->setTitle(\t('Performing checks on @count file(s)', ['@count' => count($fids)]))
->setInitMessage(\t('Starting'))
->setErrorMessage(\t('Batch has encountered an error'))
->addOperation([static::class, 'processFixedList'], [
$fids,
$force,
$batch_size,
])
->setFinishCallback([static::class, 'finished'])
->toArray();
}
/**
* Creates a batch for processing files that have periodic checks enabled.
*/
public static function buildPeriodic(bool $force, int $batch_size) {
$sources = \Drupal::config(SettingsForm::CONFIG_NAME)->get(SettingsForm::SOURCES);
if (empty($sources)) {
throw new \InvalidArgumentException("No sources specified, check the modules configuration.");
}
$builder = new BatchBuilder();
foreach ($sources as $source) {
$builder->addOperation(
[static::class, 'processSource'],
[$source, $batch_size],
);
}
return $builder
->setTitle(\t('Enumerating periodic checks from @count Source(s)', ['@count' => count($sources)]))
->setInitMessage(\t('Starting'))
->setErrorMessage(\t('Batch has encountered an error'))
->addOperation([static::class, 'processPeriodic'], [$force, $batch_size])
->setFinishCallback([static::class, 'finished'])
->toArray();
}
/**
* Check the given files.
*
* @param int[] $fids
* A list of file identifiers.
* @param bool $force
* A flag to indicate if the check should be performed even if the time
* elapsed since the last check has not exceed the required threshold.
* @param int $batch_size
* The amount of files each time this process runs.
* @param array|object $context
* Context for operations.
*/
public static function processFixedList(array $fids, bool $force, int $batch_size, &$context) {
$sandbox = &$context['sandbox'];
$results = &$context['results'];
if (!isset($sandbox['total'])) {
$sandbox['offset'] = 0;
$sandbox['total'] = count($fids);
$results['successful'] = 0;
$results['ignored'] = 0;
$results['skipped'] = 0;
$results['failed'] = 0;
$results['errors'] = [];
}
$chunk = array_slice($fids, $sandbox['offset'], $batch_size);
$end = min($sandbox['total'], $sandbox['offset'] + count($chunk));
$context['message'] = \t('Processing @start to @end of @total', [
'@start' => $sandbox['offset'],
'@end' => $end,
'@total' => $sandbox['total'],
]);
$files = \Drupal::service('entity_type.manager')->getStorage('file')->loadMultiple($chunk);
// It is possible for non existing fids to be listed in $chunk, in such
// cases this is ignored.
$results['ignored'] += count(array_diff($chunk, array_keys($files)));
static::check($files, $force, $results);
$sandbox['offset'] = $end;
$context['finished'] = $sandbox['offset'] / $sandbox['total'];
}
/**
* Enable periodic checks on files as returned by the give source view.
*/
public static function processSource(string $source, $batch_size, &$context) {
$results = &$context['results'];
// Do not track success/failure on processing source, as that is done for
// checks only. Errors however do get passed though.
if (!isset($results['errors'])) {
$results['errors'] = [];
}
/** @var \Drupal\dgi_fixity\FixityCheckServiceInterface $fixity */
$fixity = \Drupal::service('dgi_fixity.fixity_check');
$view = $fixity->source($source, $batch_size);
$view->execute();
// Only processes those which have not already enabled periodic checks.
foreach ($view->result as $row) {
try {
/** @var \Drupal\dgi_fixity\FixityCheckInterface $check */
$check = $view->field['periodic']->getEntity($row);
$check->setPeriodic(TRUE);
$check->save();
}
catch (\Exception $e) {
$results['errors'][] = \t('Encountered an exception: @exception', [
'@exception' => $e,
]);
// In practice exceptions in this case shouldn't arise, but if they do
// exit to prevent an infinite loop by exiting the operation.
$context['finished'] = 1;
return;
}
}
// End when we have exhausted all inputs.
$context['finished'] = count($view->result) == 0;
}
/**
* Checks all files which have enabled periodic fixity checks.
*
* @param bool $force
* A flag to indicate if the check should be performed even if the time
* elapsed since the last check has not exceed the required threshold.
* @param int $batch_size
* The amount of files each time this process runs.
* @param array|object $context
* Context for operations.
*/
public static function processPeriodic(bool $force, int $batch_size, &$context) {
/** @var \Drupal\dgi_fixity\FixityCheckStorageInterface $storage */
$storage = \Drupal::entityTypeManager()->getStorage('fixity_check');
$sandbox = &$context['sandbox'];
$results = &$context['results'];
if (!isset($sandbox['offset'])) {
$sandbox['offset'] = 0;
$sandbox['remaining'] = $storage->countPeriodic();
$results['successful'] = 0;
$results['ignored'] = 0;
$results['skipped'] = 0;
$results['failed'] = 0;
$results['errors'] = $results['errors'] ?? [];
}
$files = $storage->getPeriodic($sandbox['offset'], $batch_size);
$end = min($sandbox['total'], $sandbox['offset'] + count($files));
$context['message'] = \t('Processing @start to @end', [
'@start' => $sandbox['offset'],
'@end' => $end,
]);
static::check($files, $force, $results);
$sandbox['offset'] = $end;
$remaining = $storage->countPeriodic();
$progress_halted = $sandbox['remaining'] == $remaining;
$sandbox['remaining'] = $remaining;
// End when we have exhausted all inputs or progress has halted.
$context['finished'] = empty($files) || $progress_halted;
}
/**
* Performs a fixity check on the given list of files.
*
* @param \Drupal\file\FileInterface[] $files
* A list of file identifiers.
* @param bool $force
* A flag to indicate if the check should be performed even if the time
* elapsed since the last check has not exceed the required threshold.
* @param array &$results
* An associative array with the results of the fixity checks
* - missing: The number of file identifiers for which no files exist.
* - successful: The number of files that were successfully checked.
* - errors: A list of error messages if any occurred.
*/
protected static function check(array $files, bool $force, array &$results) {
/** @var \Drupal\dgi_fixity\FixityCheckServiceInterface $fixity */
$fixity = \Drupal::service('dgi_fixity.fixity_check');
foreach ($files as $file) {
try {
$result = $fixity->check($file, $force);
if ($result instanceof FixityCheckInterface) {
if ($result->passed()) {
$results['successful']++;
}
else {
$results['failed']++;
}
}
else {
// The check was not performed as the time elapsed since the last
// check did not exceed the required threshold.
$results['skipped']++;
}
}
catch (\Exception $e) {
$results['failed']++;
$results['errors'][] = \t('Encountered an exception: @exception', [
'@exception' => $e,
]);
}
}
}
/**
* Batch Finished callback.
*
* @param bool $success
* Success of the operation.
* @param array $results
* Array of results for post processing.
* @param array $operations
* Array of operations.
*/
public static function finished($success, array $results, array $operations) {
$messenger = \Drupal::messenger();
$messenger->addStatus(new PluralTranslatableMarkup(
$results['successful'] + $results['ignored'] + $results['skipped'] + $results['failed'],
'Processed @count item in total.',
'Processed @count items in total.'
));
$messenger->addStatus(new PluralTranslatableMarkup(
$results['successful'],
'@count was successful.',
'@count were successful.',
));
$messenger->addStatus(new PluralTranslatableMarkup(
$results['ignored'],
'@count was ignored.',
'@count were ignored.',
));
$messenger->addStatus(new PluralTranslatableMarkup(
$results['skipped'],
'@count was skipped.',
'@count were skipped.',
));
$messenger->addStatus(\t(
'@count failed.', ['@count' => $results['failed']]
));
$error_count = count($results['errors']);
if ($error_count > 0) {
$messenger->addMessage(new PluralTranslatableMarkup(
$error_count,
'@count error occurred.',
'@count errors occurred.',
));
foreach ($results['errors'] as $error) {
$messenger->addError($error);
}
}
}
}