You are here

public function AcceptHeaderMatcher::filter in Zircon Profile 8.0

Same name and namespace in other branches
  1. 8 core/modules/system/tests/modules/accept_header_routing_test/src/Routing/AcceptHeaderMatcher.php \Drupal\accept_header_routing_test\Routing\AcceptHeaderMatcher::filter()

Filters the route collection against a request and returns all matching routes.

Parameters

RouteCollection $collection The collection against which to match.:

Request $request A Request object against which to match.:

Return value

RouteCollection A non-empty RouteCollection of matched routes.

Throws

ResourceNotFoundException if none of the routes in $collection matches $request. This is a performance optimization to not continue the match process when a match will no longer be possible.

Overrides RouteFilterInterface::filter

File

core/modules/system/tests/modules/accept_header_routing_test/src/Routing/AcceptHeaderMatcher.php, line 24
Contains \Drupal\accept_header_routing_test\Routing\AcceptHeaderMatcher.

Class

AcceptHeaderMatcher
Filters routes based on the media type specified in the HTTP Accept headers.

Namespace

Drupal\accept_header_routing_test\Routing

Code

public function filter(RouteCollection $collection, Request $request) {

  // Generates a list of Symfony formats matching the acceptable MIME types.
  // @todo replace by proper content negotiation library.
  $acceptable_mime_types = $request
    ->getAcceptableContentTypes();
  $acceptable_formats = array_filter(array_map(array(
    $request,
    'getFormat',
  ), $acceptable_mime_types));
  $primary_format = $request
    ->getRequestFormat();
  foreach ($collection as $name => $route) {

    // _format could be a |-delimited list of supported formats.
    $supported_formats = array_filter(explode('|', $route
      ->getRequirement('_format')));
    if (empty($supported_formats)) {

      // No format restriction on the route, so it always matches. Move it to
      // the end of the collection by re-adding it.
      $collection
        ->add($name, $route);
    }
    elseif (in_array($primary_format, $supported_formats)) {

      // Perfect match, which will get a higher priority by leaving the route
      // on top of the list.
    }
    elseif (in_array('*/*', $acceptable_mime_types) || array_intersect($acceptable_formats, $supported_formats)) {

      // Move it to the end of the list.
      $collection
        ->add($name, $route);
    }
    else {

      // Remove the route if it does not match at all.
      $collection
        ->remove($name);
    }
  }
  if (count($collection)) {
    return $collection;
  }

  // We do not throw a
  // \Symfony\Component\Routing\Exception\ResourceNotFoundException here
  // because we don't want to return a 404 status code, but rather a 406.
  throw new NotAcceptableHttpException('No route found for the specified formats ' . implode(' ', $acceptable_mime_types));
}