class WorkspaceAssociation

Provides a class for CRUD operations on workspace associations.

Hierarchy

Expanded class hierarchy of WorkspaceAssociation

5 files declare their use of WorkspaceAssociation
QueryTrait.php in core/modules/workspaces/src/EntityQuery/QueryTrait.php
Tables.php in core/modules/workspaces/src/EntityQuery/Tables.php
ViewsOperations.php in core/modules/workspaces/src/Hook/ViewsOperations.php
WorkspaceAssociationTest.php in core/modules/workspaces/tests/src/Kernel/WorkspaceAssociationTest.php
workspaces.install in core/modules/workspaces/workspaces.install
Contains install, update and uninstall functions for the Workspaces module.
1 string reference to 'WorkspaceAssociation'
workspaces.services.yml in core/modules/workspaces/workspaces.services.yml
core/modules/workspaces/workspaces.services.yml
1 service uses WorkspaceAssociation
workspaces.association in core/modules/workspaces/workspaces.services.yml
Drupal\workspaces\WorkspaceAssociation

File

core/modules/workspaces/src/WorkspaceAssociation.php, line 19

Namespace

Drupal\workspaces
View source
class WorkspaceAssociation implements WorkspaceAssociationInterface, EventSubscriberInterface {
  
  /**
   * The table for the workspace association storage.
   */
  const TABLE = 'workspace_association';
  
  /**
   * The table for the workspace association revision storage.
   */
  const REVISION_TABLE = 'workspace_association_revision';
  
  /**
   * A multidimensional array of entity IDs that are associated to a workspace.
   *
   * The first level keys are workspace IDs, the second level keys are entity
   * type IDs, and the third level array are entity IDs, keyed by revision IDs.
   *
   * @var array
   */
  protected array $associatedRevisions = [];
  
  /**
   * A multidimensional array of entity IDs that were created in a workspace.
   *
   * The first level keys are workspace IDs, the second level keys are entity
   * type IDs, and the third level array are entity IDs, keyed by revision IDs.
   *
   * @var array
   */
  protected array $associatedInitialRevisions = [];
  public function __construct(protected Connection $database, protected EntityTypeManagerInterface $entityTypeManager, protected WorkspaceRepositoryInterface $workspaceRepository, protected LoggerInterface $logger) {
  }
  
  /**
   * {@inheritdoc}
   */
  public function trackEntity(RevisionableInterface $entity, WorkspaceInterface $workspace) {
    // Determine all workspaces that might be affected by this change.
    $affected_workspaces = $this->workspaceRepository
      ->getDescendantsAndSelf($workspace->id());
    // Get the currently tracked revision for this workspace.
    $tracked = $this->getTrackedEntities($workspace->id(), $entity->getEntityTypeId(), [
      $entity->id(),
    ]);
    $tracked_revision_id = NULL;
    if (isset($tracked[$entity->getEntityTypeId()])) {
      $tracked_revision_id = key($tracked[$entity->getEntityTypeId()]);
    }
    $id_field = static::getIdField($entity->getEntityTypeId());
    try {
      $transaction = $this->database
        ->startTransaction();
      // Update all affected workspaces that were tracking the current revision.
      // This means they are inheriting content and should be updated.
      if ($tracked_revision_id) {
        $this->database
          ->update(static::TABLE)
          ->fields([
          'target_entity_revision_id' => $entity->getRevisionId(),
        ])
          ->condition('workspace', $affected_workspaces, 'IN')
          ->condition('target_entity_type_id', $entity->getEntityTypeId())
          ->condition($id_field, $entity->id())
          ->condition('target_entity_revision_id', $tracked_revision_id)
          ->execute();
      }
      // Insert a new index entry for each workspace that is not tracking this
      // entity yet.
      $missing_workspaces = array_diff($affected_workspaces, $this->getEntityTrackingWorkspaceIds($entity));
      if ($missing_workspaces) {
        $insert_query = $this->database
          ->insert(static::TABLE)
          ->fields([
          'workspace',
          'target_entity_type_id',
          $id_field,
          'target_entity_revision_id',
        ]);
        foreach ($missing_workspaces as $workspace_id) {
          $insert_query->values([
            'workspace' => $workspace_id,
            'target_entity_type_id' => $entity->getEntityTypeId(),
            $id_field => $entity->id(),
            'target_entity_revision_id' => $entity->getRevisionId(),
          ]);
        }
        $insert_query->execute();
      }
      // Individual revisions are tracked in a separate table only for the
      // workspace in which they were created or updated.
      $this->database
        ->insert(static::REVISION_TABLE)
        ->fields([
        'workspace' => $workspace->id(),
        'target_entity_type_id' => $entity->getEntityTypeId(),
        $id_field => $entity->id(),
        'target_entity_revision_id' => $entity->getRevisionId(),
        'initial_revision' => (int) $entity->isDefaultRevision(),
      ])
        ->execute();
    } catch (\Exception $e) {
      if (isset($transaction)) {
        $transaction->rollBack();
      }
      Error::logException($this->logger, $e);
      throw $e;
    }
    $this->associatedRevisions = $this->associatedInitialRevisions = [];
  }
  
  /**
   * {@inheritdoc}
   */
  public function workspaceInsert(WorkspaceInterface $workspace) {
    // When a new workspace has been saved, we need to copy all the associations
    // of its parent.
    if ($workspace->hasParent()) {
      $this->initializeWorkspace($workspace);
    }
  }
  
  /**
   * {@inheritdoc}
   */
  public function getTrackedEntities($workspace_id, $entity_type_id = NULL, $entity_ids = NULL) {
    $query = $this->database
      ->select(static::TABLE);
    $query->fields(static::TABLE, [
      'target_entity_type_id',
      'target_entity_id',
      'target_entity_id_string',
      'target_entity_revision_id',
    ]);
    $query->orderBy('target_entity_revision_id', 'ASC')
      ->condition('workspace', $workspace_id);
    if ($entity_type_id) {
      $query->condition('target_entity_type_id', $entity_type_id, '=');
      if ($entity_ids) {
        $query->condition(static::getIdField($entity_type_id), $entity_ids, 'IN');
      }
    }
    $tracked_revisions = [];
    foreach ($query->execute() as $record) {
      $target_id = $record->{static::getIdField($record->target_entity_type_id)};
      $tracked_revisions[$record->target_entity_type_id][$record->target_entity_revision_id] = $target_id;
    }
    return $tracked_revisions;
  }
  
  /**
   * {@inheritdoc}
   */
  public function getTrackedEntitiesForListing($workspace_id, ?int $pager_id = NULL, int|false $limit = 50) : array {
    $query = $this->database
      ->select(static::TABLE);
    if ($limit !== FALSE) {
      $query = $query->extend(PagerSelectExtender::class)
        ->limit($limit);
      if ($pager_id) {
        $query->element($pager_id);
      }
    }
    $query->fields(static::TABLE, [
      'target_entity_type_id',
      'target_entity_id',
      'target_entity_id_string',
      'target_entity_revision_id',
    ]);
    $query->orderBy('target_entity_type_id', 'ASC')
      ->orderBy('target_entity_revision_id', 'DESC')
      ->condition('workspace', $workspace_id);
    $tracked_revisions = [];
    foreach ($query->execute() as $record) {
      $target_id = $record->{static::getIdField($record->target_entity_type_id)};
      $tracked_revisions[$record->target_entity_type_id][$record->target_entity_revision_id] = $target_id;
    }
    return $tracked_revisions;
  }
  
  /**
   * {@inheritdoc}
   */
  public function getAssociatedRevisions($workspace_id, $entity_type_id, $entity_ids = NULL) {
    $this->loadAssociatedRevisions($workspace_id);
    if ($entity_ids) {
      return array_intersect($this->associatedRevisions[$workspace_id][$entity_type_id] ?? [], $entity_ids);
    }
    else {
      return $this->associatedRevisions[$workspace_id][$entity_type_id] ?? [];
    }
  }
  
  /**
   * {@inheritdoc}
   */
  public function getAssociatedInitialRevisions(string $workspace_id, string $entity_type_id, array $entity_ids = []) {
    $this->loadAssociatedRevisions($workspace_id);
    if ($entity_ids) {
      return array_intersect($this->associatedInitialRevisions[$workspace_id][$entity_type_id] ?? [], $entity_ids);
    }
    else {
      return $this->associatedInitialRevisions[$workspace_id][$entity_type_id] ?? [];
    }
  }
  
  /**
   * Loads associated revision IDs and populates their static caches.
   *
   * @param string $workspace_id
   *   The workspace ID to load associations for.
   */
  protected function loadAssociatedRevisions(string $workspace_id) : void {
    // Only load if the associated revisions cache has not been populated for
    // this workspace. We don't need to check the associated initial revisions
    // cache because they're always populated together.
    if (!isset($this->associatedRevisions[$workspace_id])) {
      // Initialize both caches for this workspace.
      $this->associatedRevisions[$workspace_id] = [];
      $this->associatedInitialRevisions[$workspace_id] = [];
      // Get workspace candidates for regular (non-initial) revisions.
      $workspace_tree = $this->workspaceRepository
        ->loadTree();
      if (isset($workspace_tree[$workspace_id])) {
        $workspace_candidates = array_merge([
          $workspace_id,
        ], $workspace_tree[$workspace_id]['ancestors']);
      }
      else {
        $workspace_candidates = [
          $workspace_id,
        ];
      }
      // Query all the associated revisions.
      $query = $this->database
        ->select(static::REVISION_TABLE);
      $query->fields(static::REVISION_TABLE, [
        'workspace',
        'target_entity_type_id',
        'target_entity_id',
        'target_entity_id_string',
        'target_entity_revision_id',
        'initial_revision',
      ]);
      $query->orderBy('target_entity_type_id')
        ->orderBy('target_entity_revision_id')
        ->condition('workspace', $workspace_candidates, 'IN');
      foreach ($query->execute() as $record) {
        $target_id = $record->{static::getIdField($record->target_entity_type_id)};
        // Always add to associatedRevisions for all workspace candidates.
        $this->associatedRevisions[$workspace_id][$record->target_entity_type_id][$record->target_entity_revision_id] = $target_id;
        // Only add to associatedInitialRevisions if it's an initial revision
        // for the specific workspace.
        if ($record->workspace === $workspace_id && $record->initial_revision) {
          $this->associatedInitialRevisions[$workspace_id][$record->target_entity_type_id][$record->target_entity_revision_id] = $target_id;
        }
      }
    }
  }
  
  /**
   * {@inheritdoc}
   */
  public function getEntityTrackingWorkspaceIds(RevisionableInterface $entity, bool $latest_revision = FALSE) {
    $id_field = static::getIdField($entity->getEntityTypeId());
    $query = $this->database
      ->select(static::TABLE, 'wa')
      ->fields('wa', [
      'workspace',
    ])
      ->condition('[wa].[target_entity_type_id]', $entity->getEntityTypeId())
      ->condition("[wa].[{$id_field}]", $entity->id());
    // Use a self-join to get only the workspaces in which the latest revision
    // of the entity is tracked.
    if ($latest_revision) {
      $inner_select = $this->database
        ->select(static::TABLE, 'wai')
        ->condition('[wai].[target_entity_type_id]', $entity->getEntityTypeId())
        ->condition("[wai].[{$id_field}]", $entity->id());
      $inner_select->addExpression('MAX([wai].[target_entity_revision_id])', 'max_revision_id');
      $query->join($inner_select, 'waj', '[wa].[target_entity_revision_id] = [waj].[max_revision_id]');
    }
    $result = $query->execute()
      ->fetchCol();
    // Return early if the entity is not tracked in any workspace.
    if (empty($result)) {
      return [];
    }
    // Return workspace IDs sorted in tree order.
    $tree = $this->workspaceRepository
      ->loadTree();
    return array_keys(array_intersect_key($tree, array_flip($result)));
  }
  
  /**
   * {@inheritdoc}
   */
  public function deleteAssociations($workspace_id = NULL, $entity_type_id = NULL, $entity_ids = NULL, $revision_ids = NULL) {
    if (!$workspace_id && !$entity_type_id) {
      throw new \InvalidArgumentException('A workspace ID or an entity type ID must be provided.');
    }
    try {
      $transaction = $this->database
        ->startTransaction();
      $this->doDeleteAssociations(static::TABLE, $workspace_id, $entity_type_id, $entity_ids, $revision_ids);
      $this->doDeleteAssociations(static::REVISION_TABLE, $workspace_id, $entity_type_id, $entity_ids, $revision_ids);
    } catch (\Exception $e) {
      if (isset($transaction)) {
        $transaction->rollBack();
      }
      Error::logException($this->logger, $e);
      throw $e;
    }
    $this->associatedRevisions = $this->associatedInitialRevisions = [];
  }
  
  /**
   * Executes a delete query for workspace associations.
   *
   * @param string $table
   *   The database table to delete from.
   * @param string|null $workspace_id
   *   The workspace ID to filter by, or NULL to not filter by workspace.
   * @param string|null $entity_type_id
   *   The entity type ID to filter by, or NULL to not filter by entity type.
   * @param array|null $entity_ids
   *   The entity IDs to filter by, or NULL to not filter by entity IDs.
   * @param array|null $revision_ids
   *   The revision IDs to filter by, or NULL to not filter by revision IDs.
   *
   * @throws \InvalidArgumentException
   *   When required parameters are missing.
   */
  protected function doDeleteAssociations(string $table, ?string $workspace_id = NULL, ?string $entity_type_id = NULL, ?array $entity_ids = NULL, ?array $revision_ids = NULL) : void {
    $query = $this->database
      ->delete($table);
    if ($workspace_id) {
      $query->condition('workspace', $workspace_id);
    }
    if ($entity_type_id) {
      if (!$entity_ids && !$revision_ids) {
        throw new \InvalidArgumentException('A list of entity IDs or revision IDs must be provided for an entity type.');
      }
      $query->condition('target_entity_type_id', $entity_type_id, '=');
      if ($entity_ids) {
        try {
          $query->condition(static::getIdField($entity_type_id), $entity_ids, 'IN');
        } catch (PluginNotFoundException) {
          // When an entity type is being deleted, we no longer have the ability
          // to retrieve its identifier field type, so we try both.
          $query->condition($query->orConditionGroup()
            ->condition('target_entity_id', $entity_ids, 'IN')
            ->condition('target_entity_id_string', $entity_ids, 'IN'));
        }
      }
      if ($revision_ids) {
        $query->condition('target_entity_revision_id', $revision_ids, 'IN');
      }
    }
    $query->execute();
  }
  
  /**
   * {@inheritdoc}
   */
  public function initializeWorkspace(WorkspaceInterface $workspace) {
    if ($parent_id = $workspace->parent->target_id) {
      $indexed_rows = $this->database
        ->select(static::TABLE);
      $indexed_rows->addExpression(':new_id', 'workspace', [
        ':new_id' => $workspace->id(),
      ]);
      $indexed_rows->fields(static::TABLE, [
        'target_entity_type_id',
        'target_entity_id',
        'target_entity_id_string',
        'target_entity_revision_id',
      ]);
      $indexed_rows->condition('workspace', $parent_id);
      $this->database
        ->insert(static::TABLE)
        ->from($indexed_rows)
        ->execute();
    }
    $this->associatedRevisions = $this->associatedInitialRevisions = [];
  }
  
  /**
   * {@inheritdoc}
   */
  public static function getSubscribedEvents() : array {
    // Workspace association records cleanup should happen as late as possible.
    $events[WorkspacePostPublishEvent::class][] = [
      'onPostPublish',
      -500,
    ];
    return $events;
  }
  
  /**
   * Triggers clean-up operations after a workspace is published.
   *
   * @param \Drupal\workspaces\Event\WorkspacePublishEvent $event
   *   The workspace publish event.
   */
  public function onPostPublish(WorkspacePublishEvent $event) : void {
    // Cleanup associations for the published workspace as well as its
    // descendants.
    $affected_workspaces = $this->workspaceRepository
      ->getDescendantsAndSelf($event->getWorkspace()
      ->id());
    foreach ($affected_workspaces as $workspace_id) {
      $this->deleteAssociations($workspace_id);
    }
  }
  
  /**
   * Determines the target ID field name for an entity type.
   *
   * @param string $entity_type_id
   *   The entity type ID.
   *
   * @return string
   *   The name of the workspace association target ID field.
   *
   * @internal
   */
  public static function getIdField(string $entity_type_id) : string {
    static $id_field_map = [];
    if (!isset($id_field_map[$entity_type_id])) {
      $id_field = \Drupal::entityTypeManager()->getDefinition($entity_type_id)
        ->getKey('id');
      $field_map = \Drupal::service('entity_field.manager')->getFieldMap()[$entity_type_id];
      $id_field_map[$entity_type_id] = $field_map[$id_field]['type'] !== 'integer' ? 'target_entity_id_string' : 'target_entity_id';
    }
    return $id_field_map[$entity_type_id];
  }

}

Members

Title Sort descending Modifiers Object type Summary Overriden Title
WorkspaceAssociation::$associatedInitialRevisions protected property A multidimensional array of entity IDs that were created in a workspace.
WorkspaceAssociation::$associatedRevisions protected property A multidimensional array of entity IDs that are associated to a workspace.
WorkspaceAssociation::deleteAssociations public function Deletes all the workspace association records for the given workspace. Overrides WorkspaceAssociationInterface::deleteAssociations
WorkspaceAssociation::doDeleteAssociations protected function Executes a delete query for workspace associations.
WorkspaceAssociation::getAssociatedInitialRevisions public function Retrieves all content revisions that were created in a given workspace. Overrides WorkspaceAssociationInterface::getAssociatedInitialRevisions
WorkspaceAssociation::getAssociatedRevisions public function Retrieves all content revisions tracked by a given workspace. Overrides WorkspaceAssociationInterface::getAssociatedRevisions
WorkspaceAssociation::getEntityTrackingWorkspaceIds public function Gets a list of workspace IDs in which an entity is tracked. Overrides WorkspaceAssociationInterface::getEntityTrackingWorkspaceIds
WorkspaceAssociation::getIdField public static function Determines the target ID field name for an entity type.
WorkspaceAssociation::getSubscribedEvents public static function
WorkspaceAssociation::getTrackedEntities public function Retrieves the entities tracked by a given workspace. Overrides WorkspaceAssociationInterface::getTrackedEntities
WorkspaceAssociation::getTrackedEntitiesForListing public function Retrieves a paged list of entities tracked by a given workspace. Overrides WorkspaceAssociationInterface::getTrackedEntitiesForListing
WorkspaceAssociation::initializeWorkspace public function Initializes a workspace with all the associations of its parent. Overrides WorkspaceAssociationInterface::initializeWorkspace
WorkspaceAssociation::loadAssociatedRevisions protected function Loads associated revision IDs and populates their static caches.
WorkspaceAssociation::onPostPublish public function Triggers clean-up operations after a workspace is published.
WorkspaceAssociation::REVISION_TABLE constant The table for the workspace association revision storage.
WorkspaceAssociation::TABLE constant The table for the workspace association storage.
WorkspaceAssociation::trackEntity public function Updates or creates the association for a given entity and a workspace. Overrides WorkspaceAssociationInterface::trackEntity
WorkspaceAssociation::workspaceInsert public function Responds to the creation of a new workspace entity. Overrides WorkspaceAssociationInterface::workspaceInsert
WorkspaceAssociation::__construct public function

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