calendarRepository->find($genericDocDTO->identifiers['id'])?->getStoredObject(); } public function supportsGenericDoc(GenericDocDTO $genericDocDTO): bool { return $this->supportsKeyAndIdentifiers($genericDocDTO->key, $genericDocDTO->identifiers); } public function supportsKeyAndIdentifiers(string $key, array $identifiers): bool { return self::KEY === $key && array_key_exists('id', $identifiers); } public function buildOneGenericDoc(string $key, array $identifiers): ?GenericDocDTO { if (null === $calendarDoc = $this->calendarRepository->find($identifiers['id'])) { return null; } return new GenericDocDTO( self::KEY, $identifiers, \DateTimeImmutable::createFromInterface($calendarDoc->getCreatedAt() ?? new \DateTimeImmutable('now')), $calendarDoc->getCalendar()->getAccompanyingPeriod() ?? $calendarDoc->getCalendar()->getPerson() ); } /** * @throws MappingException */ public function buildFetchQueryForAccompanyingPeriod(AccompanyingPeriod $accompanyingPeriod, ?\DateTimeImmutable $startDate = null, ?\DateTimeImmutable $endDate = null, ?string $content = null, ?string $origin = null): FetchQueryInterface { $classMetadata = $this->em->getClassMetadata(CalendarDoc::class); $storedObjectMetadata = $this->em->getClassMetadata(StoredObject::class); $calendarMetadata = $this->em->getClassMetadata(Calendar::class); $query = new FetchQuery( self::KEY, sprintf("jsonb_build_object('id', cd.%s)", $classMetadata->getColumnName('id')), 'cd.'.$storedObjectMetadata->getColumnName('createdAt'), $classMetadata->getSchemaName().'.'.$classMetadata->getTableName().' AS cd' ); $query->addJoinClause( sprintf( 'JOIN %s doc_store ON doc_store.%s = cd.%s', $storedObjectMetadata->getSchemaName().'.'.$storedObjectMetadata->getTableName(), $storedObjectMetadata->getColumnName('id'), $classMetadata->getSingleAssociationJoinColumnName('storedObject') ) ); $query->addJoinClause( sprintf( 'JOIN %s calendar ON calendar.%s = cd.%s', $calendarMetadata->getSchemaName().'.'.$calendarMetadata->getTableName(), $calendarMetadata->getColumnName('id'), $classMetadata->getSingleAssociationJoinColumnName('calendar') ) ); $query->addWhereClause( sprintf( 'calendar.%s = ?', $calendarMetadata->getAssociationMapping('accompanyingPeriod')['joinColumns'][0]['name'] ), [$accompanyingPeriod->getId()], [Types::INTEGER] ); return $this->addWhereClausesToQuery($query, $startDate, $endDate, $content); } public function isAllowedForAccompanyingPeriod(AccompanyingPeriod $accompanyingPeriod): bool { return $this->security->isGranted(CalendarVoter::SEE, $accompanyingPeriod); } public function buildFetchQueryForPerson(Person $person, ?\DateTimeImmutable $startDate = null, ?\DateTimeImmutable $endDate = null, ?string $content = null, ?string $origin = null): FetchQueryInterface { $classMetadata = $this->em->getClassMetadata(CalendarDoc::class); $storedObjectMetadata = $this->em->getClassMetadata(StoredObject::class); $calendarMetadata = $this->em->getClassMetadata(Calendar::class); $query = new FetchQuery( self::KEY, sprintf("jsonb_build_object('id', cd.%s)", $classMetadata->getColumnName('id')), 'cd.'.$storedObjectMetadata->getColumnName('createdAt'), $classMetadata->getSchemaName().'.'.$classMetadata->getTableName().' AS cd' ); $query->addJoinClause( sprintf( 'JOIN %s doc_store ON doc_store.%s = cd.%s', $storedObjectMetadata->getSchemaName().'.'.$storedObjectMetadata->getTableName(), $storedObjectMetadata->getColumnName('id'), $classMetadata->getSingleAssociationJoinColumnName('storedObject') ) ); $query->addJoinClause( sprintf( 'JOIN %s calendar ON calendar.%s = cd.%s', $calendarMetadata->getSchemaName().'.'.$calendarMetadata->getTableName(), $calendarMetadata->getColumnName('id'), $classMetadata->getSingleAssociationJoinColumnName('calendar') ) ); // get the documents associated with accompanying periods in which person participates $or = []; $orParams = []; $orTypes = []; foreach ($person->getAccompanyingPeriodParticipations() as $participation) { if (!$this->security->isGranted(CalendarVoter::SEE, $participation->getAccompanyingPeriod())) { continue; } $or[] = sprintf( '(calendar.%s = ? AND cd.%s BETWEEN ?::date AND COALESCE(?::date, \'infinity\'::date))', $calendarMetadata->getSingleAssociationJoinColumnName('accompanyingPeriod'), $storedObjectMetadata->getColumnName('createdAt') ); $orParams = [...$orParams, $participation->getAccompanyingPeriod()->getId(), \DateTimeImmutable::createFromInterface($participation->getStartDate()), null === $participation->getEndDate() ? null : \DateTimeImmutable::createFromInterface($participation->getEndDate())]; $orTypes = [...$orTypes, Types::INTEGER, Types::DATE_IMMUTABLE, Types::DATE_IMMUTABLE]; } if ([] === $or) { $query->addWhereClause('TRUE = FALSE'); return $query; } $query->addWhereClause(implode(' OR ', $or), $orParams, $orTypes); return $this->addWhereClausesToQuery($query, $startDate, $endDate, $content); } public function isAllowedForPerson(Person $person): bool { // check that the person is allowed to see an accompanying period. If yes, the // ACL on each accompanying period will be checked when the query is build return $this->security->isGranted(AccompanyingPeriodVoter::SEE, $person); } private function addWhereClausesToQuery(FetchQuery $query, ?\DateTimeImmutable $startDate, ?\DateTimeImmutable $endDate, ?string $content): FetchQuery { $storedObjectMetadata = $this->em->getClassMetadata(StoredObject::class); if (null !== $startDate) { $query->addWhereClause( sprintf('doc_store.%s >= ?', $storedObjectMetadata->getColumnName('createdAt')), [$startDate], [Types::DATE_IMMUTABLE] ); } if (null !== $endDate) { $query->addWhereClause( sprintf('doc_store.%s < ?', $storedObjectMetadata->getColumnName('createdAt')), [$endDate], [Types::DATE_IMMUTABLE] ); } if (null !== $content) { $query->addWhereClause( sprintf('doc_store.%s ilike ?', $storedObjectMetadata->getColumnName('title')), ['%'.$content.'%'], [Types::STRING] ); } return $query; } }