/** * @test * @expectedException \TYPO3\Flow\Mvc\Exception\InfiniteLoopException */ public function resolveValueThrowsInfiniteLoopExceptionIfNoUniquePathSegmentCantBeFound() { $object = new \stdClass(); $this->mockPersistenceManager->expects($this->atLeastOnce())->method('getIdentifierByObject')->with($object)->will($this->returnValue('TheIdentifier')); $this->mockPersistenceManager->expects($this->atLeastOnce())->method('getObjectByIdentifier')->with('TheIdentifier')->will($this->returnValue($object)); $this->mockObjectPathMappingRepository->expects($this->once())->method('findOneByObjectTypeUriPatternAndIdentifier')->with('stdClass', 'SomeUriPattern', 'TheIdentifier')->will($this->returnValue(null)); $existingObjectPathMapping = new ObjectPathMapping(); $existingObjectPathMapping->setObjectType('stdClass'); $existingObjectPathMapping->setUriPattern('SomeUriPattern'); $existingObjectPathMapping->setPathSegment('The/Path/Segment'); $existingObjectPathMapping->setIdentifier('AnotherIdentifier'); $this->identityRoutePart->expects($this->once())->method('createPathSegmentForObject')->with($object)->will($this->returnValue('The/Path/Segment')); $this->mockObjectPathMappingRepository->expects($this->atLeastOnce())->method('findOneByObjectTypeUriPatternAndPathSegment')->will($this->returnValue($existingObjectPathMapping)); $this->identityRoutePart->setObjectType('stdClass'); $this->identityRoutePart->setUriPattern('SomeUriPattern'); $this->identityRoutePart->_call('resolveValue', $object); }
/** * Iterates through all segments in $this->uriPattern and creates * appropriate RoutePart instances. * * @return void * @throws \TYPO3\Flow\Mvc\Exception\InvalidRoutePartHandlerException * @throws \TYPO3\Flow\Mvc\Exception\InvalidUriPatternException */ public function parse() { if ($this->isParsed || $this->uriPattern === null || $this->uriPattern === '') { return; } $this->routeParts = array(); $currentRoutePartIsOptional = false; if (substr($this->uriPattern, -1) === '/') { throw new InvalidUriPatternException('The URI pattern "' . $this->uriPattern . '" of route "' . $this->getName() . '" ends with a slash, which is not allowed. You can put the trailing slash in brackets to make it optional.', 1234782997); } if ($this->uriPattern[0] === '/') { throw new InvalidUriPatternException('The URI pattern "' . $this->uriPattern . '" of route "' . $this->getName() . '" starts with a slash, which is not allowed.', 1234782983); } $matches = array(); preg_match_all(self::PATTERN_EXTRACTROUTEPARTS, $this->uriPattern, $matches, PREG_SET_ORDER); /** @var $lastRoutePart RoutePartInterface */ $lastRoutePart = null; foreach ($matches as $match) { $routePartType = empty($match['dynamic']) ? self::ROUTEPART_TYPE_STATIC : self::ROUTEPART_TYPE_DYNAMIC; $routePartName = $match['content']; if (!empty($match['optionalStart'])) { if ($lastRoutePart !== null && $lastRoutePart->isOptional()) { throw new InvalidUriPatternException('the URI pattern "' . $this->uriPattern . '" of route "' . $this->getName() . '" contains successive optional Route sections, which is not allowed.', 1234562050); } $currentRoutePartIsOptional = true; } $routePart = null; switch ($routePartType) { case self::ROUTEPART_TYPE_DYNAMIC: if ($lastRoutePart instanceof DynamicRoutePartInterface) { throw new InvalidUriPatternException('the URI pattern "' . $this->uriPattern . '" of route "' . $this->getName() . '" contains successive Dynamic Route Parts, which is not allowed.', 1218446975); } if (isset($this->routePartsConfiguration[$routePartName]['handler'])) { $routePart = $this->objectManager->get($this->routePartsConfiguration[$routePartName]['handler']); if (!$routePart instanceof DynamicRoutePartInterface) { throw new InvalidRoutePartHandlerException('routePart handlers must implement "\\TYPO3\\Flow\\Mvc\\Routing\\DynamicRoutePartInterface" in route "' . $this->getName() . '"', 1218480972); } } elseif (isset($this->routePartsConfiguration[$routePartName]['objectType'])) { $routePart = new IdentityRoutePart(); $routePart->setObjectType($this->routePartsConfiguration[$routePartName]['objectType']); if (isset($this->routePartsConfiguration[$routePartName]['uriPattern'])) { $routePart->setUriPattern($this->routePartsConfiguration[$routePartName]['uriPattern']); } } else { $routePart = new DynamicRoutePart(); } $routePartDefaultValue = ObjectAccess::getPropertyPath($this->defaults, $routePartName); if ($routePartDefaultValue !== null) { $routePart->setDefaultValue($routePartDefaultValue); } break; case self::ROUTEPART_TYPE_STATIC: $routePart = new StaticRoutePart(); if ($lastRoutePart !== null && $lastRoutePart instanceof DynamicRoutePartInterface) { /** @var DynamicRoutePartInterface $lastRoutePart */ $lastRoutePart->setSplitString($routePartName); } } $routePart->setName($routePartName); $routePart->setOptional($currentRoutePartIsOptional); $routePart->setLowerCase($this->lowerCase); if (isset($this->routePartsConfiguration[$routePartName]['options'])) { $routePart->setOptions($this->routePartsConfiguration[$routePartName]['options']); } if (isset($this->routePartsConfiguration[$routePartName]['toLowerCase'])) { $routePart->setLowerCase($this->routePartsConfiguration[$routePartName]['toLowerCase']); } $this->routeParts[] = $routePart; if (!empty($match['optionalEnd'])) { if (!$currentRoutePartIsOptional) { throw new InvalidUriPatternException('The URI pattern "' . $this->uriPattern . '" of route "' . $this->getName() . '" contains an unopened optional section.', 1234564495); } $currentRoutePartIsOptional = false; } $lastRoutePart = $routePart; } if ($currentRoutePartIsOptional) { throw new InvalidUriPatternException('The URI pattern "' . $this->uriPattern . '" of route "' . $this->getName() . '" contains an unterminated optional section.', 1234563922); } $this->isParsed = true; }