apigee_m10n-8.x-1.7/src/Controller/PricingAndPlansController.php
src/Controller/PricingAndPlansController.php
<?php
/*
* Copyright 2018 Google Inc.
*
* This program is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License version 2 as published by the
* Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
* License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc., 51
* Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
namespace Drupal\apigee_m10n\Controller;
use Drupal\Core\Access\AccessResult;
use Drupal\Core\Cache\Cache;
use Drupal\Core\Controller\ControllerBase;
use Drupal\Core\Messenger\MessengerInterface;
use Drupal\Core\Routing\RouteMatchInterface;
use Drupal\Core\Session\AccountInterface;
use Drupal\apigee_m10n\ApigeeSdkControllerFactoryInterface;
use Drupal\apigee_m10n\Entity\ProductBundle;
use Drupal\apigee_m10n\Form\RatePlanConfigForm;
use Drupal\apigee_m10n\MonetizationInterface;
use Drupal\user\UserInterface;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\HttpFoundation\RedirectResponse;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
/**
* Generates the pricing and plans page.
*/
class PricingAndPlansController extends ControllerBase {
/**
* Service for instantiating SDK controllers.
*
* @var \Drupal\apigee_m10n\ApigeeSdkControllerFactoryInterface
*/
protected $controller_factory;
/**
* The monetization service.
*
* @var \Drupal\apigee_m10n\MonetizationInterface
*/
protected $monetization;
/**
* The Messenger service.
*
* @var \Drupal\Core\Messenger\MessengerInterface
*/
protected $messenger;
/**
* PricingAndPlansController constructor.
*
* @param \Drupal\apigee_m10n\ApigeeSdkControllerFactoryInterface $sdk_controller_factory
* The SDK controller factory.
* @param \Drupal\apigee_m10n\MonetizationInterface $monetization
* The monetization service.
* @param \Drupal\Core\Messenger\MessengerInterface $messenger
* The messenger service.
*/
public function __construct(ApigeeSdkControllerFactoryInterface $sdk_controller_factory, MonetizationInterface $monetization, MessengerInterface $messenger) {
$this->controller_factory = $sdk_controller_factory;
$this->monetization = $monetization;
$this->messenger = $messenger;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container) {
return new static(
$container->get('apigee_m10n.sdk_controller_factory'),
$container->get('apigee_m10n.monetization'),
$container->get('messenger')
);
}
/**
* Checks route access.
*
* @param \Drupal\Core\Routing\RouteMatchInterface $route_match
* The current route match.
* @param \Drupal\Core\Session\AccountInterface $account
* Run access checks for this account.
*
* @return \Drupal\Core\Access\AccessResult
* Grants access to the route if passed permissions are present.
*/
public function access(RouteMatchInterface $route_match, AccountInterface $account) {
if ($this->monetization->isOrganizationApigeeXorHybrid()) {
return AccessResult::forbidden('ApigeeX does not have product bundles.');
}
$user = $route_match->getParameter('user');
return AccessResult::allowedIf(
$account->hasPermission('view rate_plan as anyone') ||
($account->hasPermission('view rate_plan') && $account->id() === $user->id())
);
}
/**
* Redirect to the users catalog page.
*
* @return \Symfony\Component\HttpFoundation\RedirectResponse
* A redirect to the current user's product bundles page.
*/
public function myPlans(): RedirectResponse {
if ($this->monetization->isOrganizationApigeeXorHybrid()) {
return $this->redirect(
'apigee_monetization.xplans',
['user' => \Drupal::currentUser()->id()],
['absolute' => TRUE]
);
}
else {
return $this->redirect(
'apigee_monetization.plans',
['user' => \Drupal::currentUser()->id()],
['absolute' => TRUE]
);
}
}
/**
* Gets a list of available product bundles for this user.
*
* @param \Drupal\user\UserInterface $user
* The drupal user/developer.
*
* @return array
* The pager render array.
*
* @throws \Drupal\Component\Plugin\Exception\InvalidPluginDefinitionException
* @throws \Drupal\Component\Plugin\Exception\PluginNotFoundException
*/
public function catalogPage(UserInterface $user) {
// A developer email is required.
if (empty($user->getEmail())) {
throw new NotFoundHttpException((string) $this->t("The user (@uid) doesn't have an email address.", ['@uid' => $user->id()]));
}
// Add a warning on Plans and packages page for Administrators.
if ($this->currentUser()->hasPermission('administer apigee monetization')) {
$this->messenger->addWarning('This page will not list plans with a start date in the future. If you are expecting a package to be available, please verify your start date in the Apigee.');
}
$rate_plans = [];
$subscription_handler = \Drupal::entityTypeManager()->getHandler('rate_plan', 'subscription_access');
// Load rate plans for each product bundle.
foreach (ProductBundle::getAvailableProductBundlesByDeveloper($user->getEmail()) as $product_bundle) {
/** @var \Drupal\apigee_m10n\Entity\ProductBundleInterface $product_bundle */
foreach ($product_bundle->get('ratePlans') as $rate_plan) {
/** @var \Drupal\apigee_m10n\Entity\RatePlanInterface $rate_plan_entity */
$rate_plan_entity = $rate_plan->entity;
if ($subscription_handler->access($rate_plan_entity, $user) == AccessResult::allowed()) {
$rate_plans["{$product_bundle->id()}:{$rate_plan->target_id}"] = $rate_plan->entity;
}
};
}
return $this->buildPage($rate_plans);
}
/**
* Builds the page for a rate plan listing.
*
* @param \Drupal\apigee_m10n\Entity\RatePlanInterface[] $plans
* A list of rate plans.
*
* @return array
* A render array for plans.
*/
protected function buildPage($plans) {
$build = [
'#theme' => 'container__pricing_and_plans',
'#children' => [],
'#attributes' => ['class' => ['pricing-and-plans']],
'#cache' => [
'contexts' => [
'user',
'url.developer',
],
'tags' => [],
'max-age' => 300,
],
'#attached' => ['library' => ['apigee_m10n/rate_plan.entity_list']],
];
// Get the view mode from product bundle config.
$view_mode = ($view_mode = $this->config(RatePlanConfigForm::CONFIG_NAME)->get('catalog_view_mode')) ? $view_mode : 'default';
$view_builder = $this->entityTypeManager()->getViewBuilder('rate_plan');
foreach ($plans as $id => $plan) {
// @todo Add a test for render cache.
$build['#cache']['tags'] = Cache::mergeTags($build['#cache']['tags'], $plan->getCacheTags());
// Generate a build array using the view builder.
$build['#children'][$id] = $view_builder->view($plan, $view_mode);
$build['#children'][$id]['#theme_wrappers'] = ['container__pricing_and_plans__item' => ['#attributes' => ['class' => ['pricing-and-plans__item']]]];
}
return $build;
}
}
