mirror of
https://gitlab.com/Chill-Projet/chill-bundles.git
synced 2025-08-02 14:07:43 +00:00
Merge branch 'ticket/list-add-opening-state' into 'ticket-app-master'
Ajout du statut opening / closed pour la liste des tickets See merge request Chill-Projet/chill-bundles!850
This commit is contained in:
commit
35844f3b73
@ -67,6 +67,19 @@ paths:
|
|||||||
type: integer
|
type: integer
|
||||||
format: integer
|
format: integer
|
||||||
minimum: 1
|
minimum: 1
|
||||||
|
- name: byCurrentState
|
||||||
|
in: query
|
||||||
|
description: the current state of the ticket
|
||||||
|
required: false
|
||||||
|
style: form
|
||||||
|
explode: false
|
||||||
|
schema:
|
||||||
|
type: array
|
||||||
|
items:
|
||||||
|
type: string
|
||||||
|
enum:
|
||||||
|
- open
|
||||||
|
- closed
|
||||||
responses:
|
responses:
|
||||||
200:
|
200:
|
||||||
description: OK
|
description: OK
|
||||||
@ -80,7 +93,7 @@ paths:
|
|||||||
results:
|
results:
|
||||||
type: array
|
type: array
|
||||||
items:
|
items:
|
||||||
$ref: '#component/schema/TicketSimple'
|
$ref: '#components/schemas/TicketSimple'
|
||||||
|
|
||||||
|
|
||||||
/1.0/ticket/motive.json:
|
/1.0/ticket/motive.json:
|
||||||
|
@ -19,7 +19,7 @@ use Twig\Environment;
|
|||||||
|
|
||||||
class EditTicketController
|
class EditTicketController
|
||||||
{
|
{
|
||||||
private string $personPerTicket;
|
private readonly string $personPerTicket;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private readonly Environment $templating,
|
private readonly Environment $templating,
|
||||||
|
@ -15,10 +15,12 @@ use Chill\MainBundle\Pagination\PaginatorFactoryInterface;
|
|||||||
use Chill\MainBundle\Serializer\Model\Collection;
|
use Chill\MainBundle\Serializer\Model\Collection;
|
||||||
use Chill\PersonBundle\Repository\PersonRepository;
|
use Chill\PersonBundle\Repository\PersonRepository;
|
||||||
use Chill\PersonBundle\Security\Authorization\PersonVoter;
|
use Chill\PersonBundle\Security\Authorization\PersonVoter;
|
||||||
|
use Chill\TicketBundle\Entity\StateEnum;
|
||||||
use Chill\TicketBundle\Repository\TicketACLAwareRepositoryInterface;
|
use Chill\TicketBundle\Repository\TicketACLAwareRepositoryInterface;
|
||||||
use Symfony\Component\HttpFoundation\JsonResponse;
|
use Symfony\Component\HttpFoundation\JsonResponse;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
|
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
|
||||||
use Symfony\Component\Routing\Annotation\Route;
|
use Symfony\Component\Routing\Annotation\Route;
|
||||||
use Symfony\Component\Security\Core\Security;
|
use Symfony\Component\Security\Core\Security;
|
||||||
use Symfony\Component\Serializer\SerializerInterface;
|
use Symfony\Component\Serializer\SerializerInterface;
|
||||||
@ -53,6 +55,17 @@ final readonly class TicketListApiController
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($request->query->has('byCurrentState')) {
|
||||||
|
try {
|
||||||
|
$params['byCurrentState'] = array_map(
|
||||||
|
fn (string $state): StateEnum => StateEnum::fromValue($state),
|
||||||
|
explode(',', $request->query->get('byCurrentState'))
|
||||||
|
);
|
||||||
|
} catch (\InvalidArgumentException $e) {
|
||||||
|
throw new BadRequestHttpException($e->getMessage(), $e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$nb = $this->ticketRepository->countTickets($params);
|
$nb = $this->ticketRepository->countTickets($params);
|
||||||
$paginator = $this->paginatorFactory->create($nb);
|
$paginator = $this->paginatorFactory->create($nb);
|
||||||
|
|
||||||
|
@ -18,4 +18,13 @@ enum StateEnum: string
|
|||||||
{
|
{
|
||||||
case OPEN = 'open';
|
case OPEN = 'open';
|
||||||
case CLOSED = 'closed';
|
case CLOSED = 'closed';
|
||||||
|
|
||||||
|
public static function fromValue(string $value): self
|
||||||
|
{
|
||||||
|
return match ($value) {
|
||||||
|
'open' => self::OPEN,
|
||||||
|
'closed' => self::CLOSED,
|
||||||
|
default => throw new \InvalidArgumentException('Invalid state value'),
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -12,6 +12,7 @@ declare(strict_types=1);
|
|||||||
namespace Chill\TicketBundle\Repository;
|
namespace Chill\TicketBundle\Repository;
|
||||||
|
|
||||||
use Chill\TicketBundle\Entity\PersonHistory;
|
use Chill\TicketBundle\Entity\PersonHistory;
|
||||||
|
use Chill\TicketBundle\Entity\StateHistory;
|
||||||
use Chill\TicketBundle\Entity\Ticket;
|
use Chill\TicketBundle\Entity\Ticket;
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Doctrine\ORM\QueryBuilder;
|
use Doctrine\ORM\QueryBuilder;
|
||||||
@ -58,6 +59,21 @@ final readonly class TicketACLAwareRepository implements TicketACLAwareRepositor
|
|||||||
$qb->andWhere($or);
|
$qb->andWhere($or);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (array_key_exists('byCurrentState', $params)) {
|
||||||
|
$qb->andWhere(
|
||||||
|
$qb->expr()->exists(sprintf(
|
||||||
|
'SELECT 1 FROM %s tp_state_%d WHERE tp_state_%d.ticket = t
|
||||||
|
AND tp_state_%d.state IN (:currentState) AND tp_state_%d.endDate IS NULL',
|
||||||
|
StateHistory::class,
|
||||||
|
++$i,
|
||||||
|
$i,
|
||||||
|
$i,
|
||||||
|
$i,
|
||||||
|
))
|
||||||
|
);
|
||||||
|
$qb->setParameter('currentState', $params['byCurrentState']);
|
||||||
|
}
|
||||||
|
|
||||||
return $qb;
|
return $qb;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -12,6 +12,7 @@ declare(strict_types=1);
|
|||||||
namespace Chill\TicketBundle\Repository;
|
namespace Chill\TicketBundle\Repository;
|
||||||
|
|
||||||
use Chill\PersonBundle\Entity\Person;
|
use Chill\PersonBundle\Entity\Person;
|
||||||
|
use Chill\TicketBundle\Entity\StateEnum;
|
||||||
use Chill\TicketBundle\Entity\Ticket;
|
use Chill\TicketBundle\Entity\Ticket;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -22,7 +23,7 @@ interface TicketACLAwareRepositoryInterface
|
|||||||
/**
|
/**
|
||||||
* Find tickets.
|
* Find tickets.
|
||||||
*
|
*
|
||||||
* @param array{byPerson?: list<Person>} $params
|
* @param array{byPerson?: list<Person>, byCurrentState: list<StateEnum>} $params
|
||||||
*
|
*
|
||||||
* @return list<Ticket>
|
* @return list<Ticket>
|
||||||
*/
|
*/
|
||||||
|
@ -19,7 +19,7 @@ use Symfony\Component\Validator\Exception\UnexpectedValueException;
|
|||||||
|
|
||||||
class SetPersonCommandConstraintValidator extends ConstraintValidator
|
class SetPersonCommandConstraintValidator extends ConstraintValidator
|
||||||
{
|
{
|
||||||
private bool $isMulti;
|
private readonly bool $isMulti;
|
||||||
|
|
||||||
public function __construct(ParameterBagInterface $parameterBag)
|
public function __construct(ParameterBagInterface $parameterBag)
|
||||||
{
|
{
|
||||||
|
@ -18,6 +18,7 @@ use Chill\PersonBundle\Entity\Person;
|
|||||||
use Chill\PersonBundle\Repository\PersonRepository;
|
use Chill\PersonBundle\Repository\PersonRepository;
|
||||||
use Chill\PersonBundle\Security\Authorization\PersonVoter;
|
use Chill\PersonBundle\Security\Authorization\PersonVoter;
|
||||||
use Chill\TicketBundle\Controller\TicketListApiController;
|
use Chill\TicketBundle\Controller\TicketListApiController;
|
||||||
|
use Chill\TicketBundle\Entity\StateEnum;
|
||||||
use Chill\TicketBundle\Entity\Ticket;
|
use Chill\TicketBundle\Entity\Ticket;
|
||||||
use Chill\TicketBundle\Repository\TicketACLAwareRepositoryInterface;
|
use Chill\TicketBundle\Repository\TicketACLAwareRepositoryInterface;
|
||||||
use PHPUnit\Framework\TestCase;
|
use PHPUnit\Framework\TestCase;
|
||||||
@ -26,6 +27,7 @@ use Prophecy\PhpUnit\ProphecyTrait;
|
|||||||
use Symfony\Component\HttpFoundation\JsonResponse;
|
use Symfony\Component\HttpFoundation\JsonResponse;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
|
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
|
||||||
use Symfony\Component\Security\Core\Security;
|
use Symfony\Component\Security\Core\Security;
|
||||||
use Symfony\Component\Serializer\SerializerInterface;
|
use Symfony\Component\Serializer\SerializerInterface;
|
||||||
|
|
||||||
@ -60,7 +62,7 @@ final class TicketListApiControllerTest extends TestCase
|
|||||||
$serializer->serialize(
|
$serializer->serialize(
|
||||||
Argument::that(fn (Collection $collection) => $collection->getItems() === $tickets),
|
Argument::that(fn (Collection $collection) => $collection->getItems() === $tickets),
|
||||||
'json',
|
'json',
|
||||||
['groups' => 'read']
|
['groups' => 'read:simple']
|
||||||
)->willReturn('{"items":[{},{}],"pagination":{}}');
|
)->willReturn('{"items":[{},{}],"pagination":{}}');
|
||||||
|
|
||||||
$personRepository = $this->prophesize(PersonRepository::class);
|
$personRepository = $this->prophesize(PersonRepository::class);
|
||||||
@ -114,7 +116,7 @@ final class TicketListApiControllerTest extends TestCase
|
|||||||
$serializer->serialize(
|
$serializer->serialize(
|
||||||
Argument::that(fn (Collection $collection) => $collection->getItems() === $tickets),
|
Argument::that(fn (Collection $collection) => $collection->getItems() === $tickets),
|
||||||
'json',
|
'json',
|
||||||
['groups' => 'read']
|
['groups' => 'read:simple']
|
||||||
)->willReturn('{"items":[{},{}],"pagination":{}}');
|
)->willReturn('{"items":[{},{}],"pagination":{}}');
|
||||||
|
|
||||||
$personRepository = $this->prophesize(PersonRepository::class);
|
$personRepository = $this->prophesize(PersonRepository::class);
|
||||||
@ -142,6 +144,93 @@ final class TicketListApiControllerTest extends TestCase
|
|||||||
$this->assertEquals('{"items":[{},{}],"pagination":{}}', $response->getContent());
|
$this->assertEquals('{"items":[{},{}],"pagination":{}}', $response->getContent());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testListTicketWithCurrentStateFilter(): void
|
||||||
|
{
|
||||||
|
// Mock dependencies
|
||||||
|
$security = $this->prophesize(Security::class);
|
||||||
|
$security->isGranted('ROLE_USER')->willReturn(true);
|
||||||
|
|
||||||
|
$ticketRepository = $this->prophesize(TicketACLAwareRepositoryInterface::class);
|
||||||
|
$tickets = [new Ticket(), new Ticket()];
|
||||||
|
$ticketRepository->countTickets(
|
||||||
|
Argument::that(fn ($params) => isset($params['byCurrentState']) && in_array(StateEnum::OPEN, $params['byCurrentState']))
|
||||||
|
)
|
||||||
|
->willReturn(2);
|
||||||
|
$ticketRepository->findTickets(
|
||||||
|
Argument::that(fn ($params) => isset($params['byCurrentState']) && in_array(StateEnum::OPEN, $params['byCurrentState'])),
|
||||||
|
0,
|
||||||
|
10
|
||||||
|
)->willReturn($tickets);
|
||||||
|
|
||||||
|
$paginator = $this->prophesize(PaginatorInterface::class);
|
||||||
|
$paginator->getCurrentPageFirstItemNumber()->willReturn(0);
|
||||||
|
$paginator->getItemsPerPage()->willReturn(10);
|
||||||
|
|
||||||
|
$paginatorFactory = $this->prophesize(PaginatorFactoryInterface::class);
|
||||||
|
$paginatorFactory->create(2)->willReturn($paginator->reveal());
|
||||||
|
|
||||||
|
$serializer = $this->prophesize(SerializerInterface::class);
|
||||||
|
$serializer->serialize(
|
||||||
|
Argument::that(fn (Collection $collection) => $collection->getItems() === $tickets),
|
||||||
|
'json',
|
||||||
|
['groups' => 'read:simple']
|
||||||
|
)->willReturn('{"items":[{},{}],"pagination":{}}');
|
||||||
|
|
||||||
|
$personRepository = $this->prophesize(PersonRepository::class);
|
||||||
|
|
||||||
|
// Create controller
|
||||||
|
$controller = new TicketListApiController(
|
||||||
|
$security->reveal(),
|
||||||
|
$ticketRepository->reveal(),
|
||||||
|
$paginatorFactory->reveal(),
|
||||||
|
$serializer->reveal(),
|
||||||
|
$personRepository->reveal()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Create request with person filter
|
||||||
|
$request = new Request(
|
||||||
|
query: ['byCurrentState' => 'open,closed']
|
||||||
|
);
|
||||||
|
|
||||||
|
// Call controller method
|
||||||
|
$response = $controller->listTicket($request);
|
||||||
|
|
||||||
|
// Assert response
|
||||||
|
$this->assertInstanceOf(JsonResponse::class, $response);
|
||||||
|
$this->assertEquals('{"items":[{},{}],"pagination":{}}', $response->getContent());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testListTicketWithCurrentStateWithInvalidFilter(): void
|
||||||
|
{
|
||||||
|
self::expectException(BadRequestHttpException::class);
|
||||||
|
|
||||||
|
// Mock dependencies
|
||||||
|
$security = $this->prophesize(Security::class);
|
||||||
|
$security->isGranted('ROLE_USER')->willReturn(true);
|
||||||
|
|
||||||
|
$ticketRepository = $this->prophesize(TicketACLAwareRepositoryInterface::class);
|
||||||
|
$paginatorFactory = $this->prophesize(PaginatorFactoryInterface::class);
|
||||||
|
$serializer = $this->prophesize(SerializerInterface::class);
|
||||||
|
$personRepository = $this->prophesize(PersonRepository::class);
|
||||||
|
|
||||||
|
// Create controller
|
||||||
|
$controller = new TicketListApiController(
|
||||||
|
$security->reveal(),
|
||||||
|
$ticketRepository->reveal(),
|
||||||
|
$paginatorFactory->reveal(),
|
||||||
|
$serializer->reveal(),
|
||||||
|
$personRepository->reveal()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Create request with person filter
|
||||||
|
$request = new Request(
|
||||||
|
query: ['byCurrentState' => 'foo']
|
||||||
|
);
|
||||||
|
|
||||||
|
// Call controller method
|
||||||
|
$response = $controller->listTicket($request);
|
||||||
|
}
|
||||||
|
|
||||||
public function testListTicketWithoutUserRole(): void
|
public function testListTicketWithoutUserRole(): void
|
||||||
{
|
{
|
||||||
// Mock dependencies
|
// Mock dependencies
|
||||||
|
@ -12,6 +12,7 @@ declare(strict_types=1);
|
|||||||
namespace Chill\TicketBundle\Tests\Repository;
|
namespace Chill\TicketBundle\Tests\Repository;
|
||||||
|
|
||||||
use Chill\PersonBundle\DataFixtures\Helper\RandomPersonHelperTrait;
|
use Chill\PersonBundle\DataFixtures\Helper\RandomPersonHelperTrait;
|
||||||
|
use Chill\TicketBundle\Entity\StateEnum;
|
||||||
use Chill\TicketBundle\Repository\TicketACLAwareRepository;
|
use Chill\TicketBundle\Repository\TicketACLAwareRepository;
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Symfony\Bundle\FrameworkBundle\Test\KernelTestCase;
|
use Symfony\Bundle\FrameworkBundle\Test\KernelTestCase;
|
||||||
@ -63,4 +64,18 @@ class TicketACLAwareRepositoryTest extends KernelTestCase
|
|||||||
|
|
||||||
self::assertIsInt($result);
|
self::assertIsInt($result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testCountTicketByCurrentStateSingleState(): void
|
||||||
|
{
|
||||||
|
$result = $this->repository->countTickets(['byCurrentState' => [StateEnum::OPEN]]);
|
||||||
|
|
||||||
|
self::assertIsInt($result);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testFindTicketByCurrentStateMultipleState(): void
|
||||||
|
{
|
||||||
|
$result = $this->repository->countTickets(['byCurrentState' => [StateEnum::OPEN, StateEnum::CLOSED]]);
|
||||||
|
|
||||||
|
self::assertIsInt($result);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user