IdAuditor.php

Same filename and directory in other branches
  1. 8.9.x core/modules/migrate/src/Audit/IdAuditor.php
  2. 10 core/modules/migrate/src/Audit/IdAuditor.php
  3. 11.x core/modules/migrate/src/Audit/IdAuditor.php

Namespace

Drupal\migrate\Audit

File

core/modules/migrate/src/Audit/IdAuditor.php

View source
<?php

namespace Drupal\migrate\Audit;

use Drupal\Core\StringTranslation\StringTranslationTrait;
use Drupal\migrate\Plugin\migrate\destination\EntityContentComplete;
use Drupal\migrate\Plugin\MigrationInterface;
// cspell:ignore destid

/**
 * Audits migrations that create content entities in the destination system.
 */
class IdAuditor implements AuditorInterface {
    use StringTranslationTrait;
    
    /**
     * {@inheritdoc}
     */
    public function audit(MigrationInterface $migration) {
        // If the migration does not opt into auditing, it passes.
        if (!$migration->isAuditable()) {
            return AuditResult::pass($migration);
        }
        $interface = HighestIdInterface::class;
        $destination = $migration->getDestinationPlugin();
        if (!$destination instanceof HighestIdInterface) {
            throw new AuditException($migration, "Destination does not implement {$interface}");
        }
        $id_map = $migration->getIdMap();
        if (!$id_map instanceof HighestIdInterface) {
            throw new AuditException($migration, "ID map does not implement {$interface}");
        }
        if ($destination->getHighestId() > $id_map->getHighestId() || $destination instanceof EntityContentComplete && !$this->auditEntityComplete($migration)) {
            return AuditResult::fail($migration, [
                $this->t('The destination system contains data which was not created by a migration.'),
            ]);
        }
        return AuditResult::pass($migration);
    }
    
    /**
     * {@inheritdoc}
     */
    public function auditMultiple(array $migrations) {
        $conflicts = [];
        foreach ($migrations as $migration) {
            $migration_id = $migration->getPluginId();
            $conflicts[$migration_id] = $this->audit($migration);
        }
        ksort($conflicts);
        return $conflicts;
    }
    
    /**
     * Audits an EntityComplete migration.
     *
     * @param \Drupal\migrate\Plugin\MigrationInterface $migration
     *   The migration to audit.
     *
     * @return bool
     *   TRUE if the audit passes and FALSE if not.
     *
     * @todo Refactor in https://www.drupal.org/project/drupal/issues/3061676 or
     *   https://www.drupal.org/project/drupal/issues/3091004
     */
    private function auditEntityComplete(MigrationInterface $migration) {
        $map_table = $migration->getIdMap()
            ->mapTableName();
        $database = \Drupal::database();
        if (!$database->schema()
            ->tableExists($map_table)) {
            throw new \InvalidArgumentException();
        }
        $query = $database->select($map_table, 'map')
            ->fields('map', [
            'destid2',
        ])
            ->range(0, 1)
            ->orderBy('destid2', 'DESC');
        $max = (int) $query->execute()
            ->fetchField();
        // Make a migration based on node_complete but with an entity_revision
        // destination.
        $revision_migration = $migration->getPluginDefinition();
        $revision_migration['id'] = $migration->getPluginId() . '-revision';
        $revision_migration['destination']['plugin'] = 'entity_revision:node';
        $revision_migration = \Drupal::service('plugin.manager.migration')->createStubMigration($revision_migration);
        // Get the highest node revision ID.
        $destination = $revision_migration->getDestinationPlugin();
        $highest = $destination->getHighestId();
        return $max <= $highest;
    }

}

Classes

Title Deprecated Summary
IdAuditor Audits migrations that create content entities in the destination system.

Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.