accompanyingPeriodRepository = $accompanyingPeriodRepository; $this->security = $security; $this->authorizationHelper = $authorizationHelper; $this->centerResolverDispatcher = $centerResolverDispatcher; } public function buildQueryOpenedAccompanyingCourseByUser(?User $user) { $qb = $this->accompanyingPeriodRepository->createQueryBuilder('ap'); $qb->where($qb->expr()->eq('ap.user', ':user')) ->andWhere( $qb->expr()->neq('ap.step', ':draft'), $qb->expr()->orX( $qb->expr()->isNull('ap.closingDate'), $qb->expr()->gt('ap.closingDate', ':now') ) ) ->setParameter('user', $user) ->setParameter('now', new DateTime('now')) ->setParameter('draft', AccompanyingPeriod::STEP_DRAFT); return $qb; } public function countByUnDispatched(array $jobs, array $services, array $administrativeLocations): int { $qb = $this->addACLByUnDispatched($this->buildQueryUnDispatched($jobs, $services, $administrativeLocations)); $qb->select('COUNT(ap)'); return $qb->getQuery()->getSingleScalarResult(); } public function countByUserOpenedAccompanyingPeriod(?User $user): int { if (null === $user) { return 0; } return $this->buildQueryOpenedAccompanyingCourseByUser($user) ->select('COUNT(ap)') ->getQuery() ->getSingleScalarResult(); } public function findByPerson( Person $person, string $role, ?array $orderBy = [], ?int $limit = null, ?int $offset = null ): array { $qb = $this->accompanyingPeriodRepository->createQueryBuilder('ap'); $scopes = $this->authorizationHelper ->getReachableCircles( $this->security->getUser(), $role, $this->centerResolverDispatcher->resolveCenter($person) ); if (0 === count($scopes)) { return []; } $qb ->join('ap.participations', 'participation') ->where($qb->expr()->eq('participation.person', ':person')) ->andWhere( $qb->expr()->orX( 'ap.confidential = FALSE', $qb->expr()->eq('ap.user', ':user') ) ) ->andWhere( $qb->expr()->orX( $qb->expr()->neq('ap.step', ':draft'), $qb->expr()->eq('ap.createdBy', ':creator') ) ) ->setParameter('draft', AccompanyingPeriod::STEP_DRAFT) ->setParameter('person', $person) ->setParameter('user', $this->security->getUser()) ->setParameter('creator', $this->security->getUser()); // add join condition for scopes $orx = $qb->expr()->orX( $qb->expr()->eq('ap.step', ':draft') ); foreach ($scopes as $key => $scope) { $orx->add($qb->expr()->isMemberOf(':scope_' . $key, 'ap.scopes')); $qb->setParameter('scope_' . $key, $scope); } $qb->andWhere($orx); return $qb->getQuery()->getResult(); } public function findByUnDispatched(array $jobs, array $services, array $administrativeLocations, ?int $limit = null, ?int $offset = null): array { $qb = $this->addACLByUnDispatched($this->buildQueryUnDispatched($jobs, $services, $administrativeLocations)); $qb->select('ap'); if (null !== $limit) { $qb->setMaxResults($limit); } if (null !== $offset) { $qb->setFirstResult($offset); } return $qb->getQuery()->getResult(); } /** * @return array|AccompanyingPeriod[] */ public function findByUserOpenedAccompanyingPeriod(?User $user, array $orderBy = [], int $limit = 0, int $offset = 50): array { if (null === $user) { return []; } $qb = $this->buildQueryOpenedAccompanyingCourseByUser($user); $qb->setFirstResult($offset) ->setMaxResults($limit); foreach ($orderBy as $field => $direction) { $qb->addOrderBy('ap.' . $field, $direction); } return $qb->getQuery()->getResult(); } private function addACLByUnDispatched(QueryBuilder $qb): QueryBuilder { $centers = $this->authorizationHelper->getReachableCenters( $this->security->getUser(), AccompanyingPeriodVoter::SEE ); $orX = $qb->expr()->orX(); if (0 === count($centers)) { return $qb->andWhere("'FALSE' = 'TRUE'"); } foreach ($centers as $key => $center) { $scopes = $this->authorizationHelper ->getReachableCircles( $this->security->getUser(), AccompanyingPeriodVoter::SEE, $center ); $and = $qb->expr()->andX( $qb->expr()->exists('SELECT part FROM ' . AccompanyingPeriodParticipation::class . ' part ' . "JOIN part.person p WHERE part.accompanyingPeriod = ap.id AND p.center = :center_{$key}") ); $qb->setParameter('center_' . $key, $center); $orScope = $qb->expr()->orX(); foreach ($scopes as $skey => $scope) { $orScope->add( $qb->expr()->isMemberOf(':scope_' . $key . '_' . $skey, 'ap.scopes') ); $qb->setParameter('scope_' . $key . '_' . $skey, $scope); } $and->add($orScope); $orX->add($and); } return $qb->andWhere($orX); } /** * @param array|UserJob[] $jobs * @param array|Scope[] $services * @param array|Location[] $locations */ private function buildQueryUnDispatched(array $jobs, array $services, array $locations): QueryBuilder { $qb = $this->accompanyingPeriodRepository->createQueryBuilder('ap'); $qb->where( $qb->expr()->andX( $qb->expr()->isNull('ap.user'), $qb->expr()->neq('ap.step', ':draft'), $qb->expr()->neq('ap.step', ':closed') ) ) ->setParameter('draft', AccompanyingPeriod::STEP_DRAFT) ->setParameter('closed', AccompanyingPeriod::STEP_CLOSED); if (0 < count($jobs)) { $qb->andWhere($qb->expr()->in('ap.job', ':jobs')) ->setParameter('jobs', $jobs); } if (0 < count($locations)) { $qb->andWhere($qb->expr()->in('ap.administrativeLocation', ':locations')) ->setParameter('locations', $locations); } if (0 < count($services)) { $or = $qb->expr()->orX(); foreach ($services as $key => $service) { $or->add($qb->expr()->isMemberOf('ap.scopes', ':scope_' . $key)); $qb->setParameter('scope_' . $key, $service); } $qb->andWhere($or); } return $qb; } }