request-interop / interface
Interoperable server request interfaces for PHP.
Requires
- php: >=8.4
- stream-interop/interface: 1.x@dev
- uri-interop/interface: 1.x@dev
Requires (Dev)
- pds/skeleton: ^1.0
- phpstan/phpstan: ^2.0
This package is auto-updated.
Last update: 2025-03-16 16:59:09 UTC
README
This package provides a standard set of interoperable interfaces for encapsulating readable server-side request values in PHP 8.4 or later, in order to reduce the global mutable state problems that exist with PHP superglobals. It reflects and refines the common practices of over a dozen different userland projects.
The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in BCP 14 (RFC 2119, RFC 8174).
Interfaces
Request-Interop defines the following interfaces:
- Request to represent the incoming request.
- RequestUpload to represent an uploaded file.
- RequestBody to represent the raw content of the request or an uploaded file.
- RequestUrl to represent the requested URL.
- RequestFactory to create instances of the above.
It also defines a RequestTypeAliases interface with PHPStan types to aid static analysis.
Notes:
-
The interfaces define readable properties, not getter methods. PHP superglobals are presented as variables and not as functions; using properties instead of methods maintains symmetry with the language. In addition, using things like array access and null-coalesce against a property looks more usually idiomatic in PHP than with a getter method; it is the difference between
$request->query['foo'] ?? 'bar'
and$request->getQuery()['foo'] ?? 'bar'
or$request->query->get('foo', 'bar')
. -
The interfaces define property hooks for
get
but notset
. The interfaces only guarantee readability; writability is outside the scope of this package.
Request
The Request interface represents copies of the PHP superglobals (or their equivalents) and values derived from them. It defines these properties:
-
cookies_array $cookies { get; }
- Corresponds to a copy of the
$_COOKIES
superglobal array or its equivalent.
- Corresponds to a copy of the
-
files_array $files { get; }
- Corresponds to a copy of the
$_FILES
superglobal array or its equivalent.
- Corresponds to a copy of the
-
headers_array $headers { get; }
- Corresponds to an array of the request headers.
- The values SHOULD be derived from
$_SERVER
or its equivalent. - Each array key MUST be the header field name in lower-kebab-case.
-
input_array $input { get; }
- Corresponds to an array of the request body values.
- The values SHOULD be a copy
$_POST
superglobal array or its equivalent. - The values MAY be derived from a parsed or decoded representation of the request body.
-
method_string $method { get; }
- Corresponds to the request method.
- The value SHOULD be derived from
$_SERVER
or its equivalent.
-
query_array $query { get; }
- Corresponds to an array of the request query values.
- The values SHOULD be a copy of
$_GET
or its equivalent.
-
server_array $server { get; }
- Corresponds to a copy of the
$_SERVER
superglobal array or its equivalent.
- Corresponds to a copy of the
-
uploads_array $uploads { get; }
- An array of RequestUpload instances.
- The values SHOULD be derived from
$_FILES
or its equivalent. - The index structure MUST correspond to the structure in which the uploaded files were indexed; cf. README-UPLOADS.md.
-
RequestUrl $url { get; }
- Corresponds to the requested URL.
- The values SHOULD be derived from
$_SERVER
or its equivalent.
-
?RequestBody $body { get; }
- Corresponds to the raw request content.
- The encapsulated resource SHOULD be
php://input
but MAY be some other resource.
Notes:
-
The
$body
property may be null. Not all implementations require the presence of the raw request body. -
There is no requirement to keep
$query
and$url->queryParams
in sync. Though they may originate from the same source, their values might diverge from each other.
RequestUpload
The RequestUpload interface represents a single uploaded file. It defines these properties and methods:
-
string $tmpName { get; }
- Corresponds to the
'tmp_name'
key in afiles_item_array
.
- Corresponds to the
-
int $error { get; }
- Corresponds to the
'error'
key in afiles_item_array
.
- Corresponds to the
-
?string $name { get; }
- Corresponds to the
'name'
key in afiles_item_array
.
- Corresponds to the
-
?string $fullPath { get; }
- Corresponds to the
'full_path'
key in afiles_item_array
.
- Corresponds to the
-
?string $type { get; }
- Corresponds to the
'type'
key in afiles_item_array
.
- Corresponds to the
-
?int $size { get; }
- Corresponds to the
'size'
key in afiles_item_array
.
- Corresponds to the
-
?RequestBody $body { get; }
- Corresponds to the raw upload content.
- The encapsulated resource MUST be the
$tmpName
file.
-
move(string|Stringable $to) : bool
- Moves the uploaded file to another location, usually via
move_uploaded_file()
.
- Moves the uploaded file to another location, usually via
Notes:
- The
$body
property may be null. Not all implementations require the presence of the raw upload body.
RequestBody
The RequestBody interface extends Stream-Interop StringableStream to afford idempotent reading from the raw content of a Request or a RequestUpload. It defines no additional properties or methods.
Implementations MAY be advertised as readonly only if they implement the Stream-Interop ReadonlyStream interface and adhere to its constraints.
Implementations MAY be advertised as immutable only if they implement the Stream-Interop ImmutableStream interface and adhere to its constraints.
RequestUrl
The RequestUrl interface extends Uri-Interop StringableComponents to afford reading the requested URL component values. It defines no additional properties or methods.
Implementations MUST validate that the scheme component and the host component are present and non-blank; when blank or not present, implementations MUST throw LogicException (or an extension thereof).
Notes:
- The interface is for a URL, not a URI. This is because the scheme and host URI components must be present and non-blank (i.e. a non-empty string of something other than whitespace characters). Cf. The Real Difference Between a URL and a URI: "A URL is a more specific version of a URI, so if the protocol is given or implied you should probably use URL."
RequestFactory
The RequestFactory interface defines the following methods.
-
newRequest()
returns a new Request instance:public function newRequest( ?cookies_array $cookies = null, ?files_array $files = null, ?headers_array $headers = null, ?input_array $input = null, ?method_string $method = null, ?query_array $query = null, ?server_array $server = null, ?uploads_array $uploads = null, ?RequestUrl $url = null, ?RequestBody $body = null, ) : Request;
-
newRequestUpload()
returns a new RequestUpload instance:public function newRequestUpload( string $tmpName, int $error, ?string $name = null, ?string $fullPath = null, ?string $type = null, ?int $size = null, ?RequestBody $body = null, ) : RequestUpload;
-
newRequestBody()
returns a new RequestBody instance:public function newRequestBody(string|resource $spec) : RequestBody;
-
newRequestUrl()
returns a new RequestUrl instance:public function newRequestUrl(server_array $server) : RequestUrl;
Notes:
-
All
newRequest()
arguments are optional. The arguments are intended to override whatever defaults the implementation may provide; i.e., providing no arguments SHOULD return the default implementation object, such as one created from the superglobals. -
The first two
newRequestUpload()
arguments are required. A RequestUpload MUST have at least a$tmpName
and an$error
code; all other values are optional, including the raw body content. -
The
newRequestBody()
method$spec
argument is either a string or a resource. If the$spec
is a string, implementations MUST treat it as a filename to be opened as a resource, as if byfopen()
, in whatever mode the implementation finds appropriate.
RequestTypeAliases
The RequestTypeAliases interface provides these custom PHPStan types to aid static analysis:
-
cookies_array
:array<string, string>
-
files_array
:array<array-key, files_group_array|files_item_array|files_array>
recursively up to 16 dimensions. -
files_group_array
:array{ tmp_name:string[], error:int[], name?:string[], full_path?:string[], type?:string[], size?:int[], }
-
files_item_array
:array{ tmp_name:string, error:int, name?:string, full_path?:string, type?:string, size?:int, }
-
headers_array
:array<lowercase-string, string>
-
input_array
:array<array-key, null|scalar|input_array>
recursively up to 16 dimensions. -
method_string
:uppercase-string
-
query_array
:array<array-key, string|query_array>
recursively up to 16 dimensions. -
server_array
:array<string, string>
-
uploads_array
:array<array-key, RequestUpload|uploads_array>
-- recursively up to 16 dimensions.
Notes:
-
The
files_*
types are defined from the$_FILES
structure. Cf. https://www.php.net/manual/en/features.file-upload.post-method.php. -
The
method_string
is not a Method interface. Usually the reason for a Method interface is to defineis(string $method) : bool
to make sure the comparison values use matching cases. However, the custommethod_string
type isuppercase-string
, which means static analysis should catch mismatched casing. -
The
query_array
type allows onlystring
, whileinput_array
allows anyscalar
. Thequery_array
values correspond to$_GET
, which is composed only of strings. However,input_array
corresponds to any parsed or decoded form of the request content body; different parsing strategies, such asjson_decode()
, may return various scalar types. -
The
server_array
type isarray<string, string>
and notarray<uppercase-string, string>
. Some servers add$_SERVER
keys in mixed case. For example, Microsoft IIS addsIIS_WasUrlRewritten
.
Implementations
Implementations advertised as readonly or immutable MUST be deeply readonly or immutable. With the exception of RequestBody implementations meeting the specified readonly or immutable conditions, they MUST NOT encapsulate any references, resources, mutable objects, objects or arrays encapsulating references or resources or mutable objects, and so on.
Implementations MAY define additional elements not specified in these interfaces; implementations advertised as readonly or immutable MUST make those additional elements deeply readonly or immutable.
Notes:
-
Reflection does not invalidate advertisements of readonly or immutable implementations. The ability of a consumer to use Reflection to mutate an implementation advertised as readonly or immutable does not constitute a failure to comply with Request-Interop.
-
Reference implementations may be found at https://github.com/request-interop/impl.
Q & A
What userland projects were used as reference points for Request-Interop?
The pre-PSR-7 versions of Aura, Cake, Code Igniter, Horde, Joomla, Klein, Lithium, MediaWiki, Nette, Phalcon, Symfony, Yaf, Yii, and Zend. See this project comparison for more information.
How is Request-Interop different from PSR-7 ServerRequestInterface?
In short:
-
ServerRequestInterface attempts to model the incoming HTTP request message, plus application-specific context, with shallow and inconsistent immutability requirements.
-
Request-Interop attempts to model the PHP superglobals, provides no space for application context, and requires that readonly or immutable implementations to be deeply so.
A longer answer is at README-PSR-7.md.
How is Request-Interop different from the Server-Side Request and Response Objects RFC?
This package is an intellectual descendant of that RFC, similar in form but much reduced in scope: only the superglobal-equivalent arrays, the method string, the URL, and the uploads array properties remain. (Notably, the URL array is now a RequestUrl interface.)