Files
Phraseanet/lib/Alchemy/Phrasea/Order/Controller/ProdOrderController.php
2016-03-22 16:14:08 +01:00

325 lines
11 KiB
PHP

<?php
/**
* This file is part of Phraseanet
*
* (c) 2005-2016 Alchemy
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace Alchemy\Phrasea\Order\Controller;
use Alchemy\Phrasea\Application\Helper\DispatcherAware;
use Alchemy\Phrasea\Application\Helper\EntityManagerAware;
use Alchemy\Phrasea\Application\Helper\UserQueryAware;
use Alchemy\Phrasea\Controller\Controller;
use Alchemy\Phrasea\Controller\Prod\OrderControllerException;
use Alchemy\Phrasea\Controller\RecordsRequest;
use Alchemy\Phrasea\Core\Event\OrderDeliveryEvent;
use Alchemy\Phrasea\Core\Event\OrderEvent;
use Alchemy\Phrasea\Core\PhraseaEvents;
use Alchemy\Phrasea\Model\Entities\Order as OrderEntity;
use Alchemy\Phrasea\Model\Entities\OrderElement;
use Alchemy\Phrasea\Model\Entities\User;
use Alchemy\Phrasea\Model\Repositories\OrderElementRepository;
use Alchemy\Phrasea\Model\Repositories\OrderRepository;
use Alchemy\Phrasea\Order\OrderFiller;
use Alchemy\Phrasea\Order\OrderValidator;
use Alchemy\Phrasea\Order\PartialOrder;
use Assert\Assertion;
use Doctrine\Common\Collections\ArrayCollection;
use Silex\Application;
use Symfony\Component\HttpFoundation\JsonResponse;
use Symfony\Component\HttpFoundation\RedirectResponse;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
class ProdOrderController extends Controller
{
use DispatcherAware;
use EntityManagerAware;
use UserQueryAware;
/**
* Create a new order
*
* @param Request $request
*
* @return RedirectResponse|JsonResponse
*/
public function createOrder(Request $request)
{
$records = RecordsRequest::fromRequest($this->app, $request, true, ['cancmd']);
try {
if ($records->isEmpty()) {
throw new OrderControllerException($this->app->trans('There is no record eligible for an order'));
}
if (null !== $deadLine = $request->request->get('deadline')) {
$deadLine = new \DateTime($deadLine);
}
$orderUsage = $request->request->get('use', '');
$order = new OrderEntity();
$order->setUser($this->getAuthenticatedUser());
$order->setDeadline($deadLine);
$order->setOrderUsage($orderUsage);
$filler = new OrderFiller($this->app['repo.collection-references'], $this->getEntityManager());
try {
$filler->assertAllRecordsHaveOrderMaster($records);
} catch (\RuntimeException $exception) {
throw new OrderControllerException($this->app->trans('There is no one to validate orders, please contact an administrator'));
}
$filler->fillOrder($order, $records);
$this->dispatch(PhraseaEvents::ORDER_CREATE, new OrderEvent($order));
$success = true;
$msg = $this->app->trans('The records have been properly ordered');
} catch (OrderControllerException $exception) {
$success = false;
$msg = $exception->getMessage();
} catch (\Exception $exception) {
$success = false;
$msg = $this->app->trans('An error occurred');
}
if ('json' === $request->getRequestFormat()) {
return $this->app->json([
'success' => $success,
'msg' => $msg,
]);
}
return $this->app->redirectPath('prod_orders', [
'success' => (int) $success,
'action' => 'send',
]);
}
/**
* Display list of orders
*
* @param Request $request
*
* @return Response
*/
public function displayOrders(Request $request)
{
$page = (int) $request->query->get('page', 1);
$offsetStart = $page - 1;
$perPage = (int) $request->query->get('per-page', 10);
$sort = $request->query->get('sort');
$baseIds = array_keys($this->getAclForUser()->get_granted_base(['order_master']));
$ordersList = $this->getOrderRepository()->listOrders($baseIds, $offsetStart, $perPage, $sort);
$total = $this->getOrderRepository()->countTotalOrders($baseIds);
return $this->render('prod/orders/order_box.html.twig', [
'page' => $page,
'perPage' => $perPage,
'total' => $total,
'previousPage' => $page < 2 ? false : ($page - 1),
'nextPage' => $page >= ceil($total / $perPage) ? false : $page + 1,
'orders' => new ArrayCollection($ordersList),
]);
}
/**
* Display a single order identified by its id
*
* @param integer $order_id
* @return Response
*/
public function displayOneOrder($order_id)
{
$order = $this->findOr404($order_id);
return $this->render('prod/orders/order_item.html.twig', [
'order' => $order,
]);
}
/**
* Send an order
*
* @param Request $request
* @param integer $order_id
* @return RedirectResponse|JsonResponse
*/
public function sendOrder(Request $request, $order_id)
{
$elementIds = $request->request->get('elements', []);
$acceptor = $this->getAuthenticatedUser();
$elements = $this->findRequestedElements($order_id, $elementIds, $acceptor);
$order = $this->findOr404($order_id);
$basket = $this->app['provider.order_basket']->provideBasketForOrderAndUser($order, $acceptor);
$orderValidator = $this->getOrderValidator();
$partialOrder = new PartialOrder($order, $elements);
$basketElements = $orderValidator->createBasketElements($partialOrder);
$orderValidator->accept($acceptor, $partialOrder);
$orderValidator->grantHD($basket->getUser(), $basketElements);
$success = false;
try {
$manager = $this->getEntityManager();
if (!empty($basketElements)) {
foreach ($basketElements as $element) {
$basket->addElement($element);
$manager->persist($element);
}
$this->dispatch(PhraseaEvents::ORDER_DELIVER, new OrderDeliveryEvent($order, $acceptor, count($basketElements)));
}
$success = true;
$manager->persist($basket);
$manager->persist($order);
$manager->flush();
} catch (\Exception $e) {
// I don't know why only basket persistence is not checked
}
if ('json' === $request->getRequestFormat()) {
return $this->app->json([
'success' => $success,
'msg' => $success
? $this->app->trans('Order has been sent')
: $this->app->trans('An error occured while sending, please retry or contact an admin if problem persists'),
'order_id' => $order_id,
]);
}
return $this->app->redirectPath('prod_orders', [
'success' => (int) $success,
'action' => 'send',
]);
}
/**
* Deny an order
*
* @param Request $request
* @param integer $order_id
* @return RedirectResponse|JsonResponse
*/
public function denyOrder(Request $request, $order_id)
{
$success = false;
$elementIds = $request->request->get('elements', []);
$acceptor = $this->getAuthenticatedUser();
$elements = $this->findRequestedElements($order_id, $elementIds, $acceptor);
$order = $this->findOr404($order_id);
$this->getOrderValidator()->deny($acceptor, new PartialOrder($order, $elements));
try {
if (!empty($elements)) {
$this->dispatch(PhraseaEvents::ORDER_DENY, new OrderDeliveryEvent($order, $acceptor, count($elements)));
}
$success = true;
$manager = $this->getEntityManager();
$manager->persist($order);
$manager->flush();
} catch (\Exception $e) {
}
if ('json' === $request->getRequestFormat()) {
return $this->app->json([
'success' => $success,
'msg' => $success
? $this->app->trans('Order has been denied')
: $this->app->trans('An error occured while denying, please retry or contact an admin if problem persists'),
'order_id' => $order_id,
]);
}
return $this->app->redirectPath('prod_orders', [
'success' => (int) $success,
'action' => 'send',
]);
}
/**
* @return OrderRepository
*/
private function getOrderRepository()
{
return $this->app['repo.orders'];
}
/**
* @return OrderElementRepository
*/
private function getOrderElementRepository()
{
return $this->app['repo.order-elements'];
}
/**
* @param int $orderId
* @return OrderEntity
*/
private function findOr404($orderId)
{
if (null === $order = $this->getOrderRepository()->find($orderId)) {
throw new NotFoundHttpException('Order not found');
}
return $order;
}
/**
* @param int $orderId
* @param array<int> $elementIds
* @param User $acceptor
* @return OrderElement[]
*/
private function findRequestedElements($orderId, $elementIds, User $acceptor)
{
try {
Assertion::isArray($elementIds);
} catch (\Exception $exception) {
throw new BadRequestHttpException('Improper request', $exception);
}
$elements = $this->getOrderElementRepository()->findBy([
'id' => $elementIds,
'order' => $orderId,
]);
if (count($elements) !== count($elementIds)) {
throw new NotFoundHttpException(sprintf('At least one requested element does not exists or does not belong to order "%s"', $orderId));
}
if (!$this->getOrderValidator()->isGrantedValidation($acceptor, $elements)) {
throw new AccessDeniedHttpException('At least one element is in a collection you have no access to.');
}
return $elements;
}
/**
* @return OrderValidator
*/
private function getOrderValidator()
{
return $this->app['validator.order'];
}
}