From 090d5d0eae0f022016bb405cc138529c7c0c33c2 Mon Sep 17 00:00:00 2001 From: Camilo Ernesto Escobar Bedoya <52322-camilo.escobar@users.noreply.drupalcode.org> Date: Fri, 1 Dec 2023 13:38:01 -0700 Subject: [PATCH] Issue #3391389 by camilo.escobar, owenbush: Adding an instance to a Series: Computed fields bring down the database server --- .../ComputedField/AvailabilityCount.php | 55 +++++++++++++++- .../ComputedField/RegistrationCount.php | 54 ++++++++++++++- .../Plugin/ComputedField/WaitlistCount.php | 54 ++++++++++++++- .../src/RegistrationCreationService.php | 65 +++++++++++++++++-- 4 files changed, 213 insertions(+), 15 deletions(-) diff --git a/modules/recurring_events_registration/src/Plugin/ComputedField/AvailabilityCount.php b/modules/recurring_events_registration/src/Plugin/ComputedField/AvailabilityCount.php index a14dc458..b2634002 100644 --- a/modules/recurring_events_registration/src/Plugin/ComputedField/AvailabilityCount.php +++ b/modules/recurring_events_registration/src/Plugin/ComputedField/AvailabilityCount.php @@ -5,18 +5,69 @@ namespace Drupal\recurring_events_registration\Plugin\ComputedField; use Drupal\Core\Field\FieldItemList; use Drupal\Core\TypedData\ComputedItemListTrait; use Drupal\recurring_events_registration\Traits\RegistrationCreationServiceTrait; +use Drupal\Core\TypedData\DataDefinitionInterface; +use Drupal\Core\TypedData\TypedDataInterface; +/** + * A computed field that provides the availabilty count of an Event Instance. + */ class AvailabilityCount extends FieldItemList { use ComputedItemListTrait; use RegistrationCreationServiceTrait; + /** + * The Request stack. + * + * @var \Drupal\Core\Http\RequestStack + */ + protected $requestStack; + + /** + * {@inheritdoc} + */ + public function __construct(DataDefinitionInterface $definition, $name = NULL, TypedDataInterface $parent = NULL) { + parent::__construct($definition, $name, $parent); + // @todo Look for a better way to inject this service. + $this->requestStack = \Drupal::service('request_stack'); + } + /** * {@inheritDoc} */ protected function computeValue() { - $entity = $this->getEntity(); - $this->list[0] = $this->createItem(0, $this->getRegistrationCreationService($entity)->retrieveAvailability()); + // When saving or editing some entities, we are not interested in + // calculating the values for its computed fields. The resulting values of + // these computed fields are actually useful when getting/viewing/reading + // the entities, for example, when retrieving an entity data from a GET + // request to a REST or JSON:API endpoint. + // If the request has the 'POST' method, assign an empty value to the + // computed field and return. + $current_request = $this->requestStack->getCurrentRequest(); + $route_name = $current_request->attributes->get('_route'); + + // Exclude 'entity.eventseries.add_instance_form': When a new Event + // Instance is being created via the "Add instance" option from the Series, + // we do not want the computation to be done during the POST request. + // @see https://www.drupal.org/project/recurring_events/issues/3391389 + $excluded_routes = [ + 'entity.eventseries.add_instance_form', + ]; + + if ($current_request->getMethod() == 'POST' && in_array($route_name, $excluded_routes)) { + $this->list[0] = $this->createItem(0, 0); + return; + } + + /* + * The ComputedItemListTrait only calls this once on the same instance; from + * then on, the value is automatically cached in $this->items, for use by + * methods like getValue(). + */ + if (!isset($this->list[0])) { + $entity = $this->getEntity(); + $this->list[0] = $this->createItem(0, $this->getRegistrationCreationService($entity)->retrieveAvailability()); + } } } diff --git a/modules/recurring_events_registration/src/Plugin/ComputedField/RegistrationCount.php b/modules/recurring_events_registration/src/Plugin/ComputedField/RegistrationCount.php index 497f8e4f..c624758c 100644 --- a/modules/recurring_events_registration/src/Plugin/ComputedField/RegistrationCount.php +++ b/modules/recurring_events_registration/src/Plugin/ComputedField/RegistrationCount.php @@ -5,21 +5,69 @@ namespace Drupal\recurring_events_registration\Plugin\ComputedField; use Drupal\Core\Field\FieldItemList; use Drupal\Core\TypedData\ComputedItemListTrait; use Drupal\recurring_events_registration\Traits\RegistrationCreationServiceTrait; +use Drupal\Core\TypedData\DataDefinitionInterface; +use Drupal\Core\TypedData\TypedDataInterface; /** - * Class RegistrationCount. + * A computed field that provides the registration count of an Event Instance. */ class RegistrationCount extends FieldItemList { use ComputedItemListTrait; use RegistrationCreationServiceTrait; + /** + * The Request stack. + * + * @var \Drupal\Core\Http\RequestStack + */ + protected $requestStack; + + /** + * {@inheritdoc} + */ + public function __construct(DataDefinitionInterface $definition, $name = NULL, TypedDataInterface $parent = NULL) { + parent::__construct($definition, $name, $parent); + // @todo Look for a better way to inject this service. + $this->requestStack = \Drupal::service('request_stack'); + } + /** * {@inheritDoc} */ protected function computeValue() { - $entity = $this->getEntity(); - $this->list[0] = $this->createItem(0, count($this->getRegistrationCreationService($entity)->retrieveRegisteredParties(TRUE, FALSE))); + // When saving or editing some entities, we are not interested in + // calculating the values for its computed fields. The resulting values of + // these computed fields are actually useful when getting/viewing/reading + // the entities, for example, when retrieving an entity data from a GET + // request to a REST or JSON:API endpoint. + // If the request has the 'POST' method, assign an empty value to the + // computed field and return. + $current_request = $this->requestStack->getCurrentRequest(); + $route_name = $current_request->attributes->get('_route'); + + // Exclude 'entity.eventseries.add_instance_form': When a new Event + // Instance is being created via the "Add instance" option from the Series, + // we do not want the computation to be done during the POST request. + // @see https://www.drupal.org/project/recurring_events/issues/3391389 + $excluded_routes = [ + 'entity.eventseries.add_instance_form', + ]; + + if ($current_request->getMethod() == 'POST' && in_array($route_name, $excluded_routes)) { + $this->list[0] = $this->createItem(0, 0); + return; + } + + /* + * The ComputedItemListTrait only calls this once on the same instance; from + * then on, the value is automatically cached in $this->items, for use by + * methods like getValue(). + */ + if (!isset($this->list[0])) { + $entity = $this->getEntity(); + $this->list[0] = $this->createItem(0, $this->getRegistrationCreationService($entity)->retrieveRegisteredPartiesCount(TRUE, FALSE)); + } } } diff --git a/modules/recurring_events_registration/src/Plugin/ComputedField/WaitlistCount.php b/modules/recurring_events_registration/src/Plugin/ComputedField/WaitlistCount.php index 5651ffe1..45737d82 100644 --- a/modules/recurring_events_registration/src/Plugin/ComputedField/WaitlistCount.php +++ b/modules/recurring_events_registration/src/Plugin/ComputedField/WaitlistCount.php @@ -5,21 +5,69 @@ namespace Drupal\recurring_events_registration\Plugin\ComputedField; use Drupal\Core\Field\FieldItemList; use Drupal\Core\TypedData\ComputedItemListTrait; use Drupal\recurring_events_registration\Traits\RegistrationCreationServiceTrait; +use Drupal\Core\TypedData\DataDefinitionInterface; +use Drupal\Core\TypedData\TypedDataInterface; /** - * Class WaitlistCount. + * A computed field that provides the waitlist count of an Event Instance. */ class WaitlistCount extends FieldItemList { use ComputedItemListTrait; use RegistrationCreationServiceTrait; + /** + * The Request stack. + * + * @var \Drupal\Core\Http\RequestStack + */ + protected $requestStack; + + /** + * {@inheritdoc} + */ + public function __construct(DataDefinitionInterface $definition, $name = NULL, TypedDataInterface $parent = NULL) { + parent::__construct($definition, $name, $parent); + // @todo Look for a better way to inject this service. + $this->requestStack = \Drupal::service('request_stack'); + } + /** * {@inheritDoc} */ protected function computeValue() { - $entity = $this->getEntity(); - $this->list[0] = $this->createItem(0, count($this->getRegistrationCreationService($entity)->retrieveRegisteredParties(FALSE, TRUE))); + // When saving or editing some entities, we are not interested in + // calculating the values for its computed fields. The resulting values of + // these computed fields are actually useful when getting/viewing/reading + // the entities, for example, when retrieving an entity data from a GET + // request to a REST or JSON:API endpoint. + // If the request has the 'POST' method, assign an empty value to the + // computed field and return. + $current_request = $this->requestStack->getCurrentRequest(); + $route_name = \Drupal::routeMatch()->getRouteName(); + + // Exclude 'entity.eventseries.add_instance_form': When a new Event + // Instance is being created via the "Add instance" option from the Series, + // we do not want the computation to be done during the POST request. + // @see https://www.drupal.org/project/recurring_events/issues/3391389 + $excluded_routes = [ + 'entity.eventseries.add_instance_form', + ]; + + if ($current_request->getMethod() == 'POST' && in_array($route_name, $excluded_routes)) { + $this->list[0] = $this->createItem(0, 0); + return; + } + + /* + * The ComputedItemListTrait only calls this once on the same instance; from + * then on, the value is automatically cached in $this->items, for use by + * methods like getValue(). + */ + if (!isset($this->list[0])) { + $entity = $this->getEntity(); + $this->list[0] = $this->createItem(0, $this->getRegistrationCreationService($entity)->retrieveRegisteredPartiesCount(FALSE, TRUE)); + } } } diff --git a/modules/recurring_events_registration/src/RegistrationCreationService.php b/modules/recurring_events_registration/src/RegistrationCreationService.php index 4925034d..96ffb734 100644 --- a/modules/recurring_events_registration/src/RegistrationCreationService.php +++ b/modules/recurring_events_registration/src/RegistrationCreationService.php @@ -168,7 +168,7 @@ class RegistrationCreationService { } /** - * Retreive all registered parties. + * Retrieve all registered parties. * * @param bool $include_nonwaitlisted * Whether or not to include non-waitlisted registrants. @@ -220,6 +220,56 @@ class RegistrationCreationService { return $parties; } + /** + * Retrieves the count of all registered parties. + * + * @param bool $include_nonwaitlisted + * Whether or not to include non-waitlisted registrants. + * @param bool $include_waitlisted + * Whether or not to include waitlisted registrants. + * @param int $uid + * The user ID for whom to retrieve registrants. + * + * @return int + * The count of registrants. + */ + public function retrieveRegisteredPartiesCount($include_nonwaitlisted = TRUE, $include_waitlisted = TRUE, $uid = FALSE) { + $query = $this->storage->getQuery(); + + if ($include_nonwaitlisted && !$include_waitlisted) { + $query->condition('waitlist', 0); + } + elseif (!$include_nonwaitlisted && $include_waitlisted) { + $query->condition('waitlist', 1); + } + + if (!$include_waitlisted) { + $query->condition('waitlist', 0); + } + + if ($uid) { + $query->condition('user_id', $uid); + } + + switch ($this->getRegistrationType()) { + case 'series': + if (!empty($this->eventSeries->id())) { + $query->condition('eventseries_id', $this->eventSeries->id()); + } + break; + + case 'instance': + if (!empty($this->eventInstance->id())) { + $query->condition('eventinstance_id', $this->eventInstance->id()); + } + break; + } + + $result = $query->count()->execute(); + + return $result; + } + /** * Retreive all registered parties for a series. * @@ -302,14 +352,15 @@ class RegistrationCreationService { */ public function retrieveAvailability() { $availability = 0; - $parties = $this->retrieveRegisteredParties(TRUE, FALSE); + + $parties_count = $this->retrieveRegisteredPartiesCount(TRUE, FALSE); $capacity = $this->eventSeries->event_registration->capacity; if (empty($capacity)) { // Set capacity to unlimited if no capacity is specified. return -1; } - $availability = $capacity - count($parties); + $availability = $capacity - $parties_count; if ($availability < 0) { $availability = 0; } @@ -828,14 +879,15 @@ class RegistrationCreationService { * An array of roles that are allowed to register for this event. */ public function registrationPermittedRoles() { - // Remove extra spaces from the list of roles + // Remove extra spaces from the list of roles. $permitted_roles_string = str_replace(' ', '', $this->eventSeries->event_registration->permitted_roles); - // Convert the string into an array of roles + // Convert the string into an array of roles. $permitted_roles = []; if (!empty($permitted_roles_string)) { - if (strpos($permitted_roles_string, ',')) + if (strpos($permitted_roles_string, ',')) { $permitted_roles = explode(',', $permitted_roles_string); + } else { $permitted_roles[] = $permitted_roles_string; } @@ -843,5 +895,4 @@ class RegistrationCreationService { return $permitted_roles; } - } -- GitLab