You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
When the route name is a string literal, it should be possible to analyze calls to UrlGeneratorInterface::generate to report missing parameters in the array shape of the second argument.
Routes available for url generation can be read from the compiled file in var/cache/dev/url_generating_routes.php. The format of this file is semi-internal (officially, the guarantee we make is that the constructor of CompiledUrlGenerator takes an array in the format dumped by CompiledUrlGeneratorDumper, but the format does not change often so it should be safe to use it with a functional test ensuring the format does not break for new Symfony versions).
This url_generating_routes.php is a PHP file returning an array with this format (top-level keys are route names):
array<string, array{
0: string[], // names of route variables.
1: array<string, mixed>, // route defaults
2: array<string, string>, // variable requirements
3: list<array{'text', string}|array{
0: 'variable',
1: string, // static prefix
2: string, // regex pattern for the requirement
3: string, // variable name
4?: bool, // whether it needs unicode matching
5?: bool, // whether the token is marked as important
}>, // tokens for url generation for the path
4. list, // tokens for url generation for the host (same type than for the path tokens)
5. string[], // allowed schemes (empty when no scheme restriction)
6. list<string>, // deprecation messages
}>
the parameters passed in the second argument of UrlGeneratorInterface::generate will be handled this way (I'm talking about the keys in the array here):
_fragment will be used to generate the fragment part of the URL
if the name is one of the route variables, it is used to fill that variable (this case requires values to be castable to strings, so it supports scalars and Stringable)
if the name corresponds to a route default (with the default value), it is skipped
otherwise, the name-value pair is added to the query string (this case supports having array values for parameters)
Variables can be optional in 3 ways:
when they have a default in the route definition
when they have a value provided by the routing RequestContext. By default in core, only _locale is managed there, based on the current locale, but projects might add extra ones (maybe this can be managed through a configuration setting for the extension)
_locale defaults to the default locale configured (if it does not have a default set previously) so it is always optional
I suggest that the parameters are validated this way:
if the route name (first argument of the method) is not a known string literal, do nothing
if the inferred type is not an array shape, maybe do nothing (I don't think we want to report errors for cases reading $request->attributes->get('_route_params') for instance, as that would be too annoying to force refinement in such cases)
build an unsealed array shape based on route variables, with a key for each route variable (see key 0 of the shape of url_generating_routes.php) and a type of scalar|Stringable (or whatever phpstan uses to describe the type supported by a (string) cast), with the key being optional if the variable is optional
check whether the provided parameters are assignable to that expected shape
Bonus point: the logic running those checks should be provided through a reusable service, so that it can be reused in twigstan for the analysis of the path() and url() Twig functions (which are wrappers around the UrlGeneratorInterface). It should also be reused for AbstractController::generateUrl and AbstractController::redirectToRoute.
The text was updated successfully, but these errors were encountered:
Another bonus point: once we have the infrastructure to read url_generating_routes.php, we might report a deprecation (based on integration with phpstan-deprecation-rules probably) when generating a URL for a name that is deprecated (i.e. it has a non-empty list of deprecations associated with it)
When the route name is a string literal, it should be possible to analyze calls to
UrlGeneratorInterface::generate
to report missing parameters in the array shape of the second argument.Routes available for url generation can be read from the compiled file in
var/cache/dev/url_generating_routes.php
. The format of this file is semi-internal (officially, the guarantee we make is that the constructor ofCompiledUrlGenerator
takes an array in the format dumped byCompiledUrlGeneratorDumper
, but the format does not change often so it should be safe to use it with a functional test ensuring the format does not break for new Symfony versions).This
url_generating_routes.php
is a PHP file returning an array with this format (top-level keys are route names):the parameters passed in the second argument of
UrlGeneratorInterface::generate
will be handled this way (I'm talking about the keys in the array here):_fragment
will be used to generate the fragment part of the URLVariables can be optional in 3 ways:
_locale
is managed there, based on the current locale, but projects might add extra ones (maybe this can be managed through a configuration setting for the extension)_locale
defaults to the default locale configured (if it does not have a default set previously) so it is always optionalI suggest that the parameters are validated this way:
$request->attributes->get('_route_params')
for instance, as that would be too annoying to force refinement in such cases)0
of the shape ofurl_generating_routes.php
) and a type ofscalar|Stringable
(or whatever phpstan uses to describe the type supported by a(string)
cast), with the key being optional if the variable is optionalBonus point: the logic running those checks should be provided through a reusable service, so that it can be reused in twigstan for the analysis of the
path()
andurl()
Twig functions (which are wrappers around the UrlGeneratorInterface). It should also be reused forAbstractController::generateUrl
andAbstractController::redirectToRoute
.The text was updated successfully, but these errors were encountered: