drupalorg-1.0.x-dev/src/Controller/WebhooksController.php

src/Controller/WebhooksController.php
<?php

namespace Drupal\drupalorg\Controller;

use Drupal\Component\Serialization\Json;
use Drupal\Core\Controller\ControllerBase;
use Drupal\Core\Queue\QueueFactory;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\HttpFoundation\JsonResponse;
use Symfony\Component\HttpFoundation\Request;

/**
 * Controller to manage system webhooks.
 */
class WebhooksController extends ControllerBase {

  /**
   * Queue factory service.
   *
   * @var \Drupal\Core\Queue\QueueFactory
   */
  protected QueueFactory $queueFactory;

  /**
   * {@inheritdoc}
   */
  public static function create(ContainerInterface $container) {
    return new static(
      $container->get('queue')
    );
  }

  /**
   * Construct method.
   *
   * @param \Drupal\Core\Queue\QueueFactory $queue_factory
   *   Queue factory.
   */
  public function __construct(QueueFactory $queue_factory) {
    $this->queueFactory = $queue_factory;
  }

  /**
   * Webhook callback.
   *
   * @param \Symfony\Component\HttpFoundation\Request $request
   *   The current request.
   *
   * @return \Symfony\Component\HttpFoundation\JsonResponse
   *   Response object.
   */
  public function projectWebhook(Request $request): JsonResponse {
    $result = [
      'status' => 'error',
      'message' => $this->t('Unknown error'),
    ];

    // Check headers and token.
    $gitlab_token = $this->config('drupalorg.gitlab_settings')->get('webhook_token');
    if (
      !empty($gitlab_token) &&
      ($request->server->get('HTTP_X_GITLAB_TOKEN') === $gitlab_token)
    ) {
      $data = Json::decode($request->getContent());
      if (empty($data['event_name'])) {
        $result['message'] = $this->t('Webhook payload does not contain the event name');
      }
      elseif ($data['event_name'] == 'repository_update') {
        $this->queueFactory->get('drupalorg_project_activity_webhook_queue_worker')->createItem([
          'event_name' => 'repository_update',
          'project_id' => $data['project_id'] ?? NULL,
        ]);
        $result['message'] = $this->t('Event was queued successfully');
        $result['status'] = 'success';
      }
      else {
        $result['message'] = $this->t('Event not covered by code');
        $result['status'] = 'warning';
      }
    }
    else {
      $result['message'] = $this->t('Invalid token');
    }

    return new JsonResponse($result);
  }

  /**
   * Webhook callback for activities related to contributions.
   *
   * @param \Symfony\Component\HttpFoundation\Request $request
   *   The current request.
   *
   * @return \Symfony\Component\HttpFoundation\JsonResponse
   *   Response object.
   */
  public function contributionActivityWebhook(Request $request): JsonResponse {
    $result = [
      'status' => 'error',
      'message' => $this->t('Unknown error'),
    ];

    // Requests might come from www.drupal.org issues or GitLab issues.
    $gitlab_token = $this->config('drupalorg.gitlab_settings')->get('webhook_token');
    $drupalorg_credit_migration_token = $this->config('drupalorg.settings')->get('credit_migration_token');
    if (
      !empty($gitlab_token) &&
      ($request->server->get('HTTP_X_GITLAB_TOKEN') === $gitlab_token)
    ) {
      $result = $this->contributionActivityWebhookFromGitLab($request);
    }
    elseif (
      !empty($drupalorg_credit_migration_token) &&
      ($request->headers->get('Drupalorg-Credit-Migration-Token') === $drupalorg_credit_migration_token)
    ) {
      // Depends on https://www.drupal.org/project/drupalorg/issues/3327584
      $result = $this->contributionActivityWebhookFromDrupal($request);
    }
    else {
      $result['message'] = $this->t('Invalid token');
    }

    return new JsonResponse($result);
  }

  /**
   * Process the webhook request from gitlab.
   *
   * @param \Symfony\Component\HttpFoundation\Request $request
   *   Request object.
   *
   * @return array
   *   Result to return.
   */
  protected function contributionActivityWebhookFromGitLab(Request $request): array {
    $result = [
      'status' => 'error',
    ];

    $data = Json::decode($request->getContent());
    if (empty($data['event_type'])) {
      $result['message'] = $this->t('Webhook payload does not contain the event type');
    }
    elseif ($data['event_type'] == 'issue') {
      $this->queueFactory->get('drupalorg_contribution_activity_webhook_queue_worker')->createItem([
        'event_name' => 'issue_update',
        'project_id' => $data['object_attributes']['project_id'] ?? NULL,
        'iid' => $data['object_attributes']['iid'] ?? NULL,
        'url' => $data['object_attributes']['url'],
        'action' => $data['object_attributes']['action'],
      ]);
      $result['message'] = $this->t('Event was queued successfully');
      $result['status'] = 'success';
    }
    elseif ($data['event_type'] == 'merge_request') {
      $this->queueFactory->get('drupalorg_contribution_activity_webhook_queue_worker')->createItem([
        'event_name' => 'merge_request_update',
        'project_id' => $data['project']['id'] ?? $data['object_attributes']['target_project_id'] ?? NULL,
        'iid' => $data['object_attributes']['iid'] ?? NULL,
        'url' => $data['object_attributes']['url'],
        'action' => $data['object_attributes']['action'],
      ]);
      $result['message'] = $this->t('Event was queued successfully');
      $result['status'] = 'success';
    }
    elseif ($data['event_type'] == 'note') {
      // For now, we only consider issues and merge requests.
      $type = '-unknown-';
      if (!empty($data['issue'])) {
        $type = 'issue';
      }
      elseif (!empty($data['merge_request'])) {
        $type = 'merge_request';
      }

      if (!empty($data[$type])) {
        $this->queueFactory->get('drupalorg_contribution_activity_webhook_queue_worker')->createItem([
          'event_name' => 'comment_update',
          'project_id' => $data['project_id'] ?? NULL,
          'iid' => $data[$type]['iid'] ?? NULL,
          'url' => $data[$type]['url'],
          'comment' => $data['object_attributes']['note'] ?? NULL,
          'comment_id' => $data['object_attributes']['id'] ?? NULL,
          'user' => $data['user'] ?? NULL,
        ]);
        $result['message'] = $this->t('Event was queued successfully');
        $result['status'] = 'success';
      }
      else {
        $result['message'] = $this->t('Comments are only processed for issues and merge requests.');
      }
    }
    elseif ($data['event_type'] == 'award') {
      $this->queueFactory->get('drupalorg_contribution_activity_webhook_queue_worker')->createItem([
        'event_name' => 'emoji',
        'issue_link' => $data['issue']['url'] ?? NULL,
        'note' => $data['note']['note'] ?? NULL,
        'emoji' => $data['object_attributes']['name'] ?? NULL,
        'user_id' => $data['user']['id'] ?? NULL,
      ]);
      $result['message'] = $this->t('Event was queued successfully');
      $result['status'] = 'success';
    }
    else {
      $result['message'] = $this->t('Event not covered by code');
      $result['status'] = 'warning';
    }

    return $result;
  }

  /**
   * Process the webhook request from Drupal.
   *
   * @param \Symfony\Component\HttpFoundation\Request $request
   *   Request object.
   *
   * @return array
   *   Result to return.
   */
  protected function contributionActivityWebhookFromDrupal(Request $request): array {
    $result = [
      'status' => 'error',
    ];

    $data = $request->request->all();
    if (empty($data['event_type'])) {
      $result['message'] = $this->t('Webhook payload does not contain the event type');
    }
    elseif ($data['event_type'] == 'issue') {
      $action = $data['action'] ?? '-';
      $this->queueFactory->get('drupalorg_contribution_activity_webhook_queue_worker')->createItem([
        'event_name' => 'drupalorg_issue_update',
        'url' => $data['url'],
        'action' => $action,
      ]);
      $result['message'] = $this->t('Event was queued successfully');
      $result['status'] = 'success';
    }
    elseif ($data['event_type'] == 'issue_migration') {
      $this->queueFactory->get('drupalorg_contribution_activity_webhook_queue_worker')->createItem([
        'event_name' => 'drupalorg_issue_migrated',
        'url' => $data['url'],
        'new_url' => $data['new_url'],
      ]);
      $result['message'] = $this->t('Event was queued successfully');
      $result['status'] = 'success';
    }
    elseif ($data['event_type'] == 'comment') {
      $this->queueFactory->get('drupalorg_contribution_activity_webhook_queue_worker')->createItem([
        'event_name' => 'drupalorg_comment_update',
        'url' => $data['url'],
      ]);
      $result['message'] = $this->t('Event was queued successfully');
      $result['status'] = 'success';
    }
    else {
      $result['message'] = $this->t('Event not covered by code');
      $result['status'] = 'warning';
    }

    return $result;
  }

  /**
   * Webhook callback for activities related to security issues.
   *
   * @param \Symfony\Component\HttpFoundation\Request $request
   *   The current request.
   *
   * @return \Symfony\Component\HttpFoundation\JsonResponse
   *   Response object.
   */
  public function securityIssueWebhook(Request $request): JsonResponse {
    $result = [
      'status' => 'error',
    ];

    $gitlab_token = $this->config('drupalorg.gitlab_settings')->get('webhook_token');
    if (
      !empty($gitlab_token) &&
      ($request->server->get('HTTP_X_GITLAB_TOKEN') === $gitlab_token)
    ) {
      $data = Json::decode($request->getContent());
      $item = [
        'object_kind' => $data['object_kind'],
        'action' => $data['object_attributes']['action'],
        'project_id' => $data['object_attributes']['project_id'] ?? NULL,
        'object_id' => $data['object_attributes']['id'] ?? NULL,
        'iid' => $data['object_attributes']['iid'] ?? $data['issue']['iid'] ?? NULL,
        'discussion_id' => $data['object_attributes']['discussion_id'] ?? NULL,
      ];
      if (!empty($data['changes']['labels']['previous'])) {
        $item['previous_labels'] = array_column($data['changes']['labels']['previous'], 'id', 'title');
      }
      if (!empty($data['changes']['labels']['current'])) {
        $item['current_labels'] = array_column($data['changes']['labels']['current'], 'id', 'title');
      }
      $this->queueFactory->get('drupalorg_security_issue_webhook_queue_worker')->createItem($item);
      $result['message'] = $this->t('Event was queued successfully');
      $result['status'] = 'success';
    }
    else {
      $result['message'] = $this->t('Invalid token');
    }

    return new JsonResponse($result);
  }

}

Главная | Обратная связь

drupal hosting | друпал хостинг | it patrol .inc