You are here

final class RoleBasedAccessSettingsBatch in Apigee Edge 8

Utility function for API product RBAC settings save batch.

Hierarchy

Expanded class hierarchy of RoleBasedAccessSettingsBatch

File

modules/apigee_edge_apiproduct_rbac/src/RoleBasedAccessSettingsBatch.php, line 31

Namespace

Drupal\apigee_edge_apiproduct_rbac
View source
final class RoleBasedAccessSettingsBatch {

  /**
   * Batch operation callback.
   *
   * @param array $product_name_display_name_map
   *   Associative array where keys are the names (ids) of API products and
   *   values are their display names.
   * @param array $product_name_rids_map
   *   Associative array where keys are the API product names (ids) and values
   *   are array with roles ids that should have access to an API product.
   *   Rids (roles) with bypass permission should be excluded from values!
   * @param string|null $attribute_name
   *   Name of the attribute that stores the assigned roles in an API product.
   *   Default is the currently saved configuration.
   * @param string|null $original_attribute_name
   *   Name of the attribute that originally stored the role assignments.
   * @param array $context
   *   Batch context.
   *
   * @see callback_batch_operation()
   */
  public static function batchOperation(array $product_name_display_name_map, array $product_name_rids_map, string $attribute_name, string $original_attribute_name, array &$context) : void {
    if (!isset($context['sandbox']['progress'])) {
      $context['sandbox']['progress'] = 0;
      $context['sandbox']['max'] = count($product_name_display_name_map);
    }

    // Process API products by groups of 5.

    /** @var \Apigee\Edge\Api\Management\Controller\ApiProductControllerInterface $controller */
    $controller = \Drupal::service('apigee_edge.controller.api_product');
    foreach (array_slice($product_name_display_name_map, $context['sandbox']['progress'], 5) as $product_name => $product_display_name) {
      $context['message'] = t('Updating %d API Product...', [
        '%d' => $product_display_name,
      ]);
      $rids = $product_name_rids_map[$product_name] ?? [];
      try {
        $attributes = $controller
          ->getAttributes($product_name);

        // Ensure that we do not leave remnants.
        // Even if $attributeName === $originalAttributeName it is better to
        // always clear its value.
        $attributes
          ->delete($original_attribute_name);
        if ($rids) {
          $normalized_rids = [];

          // Do not save redundant (authenticated) roles if "authenticated user"
          // role is present in rids.
          if (in_array(AccountInterface::AUTHENTICATED_ROLE, $normalized_rids)) {
            $normalized_rids[] = AccountInterface::AUTHENTICATED_ROLE;
            if (in_array(AccountInterface::ANONYMOUS_ROLE, $rids)) {
              $normalized_rids[] = AccountInterface::ANONYMOUS_ROLE;
            }
          }
          else {
            $normalized_rids = $rids;
          }
          $attributes
            ->add($attribute_name, implode(APIGEE_EDGE_APIPRODUCT_RBAC_ATTRIBUTE_VALUE_DELIMITER, $normalized_rids));
        }
        $controller
          ->updateAttributes($product_name, $attributes);
        $context['results']['success'][$product_name] = Xss::filter($product_display_name);
      } catch (ApiException $e) {
        $message = Xss::filter($product_display_name);
        if ($e instanceof ApiResponseException) {
          $message = t('@product (Reason: @reason.)', [
            '@product' => $product_display_name,
            '@reason' => $e
              ->getMessage(),
          ]);
        }
        $context['results']['failed'][$product_name] = $message;
      } finally {
        $context['sandbox']['progress']++;
      }
    }

    // Inform the batch engine that we are not finished,
    // and provide an estimation of the completion level we reached.
    if ($context['sandbox']['progress'] != $context['sandbox']['max']) {
      $context['finished'] = $context['sandbox']['progress'] / $context['sandbox']['max'];
    }
  }

  /**
   * Batch finished callback.
   *
   * @see callback_batch_finished()
   */
  public static function batchFinishedCallback(bool $success, array $results, array $operations) {

    /** @var array $updated */
    $updated = $results['success'] ?? [];

    /** @var array $failed */
    $failed = $results['failed'] ?? [];
    if ($success && !empty($updated) && empty($failed)) {
      \Drupal::messenger()
        ->addStatus(t('All API product attributes have been updated successfully.'));
    }
    elseif (!empty($updated) || !empty($failed)) {
      if (!empty($updated)) {
        $items = [
          '#theme' => 'item_list',
          '#items' => $updated,
        ];
        $message = \Drupal::translation()
          ->formatPlural(count($updated), '@product API product successfully updated.', '@count API products successfully updated: @products.', [
          '@product' => reset($updated),
          '@products' => \Drupal::service('renderer')
            ->render($items),
        ]);
        \Drupal::messenger()
          ->addStatus($message);
      }
      if (!empty($failed)) {
        $items = [
          '#theme' => 'item_list',
          '#items' => $failed,
        ];
        $message = \Drupal::translation()
          ->formatPlural(count($failed), 'An API product failed failed: @product.', '@count API products could not be updated: @products.', [
          '@product' => reset($failed),
          '@products' => \Drupal::service('renderer')
            ->render($items),
        ]);
        \Drupal::messenger()
          ->addError($message);
      }
    }
    else {
      \Drupal::messenger()
        ->addStatus(t('No operation performed.'));
    }
  }

}

Members

Namesort descending Modifiers Type Description Overrides
RoleBasedAccessSettingsBatch::batchFinishedCallback public static function Batch finished callback.
RoleBasedAccessSettingsBatch::batchOperation public static function Batch operation callback.