This commit is contained in:
Benoît Burnichon
2016-03-17 18:00:58 +01:00
parent 9398891f5c
commit f07b7b6722
5 changed files with 174 additions and 32 deletions

View File

@@ -21,14 +21,19 @@ use Alchemy\Phrasea\Model\Entities\Basket;
use Alchemy\Phrasea\Model\Entities\BasketElement; use Alchemy\Phrasea\Model\Entities\BasketElement;
use Alchemy\Phrasea\Model\Entities\Order as OrderEntity; use Alchemy\Phrasea\Model\Entities\Order as OrderEntity;
use Alchemy\Phrasea\Model\Entities\Order; use Alchemy\Phrasea\Model\Entities\Order;
use Alchemy\Phrasea\Model\Entities\OrderElement;
use Alchemy\Phrasea\Model\Repositories\OrderElementRepository;
use Alchemy\Phrasea\Model\Repositories\OrderRepository; use Alchemy\Phrasea\Model\Repositories\OrderRepository;
use Alchemy\Phrasea\Order\OrderFiller; use Alchemy\Phrasea\Order\OrderFiller;
use Assert\Assertion;
use Doctrine\Common\Collections\ArrayCollection; use Doctrine\Common\Collections\ArrayCollection;
use Silex\Application; use Silex\Application;
use Symfony\Component\HttpFoundation\JsonResponse; use Symfony\Component\HttpFoundation\JsonResponse;
use Symfony\Component\HttpFoundation\RedirectResponse; use Symfony\Component\HttpFoundation\RedirectResponse;
use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response; use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
class OrderController extends Controller class OrderController extends Controller
@@ -155,56 +160,67 @@ class OrderController extends Controller
*/ */
public function sendOrder(Request $request, $order_id) public function sendOrder(Request $request, $order_id)
{ {
$success = false; $elementIds = $request->request->get('elements', []);
try {
Assertion::isArray($elementIds);
} catch (\Exception $exception) {
throw new BadRequestHttpException('Improper request', $exception);
}
/** @var OrderElement[] $elements */
$elements = $this->getOrderElementRepository()->findBy([
'id' => $elementIds,
'order' => $order_id,
]);
if (count($elements) !== count($elementIds)) {
throw new NotFoundHttpException(sprintf('At least one requested element does not exists or does not belong to order "%s"', $order_id));
}
/** @var Order $order */ /** @var Order $order */
if (null === $order = $this->getOrderRepository()->find($order_id)) { if (null === $order = $this->getOrderRepository()->find($order_id)) {
throw new NotFoundHttpException('Order not found'); throw new NotFoundHttpException('Order not found');
} }
$manager = $this->getEntityManager(); $success = false;
$basket = $order->getBasket();
if (null === $basket) {
$basket = new Basket();
$basket->setName($this->app->trans('Commande du %date%', [
'%date%' => $order->getCreatedOn()->format('Y-m-d'),
]));
$basket->setUser($order->getUser());
$basket->setPusher($this->getAuthenticatedUser());
$manager->persist($basket); $acceptor = $this->getAuthenticatedUser();
$manager->flush();
if ($this->app['validator.order']->isGrantedValidation($acceptor, $elements)) {
throw new AccessDeniedHttpException('At least one element is in a collection you have no access to.');
} }
$basket = $this->app['provider.order_basket']->provideBasketForOrderAndUser($order, $acceptor);
$n = 0; $n = 0;
$elements = $request->request->get('elements', []);
foreach ($order->getElements() as $orderElement) {
if (in_array($orderElement->getId(), $elements)) {
$sbas_id = \phrasea::sbasFromBas($this->app, $orderElement->getBaseId());
$record = new \record_adapter($this->app, $sbas_id, $orderElement->getRecordId());
$basketElement = new BasketElement(); foreach ($elements as $element) {
$basketElement->setRecord($record); $sbas_id = \phrasea::sbasFromBas($this->app, $element->getBaseId());
$basketElement->setBasket($basket); $record = new \record_adapter($this->app, $sbas_id, $element->getRecordId());
$orderElement->setOrderMaster($this->getAuthenticatedUser()); $basketElement = new BasketElement();
$orderElement->setDeny(false); $basketElement->setRecord($record);
$orderElement->getOrder()->setBasket($basket); $basketElement->setBasket($basket);
$basket->addElement($basketElement); $element->setOrderMaster($acceptor);
$element->setDeny(false);
$element->getOrder()->setBasket($basket);
$n++; $basket->addElement($basketElement);
$this->getAclForUser($basket->getUser())->grant_hd_on($record, $this->getAuthenticatedUser(), 'order');
} $n++;
$this->getAclForUser($basket->getUser())->grant_hd_on($record, $acceptor, 'order');
} }
try { try {
if ($n > 0) { if ($n > 0) {
$order->setTodo($order->getTodo() - $n); $order->setTodo($order->getTodo() - $n);
$this->dispatch(PhraseaEvents::ORDER_DELIVER, new OrderDeliveryEvent($order, $this->getAuthenticatedUser(), $n)); $this->dispatch(PhraseaEvents::ORDER_DELIVER, new OrderDeliveryEvent($order, $acceptor, $n));
} }
$success = true; $success = true;
// There was a basketElement persist here. Seems useless as all entities are managed. $manager = $this->getEntityManager();
$manager->persist($basket); $manager->persist($basket);
$manager->persist($order); $manager->persist($order);
$manager->flush(); $manager->flush();
@@ -293,4 +309,12 @@ class OrderController extends Controller
{ {
return $this->app['repo.orders']; return $this->app['repo.orders'];
} }
/**
* @return OrderElementRepository
*/
private function getOrderElementRepository()
{
return $this->app['repo.order-elements'];
}
} }

View File

@@ -15,6 +15,8 @@ use Alchemy\Phrasea\Application as PhraseaApplication;
use Alchemy\Phrasea\Controller\LazyLocator; use Alchemy\Phrasea\Controller\LazyLocator;
use Alchemy\Phrasea\Controller\Prod\OrderController; use Alchemy\Phrasea\Controller\Prod\OrderController;
use Alchemy\Phrasea\ControllerProvider\ControllerProviderTrait; use Alchemy\Phrasea\ControllerProvider\ControllerProviderTrait;
use Alchemy\Phrasea\Order\OrderBasketProvider;
use Alchemy\Phrasea\Order\OrderValidator;
use Silex\Application; use Silex\Application;
use Silex\ControllerProviderInterface; use Silex\ControllerProviderInterface;
use Silex\ServiceProviderInterface; use Silex\ServiceProviderInterface;
@@ -25,6 +27,17 @@ class Order implements ControllerProviderInterface, ServiceProviderInterface
public function register(Application $app) public function register(Application $app)
{ {
$app['provider.order_basket'] = $app->share(function (PhraseaApplication $app) {
return new OrderBasketProvider($app['orm.em'], $app['translator']);
});
$app['validator.order'] = $app->share(function (PhraseaApplication $app) {
$orderValidator = new OrderValidator();
$orderValidator->setAclProvider($app['acl']);
return $orderValidator;
});
$app['controller.prod.order'] = $app->share(function (PhraseaApplication $app) { $app['controller.prod.order'] = $app->share(function (PhraseaApplication $app) {
return (new OrderController($app)) return (new OrderController($app))
->setDispatcher($app['dispatcher']) ->setDispatcher($app['dispatcher'])

View File

@@ -0,0 +1,55 @@
<?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;
use Alchemy\Phrasea\Model\Entities\Basket;
use Alchemy\Phrasea\Model\Entities\Order;
use Alchemy\Phrasea\Model\Entities\User;
use Doctrine\ORM\EntityManager;
use Symfony\Component\Translation\TranslatorInterface;
class OrderBasketProvider
{
/**
* @var EntityManager
*/
private $manager;
/**
* @var TranslatorInterface
*/
private $translator;
public function __construct(EntityManager $manager, TranslatorInterface $translator)
{
$this->manager = $manager;
$this->translator = $translator;
}
public function provideBasketForOrderAndUser(Order $order, User $acceptor)
{
$basket = $order->getBasket();
if (null === $basket) {
$basket = new Basket();
$basket->setName($this->translator->trans('Commande du %date%', [
'%date%' => $order->getCreatedOn()->format('Y-m-d'),
]));
$basket->setUser($order->getUser());
$basket->setPusher($acceptor);
$this->manager->persist($basket);
$this->manager->flush($basket);
}
return $basket;
}
}

View File

@@ -0,0 +1,38 @@
<?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;
use Alchemy\Phrasea\Application\Helper\AclAware;
use Alchemy\Phrasea\Model\Entities\OrderElement;
use Alchemy\Phrasea\Model\Entities\User;
class OrderValidator
{
use AclAware;
/**
* @param User $acceptor
* @param OrderElement[] $elements
* @return bool
*/
public function isGrantedValidation(User $acceptor, $elements)
{
$acceptableCollections = $this->getAclForUser($acceptor)->getOrderMasterCollectionsBaseIds();
$elementsCollections = [];
foreach ($elements as $element) {
$elementsCollections[$element->getBaseId()] = true;
}
return empty(array_diff(array_keys($elementsCollections), $acceptableCollections));
}
}

View File

@@ -1761,11 +1761,11 @@ class ACL implements cache_cacheableInterface
} }
/** /**
* Returns an array of collections on which the user is 'order master' * Returns base ids on which user is 'order master'
* *
* @return collection[] * @return array
*/ */
public function get_order_master_collections() public function getOrderMasterCollectionsBaseIds()
{ {
$sql = 'SELECT base_id FROM basusr WHERE order_master="1" AND usr_id= :usr_id'; $sql = 'SELECT base_id FROM basusr WHERE order_master="1" AND usr_id= :usr_id';
$result = $this->app->getApplicationBox() $result = $this->app->getApplicationBox()
@@ -1780,6 +1780,18 @@ class ACL implements cache_cacheableInterface
$baseIds[] = $item['base_id']; $baseIds[] = $item['base_id'];
} }
return $baseIds;
}
/**
* Returns an array of collections on which the user is 'order master'
*
* @return collection[]
*/
public function get_order_master_collections()
{
$baseIds = $this->getOrderMasterCollectionsBaseIds();
$groups = []; $groups = [];
foreach ($this->app['repo.collection-references']->findHavingOrderMaster($baseIds) as $index => $reference) { foreach ($this->app['repo.collection-references']->findHavingOrderMaster($baseIds) as $index => $reference) {