mirror of
https://gitee.com/hyperf/hyperf.git
synced 2024-12-04 20:58:13 +08:00
Upgrade the minimum php version to 8.0 for http-message.
This commit is contained in:
parent
c49e7e4474
commit
355aabfa71
17
src/contract/src/Arrayable.php
Normal file
17
src/contract/src/Arrayable.php
Normal file
@ -0,0 +1,17 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
/**
|
||||
* This file is part of Hyperf.
|
||||
*
|
||||
* @link https://www.hyperf.io
|
||||
* @document https://hyperf.wiki
|
||||
* @contact group@hyperf.io
|
||||
* @license https://github.com/hyperf/hyperf/blob/master/LICENSE
|
||||
*/
|
||||
namespace Hyperf\Contract;
|
||||
|
||||
interface Arrayable
|
||||
{
|
||||
public function toArray(): array;
|
||||
}
|
@ -23,22 +23,13 @@ trait MessageTrait
|
||||
/**
|
||||
* @var array lowercase headers
|
||||
*/
|
||||
protected $headerNames;
|
||||
protected array $headerNames = [];
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
protected $headers = [];
|
||||
protected array $headers = [];
|
||||
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
protected $protocol = '1.1';
|
||||
protected string $protocol = '1.1';
|
||||
|
||||
/**
|
||||
* @var null|StreamInterface
|
||||
*/
|
||||
protected $stream;
|
||||
protected ?StreamInterface $stream = null;
|
||||
|
||||
/**
|
||||
* Retrieves the HTTP protocol version as a string.
|
||||
@ -190,10 +181,7 @@ trait MessageTrait
|
||||
return $new;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return static
|
||||
*/
|
||||
public function withHeaders(array $headers)
|
||||
public function withHeaders(array $headers): static
|
||||
{
|
||||
$new = clone $this;
|
||||
foreach ($headers as $name => $value) {
|
||||
@ -314,7 +302,7 @@ trait MessageTrait
|
||||
* @throws \RuntimeException
|
||||
* @return array|string wanted part or all parts as array($firstName => firstPart, partname => value)
|
||||
*/
|
||||
public function getHeaderField($name, $wantedPart = '0', $firstName = '0')
|
||||
public function getHeaderField(string $name, string $wantedPart = '0', string $firstName = '0')
|
||||
{
|
||||
return Decode::splitHeaderField($this->getHeaderLine($name), $wantedPart, $firstName);
|
||||
}
|
||||
@ -338,10 +326,7 @@ trait MessageTrait
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @return static
|
||||
*/
|
||||
private function setHeaders(array $headers)
|
||||
private function setHeaders(array $headers): static
|
||||
{
|
||||
$this->headerNames = $this->headers = [];
|
||||
foreach ($headers as $header => $value) {
|
||||
@ -374,7 +359,7 @@ trait MessageTrait
|
||||
* @return string[] Trimmed header values
|
||||
* @see https://tools.ietf.org/html/rfc7230#section-3.2.4
|
||||
*/
|
||||
private function trimHeaderValues(array $values)
|
||||
private function trimHeaderValues(array $values): array
|
||||
{
|
||||
return array_map(function ($value) {
|
||||
return trim((string) $value, " \t");
|
||||
|
@ -21,27 +21,13 @@ class Request implements RequestInterface
|
||||
{
|
||||
use MessageTrait;
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
protected $server = [];
|
||||
protected array $server = [];
|
||||
|
||||
/**
|
||||
* @var UriInterface
|
||||
*/
|
||||
protected $uri;
|
||||
protected UriInterface $uri;
|
||||
|
||||
/**
|
||||
* Http Method.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $method;
|
||||
protected string $method;
|
||||
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
protected $requestTarget;
|
||||
protected ?string $requestTarget = null;
|
||||
|
||||
/**
|
||||
* @param string $method HTTP method
|
||||
@ -52,9 +38,9 @@ class Request implements RequestInterface
|
||||
*/
|
||||
public function __construct(
|
||||
string $method,
|
||||
$uri,
|
||||
string|UriInterface $uri,
|
||||
array $headers = [],
|
||||
$body = null,
|
||||
mixed $body = null,
|
||||
string $version = '1.1'
|
||||
) {
|
||||
if (! $uri instanceof UriInterface) {
|
||||
@ -185,7 +171,7 @@ class Request implements RequestInterface
|
||||
* default if the URI contains a host component. If the URI does not
|
||||
* contain a host component, any pre-existing Host header MUST be carried
|
||||
* over to the returned request.
|
||||
* You can opt-in to preserving the original state of the Host header by
|
||||
* You can opt in to preserving the original state of the Host header by
|
||||
* setting `$preserveHost` to `true`. When `$preserveHost` is set to
|
||||
* `true`, this method interacts with the Host header in the following ways:
|
||||
* - If the Host header is missing or empty, and the new URI contains
|
||||
@ -226,7 +212,7 @@ class Request implements RequestInterface
|
||||
*
|
||||
* @see http://tools.ietf.org/html/rfc7230#section-5.4
|
||||
*/
|
||||
private function updateHostFromUri()
|
||||
private function updateHostFromUri(): void
|
||||
{
|
||||
$host = $this->uri->getHost();
|
||||
|
||||
|
@ -17,23 +17,17 @@ class Response implements ResponseInterface
|
||||
{
|
||||
use MessageTrait;
|
||||
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
protected $reasonPhrase = '';
|
||||
protected string $reasonPhrase = '';
|
||||
|
||||
protected int $statusCode = 200;
|
||||
|
||||
protected string $charset = 'utf-8';
|
||||
|
||||
/**
|
||||
* @var int
|
||||
* Map of standard HTTP status code/reason phrases.
|
||||
* @var array<int, string>
|
||||
*/
|
||||
protected $statusCode = 200;
|
||||
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
protected $charset = 'utf-8';
|
||||
|
||||
/** @var array Map of standard HTTP status code/reason phrases */
|
||||
private static $phrases
|
||||
private static array $phrases
|
||||
= [
|
||||
100 => 'Continue',
|
||||
101 => 'Switching Protocols',
|
||||
@ -95,10 +89,7 @@ class Response implements ResponseInterface
|
||||
511 => 'Network Authentication Required',
|
||||
];
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private $attributes = [];
|
||||
private array $attributes = [];
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
@ -115,7 +106,7 @@ class Response implements ResponseInterface
|
||||
*
|
||||
* @return array attributes derived from the request
|
||||
*/
|
||||
public function getAttributes()
|
||||
public function getAttributes(): array
|
||||
{
|
||||
return $this->attributes;
|
||||
}
|
||||
@ -332,7 +323,7 @@ class Response implements ResponseInterface
|
||||
303,
|
||||
307,
|
||||
308,
|
||||
]) && ($location === null ?: $location == $this->getHeaderLine('Location'));
|
||||
]) && ($location === null || $location == $this->getHeaderLine('Location'));
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -11,7 +11,9 @@ declare(strict_types=1);
|
||||
*/
|
||||
namespace Hyperf\HttpMessage\Cookie;
|
||||
|
||||
class Cookie
|
||||
use Stringable;
|
||||
|
||||
class Cookie implements Stringable
|
||||
{
|
||||
public const SAMESITE_LAX = 'lax';
|
||||
|
||||
@ -19,23 +21,11 @@ class Cookie
|
||||
|
||||
public const SAMESITE_NONE = 'none';
|
||||
|
||||
protected $name;
|
||||
protected int $expire;
|
||||
|
||||
protected $value;
|
||||
protected string $path;
|
||||
|
||||
protected $domain;
|
||||
|
||||
protected $expire;
|
||||
|
||||
protected $path;
|
||||
|
||||
protected $secure;
|
||||
|
||||
protected $httpOnly;
|
||||
|
||||
private $raw;
|
||||
|
||||
private $sameSite;
|
||||
private ?string $sameSite = null;
|
||||
|
||||
/**
|
||||
* @param string $name The name of the cookie
|
||||
@ -51,14 +41,14 @@ class Cookie
|
||||
* @throws \InvalidArgumentException
|
||||
*/
|
||||
public function __construct(
|
||||
string $name,
|
||||
string $value = '',
|
||||
protected string $name,
|
||||
protected string $value = '',
|
||||
$expire = 0,
|
||||
string $path = '/',
|
||||
string $domain = '',
|
||||
bool $secure = false,
|
||||
bool $httpOnly = true,
|
||||
bool $raw = false,
|
||||
protected string $domain = '',
|
||||
protected bool $secure = false,
|
||||
protected bool $httpOnly = true,
|
||||
protected bool $raw = false,
|
||||
?string $sameSite = null
|
||||
) {
|
||||
// from PHP source code
|
||||
@ -81,14 +71,8 @@ class Cookie
|
||||
}
|
||||
}
|
||||
|
||||
$this->name = $name;
|
||||
$this->value = $value;
|
||||
$this->domain = $domain;
|
||||
$this->expire = 0 < $expire ? (int) $expire : 0;
|
||||
$this->path = empty($path) ? '/' : $path;
|
||||
$this->secure = (bool) $secure;
|
||||
$this->httpOnly = (bool) $httpOnly;
|
||||
$this->raw = (bool) $raw;
|
||||
|
||||
if ($sameSite !== null) {
|
||||
$sameSite = strtolower($sameSite);
|
||||
@ -110,7 +94,7 @@ class Cookie
|
||||
{
|
||||
$str = ($this->isRaw() ? $this->getName() : urlencode($this->getName())) . '=';
|
||||
|
||||
if ((string) $this->getValue() === '') {
|
||||
if ($this->getValue() === '') {
|
||||
$str .= 'deleted; expires=' . gmdate('D, d-M-Y H:i:s T', time() - 31536001) . '; max-age=-31536001';
|
||||
} else {
|
||||
$str .= $this->isRaw() ? $this->getValue() : rawurlencode($this->getValue());
|
||||
@ -148,11 +132,8 @@ class Cookie
|
||||
|
||||
/**
|
||||
* Creates cookie from raw header string.
|
||||
*
|
||||
* @param string $cookie
|
||||
* @param bool $decode
|
||||
*/
|
||||
public static function fromString($cookie, $decode = false)
|
||||
public static function fromString(string $cookie, bool $decode = false): self
|
||||
{
|
||||
$data = [
|
||||
'expires' => 0,
|
||||
@ -164,7 +145,7 @@ class Cookie
|
||||
'samesite' => null,
|
||||
];
|
||||
foreach (explode(';', $cookie) as $part) {
|
||||
if (strpos($part, '=') === false) {
|
||||
if (! str_contains($part, '=')) {
|
||||
$key = trim($part);
|
||||
$value = true;
|
||||
} else {
|
||||
@ -205,110 +186,88 @@ class Cookie
|
||||
|
||||
/**
|
||||
* Gets the name of the cookie.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getName()
|
||||
public function getName(): string
|
||||
{
|
||||
return $this->name;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the value of the cookie.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getValue()
|
||||
public function getValue(): string
|
||||
{
|
||||
return $this->value;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the domain that the cookie is available to.
|
||||
*
|
||||
* @return null|string
|
||||
*/
|
||||
public function getDomain()
|
||||
public function getDomain(): string
|
||||
{
|
||||
return $this->domain;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the time the cookie expires.
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getExpiresTime()
|
||||
public function getExpiresTime(): int
|
||||
{
|
||||
return $this->expire;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the max-age attribute.
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getMaxAge()
|
||||
public function getMaxAge(): int
|
||||
{
|
||||
return $this->expire !== 0 ? $this->expire - time() : 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the path on the server in which the cookie will be available on.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getPath()
|
||||
public function getPath(): string
|
||||
{
|
||||
return $this->path;
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks whether the cookie should only be transmitted over a secure HTTPS connection from the client.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function isSecure()
|
||||
public function isSecure(): bool
|
||||
{
|
||||
return $this->secure;
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks whether the cookie will be made accessible only through the HTTP protocol.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function isHttpOnly()
|
||||
public function isHttpOnly(): bool
|
||||
{
|
||||
return $this->httpOnly;
|
||||
}
|
||||
|
||||
/**
|
||||
* Whether this cookie is about to be cleared.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function isCleared()
|
||||
public function isCleared(): bool
|
||||
{
|
||||
return $this->expire < time();
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if the cookie value should be sent with no url encoding.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function isRaw()
|
||||
public function isRaw(): bool
|
||||
{
|
||||
return $this->raw;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the SameSite attribute.
|
||||
*
|
||||
* @return null|string
|
||||
*/
|
||||
public function getSameSite()
|
||||
public function getSameSite(): ?string
|
||||
{
|
||||
return $this->sameSite;
|
||||
}
|
||||
|
@ -19,11 +19,11 @@ use Psr\Http\Message\ResponseInterface;
|
||||
*/
|
||||
class CookieJar implements CookieJarInterface
|
||||
{
|
||||
/** @var SetCookie[] Loaded cookie data */
|
||||
private $cookies = [];
|
||||
|
||||
/** @var bool */
|
||||
private $strictMode;
|
||||
/**
|
||||
* Loaded cookie data.
|
||||
* @var SetCookie[]
|
||||
*/
|
||||
private array $cookies = [];
|
||||
|
||||
/**
|
||||
* @param bool $strictMode set to true to throw exceptions when invalid
|
||||
@ -32,10 +32,8 @@ class CookieJar implements CookieJarInterface
|
||||
* arrays that can be used with the SetCookie
|
||||
* constructor
|
||||
*/
|
||||
public function __construct($strictMode = false, $cookieArray = [])
|
||||
public function __construct(private $strictMode = false, $cookieArray = [])
|
||||
{
|
||||
$this->strictMode = $strictMode;
|
||||
|
||||
foreach ($cookieArray as $cookie) {
|
||||
if (! ($cookie instanceof SetCookie)) {
|
||||
$cookie = new SetCookie($cookie);
|
||||
@ -49,10 +47,8 @@ class CookieJar implements CookieJarInterface
|
||||
*
|
||||
* @param array $cookies Cookies to create the jar from
|
||||
* @param string $domain Domain to set the cookies to
|
||||
*
|
||||
* @return self
|
||||
*/
|
||||
public static function fromArray(array $cookies, $domain)
|
||||
public static function fromArray(array $cookies, string $domain): self
|
||||
{
|
||||
$cookieJar = new self();
|
||||
foreach ($cookies as $name => $value) {
|
||||
@ -73,12 +69,9 @@ class CookieJar implements CookieJarInterface
|
||||
*
|
||||
* @param SetCookie $cookie being evaluated
|
||||
* @param bool $allowSessionCookies If we should persist session cookies
|
||||
* @return bool
|
||||
*/
|
||||
public static function shouldPersist(
|
||||
SetCookie $cookie,
|
||||
$allowSessionCookies = false
|
||||
) {
|
||||
public static function shouldPersist(SetCookie $cookie, bool $allowSessionCookies = false): bool
|
||||
{
|
||||
if ($cookie->getExpires() || $allowSessionCookies) {
|
||||
if (! $cookie->getDiscard()) {
|
||||
return true;
|
||||
@ -94,7 +87,7 @@ class CookieJar implements CookieJarInterface
|
||||
* @param string $name cookie name to search for
|
||||
* @return null|SetCookie cookie that was found or null if not found
|
||||
*/
|
||||
public function getCookieByName(string $name)
|
||||
public function getCookieByName(string $name): ?SetCookie
|
||||
{
|
||||
foreach ($this->cookies as $cookie) {
|
||||
if ($cookie->getName() !== null && strcasecmp($cookie->getName(), $name) === 0) {
|
||||
@ -104,7 +97,7 @@ class CookieJar implements CookieJarInterface
|
||||
return null;
|
||||
}
|
||||
|
||||
public function toArray()
|
||||
public function toArray(): array
|
||||
{
|
||||
return array_map(function (SetCookie $cookie) {
|
||||
return $cookie->toArray();
|
||||
@ -235,7 +228,7 @@ class CookieJar implements CookieJarInterface
|
||||
if (! $sc->getDomain()) {
|
||||
$sc->setDomain($request->getUri()->getHost());
|
||||
}
|
||||
if (strpos($sc->getPath(), '/') !== 0) {
|
||||
if (! str_starts_with($sc->getPath(), '/')) {
|
||||
$sc->setPath($this->getCookiePathFromRequest($request));
|
||||
}
|
||||
$this->setCookie($sc);
|
||||
@ -271,16 +264,14 @@ class CookieJar implements CookieJarInterface
|
||||
* Computes cookie path following RFC 6265 section 5.1.4.
|
||||
*
|
||||
* @see https://tools.ietf.org/html/rfc6265#section-5.1.4
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
private function getCookiePathFromRequest(RequestInterface $request)
|
||||
private function getCookiePathFromRequest(RequestInterface $request): string
|
||||
{
|
||||
$uriPath = $request->getUri()->getPath();
|
||||
if ($uriPath === '') {
|
||||
return '/';
|
||||
}
|
||||
if (strpos($uriPath, '/') !== 0) {
|
||||
if (! str_starts_with($uriPath, '/')) {
|
||||
return '/';
|
||||
}
|
||||
if ($uriPath === '/') {
|
||||
@ -297,7 +288,7 @@ class CookieJar implements CookieJarInterface
|
||||
* If a cookie already exists and the server asks to set it again with a
|
||||
* null value, the cookie must be deleted.
|
||||
*/
|
||||
private function removeCookieIfEmpty(SetCookie $cookie)
|
||||
private function removeCookieIfEmpty(SetCookie $cookie): void
|
||||
{
|
||||
$cookieValue = $cookie->getValue();
|
||||
if ($cookieValue === null || $cookieValue === '') {
|
||||
|
@ -11,6 +11,7 @@ declare(strict_types=1);
|
||||
*/
|
||||
namespace Hyperf\HttpMessage\Cookie;
|
||||
|
||||
use Hyperf\Contract\Arrayable;
|
||||
use Psr\Http\Message\RequestInterface;
|
||||
use Psr\Http\Message\ResponseInterface;
|
||||
|
||||
@ -24,7 +25,7 @@ use Psr\Http\Message\ResponseInterface;
|
||||
*
|
||||
* @see http://docs.python.org/2/library/cookielib.html Inspiration
|
||||
*/
|
||||
interface CookieJarInterface extends \Countable, \IteratorAggregate
|
||||
interface CookieJarInterface extends \Countable, \IteratorAggregate, Arrayable
|
||||
{
|
||||
/**
|
||||
* Create a request with added cookie headers.
|
||||
@ -84,11 +85,4 @@ interface CookieJarInterface extends \Countable, \IteratorAggregate
|
||||
* to RFC 2965.
|
||||
*/
|
||||
public function clearSessionCookies();
|
||||
|
||||
/**
|
||||
* Converts the cookie jar to an array.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function toArray();
|
||||
}
|
||||
|
@ -11,33 +11,26 @@ declare(strict_types=1);
|
||||
*/
|
||||
namespace Hyperf\HttpMessage\Cookie;
|
||||
|
||||
use Hyperf\Utils\Codec\Json;
|
||||
|
||||
/**
|
||||
* Persists non-session cookies using a JSON formatted file.
|
||||
*/
|
||||
class FileCookieJar extends CookieJar
|
||||
{
|
||||
/** @var string filename */
|
||||
private $filename;
|
||||
|
||||
/** @var bool Control whether to persist session cookies or not. */
|
||||
private $storeSessionCookies;
|
||||
|
||||
/**
|
||||
* Create a new FileCookieJar object.
|
||||
*
|
||||
* @param string $cookieFile File to store the cookie data
|
||||
* @param bool $storeSessionCookies set to true to store session cookies
|
||||
* in the cookie jar
|
||||
* @param string $filename File to store the cookie data
|
||||
* @param bool $storeSessionCookies Control whether to persist session cookies or not.
|
||||
* Set to true to store session cookies in the cookie jar.
|
||||
*
|
||||
* @throws \RuntimeException if the file cannot be found or created
|
||||
*/
|
||||
public function __construct($cookieFile, $storeSessionCookies = false)
|
||||
public function __construct(private string $filename, private bool $storeSessionCookies = false)
|
||||
{
|
||||
$this->filename = $cookieFile;
|
||||
$this->storeSessionCookies = $storeSessionCookies;
|
||||
|
||||
if (file_exists($cookieFile)) {
|
||||
$this->load($cookieFile);
|
||||
if (file_exists($filename)) {
|
||||
$this->load($filename);
|
||||
}
|
||||
}
|
||||
|
||||
@ -55,7 +48,7 @@ class FileCookieJar extends CookieJar
|
||||
* @param string $filename File to save
|
||||
* @throws \RuntimeException if the file cannot be found or created
|
||||
*/
|
||||
public function save($filename)
|
||||
public function save(string $filename): void
|
||||
{
|
||||
$json = [];
|
||||
foreach ($this as $cookie) {
|
||||
@ -65,7 +58,7 @@ class FileCookieJar extends CookieJar
|
||||
}
|
||||
}
|
||||
|
||||
$jsonStr = \GuzzleHttp\json_encode($json);
|
||||
$jsonStr = Json::encode($json);
|
||||
if (file_put_contents($filename, $jsonStr) === false) {
|
||||
throw new \RuntimeException("Unable to save file {$filename}");
|
||||
}
|
||||
@ -79,7 +72,7 @@ class FileCookieJar extends CookieJar
|
||||
* @param string $filename cookie file to load
|
||||
* @throws \RuntimeException if the file cannot be loaded
|
||||
*/
|
||||
public function load($filename)
|
||||
public function load(string $filename): void
|
||||
{
|
||||
$json = file_get_contents($filename);
|
||||
if ($json === false) {
|
||||
@ -89,9 +82,9 @@ class FileCookieJar extends CookieJar
|
||||
return;
|
||||
}
|
||||
|
||||
$data = \GuzzleHttp\json_decode($json, true);
|
||||
$data = Json::decode($json);
|
||||
if (is_array($data)) {
|
||||
foreach (json_decode($json, true) as $cookie) {
|
||||
foreach ($data as $cookie) {
|
||||
$this->setCookie(new SetCookie($cookie));
|
||||
}
|
||||
} elseif (strlen($data)) {
|
||||
|
@ -12,32 +12,21 @@ declare(strict_types=1);
|
||||
namespace Hyperf\HttpMessage\Cookie;
|
||||
|
||||
/**
|
||||
* Persists cookies in the client session.
|
||||
* Persists cookies in the client session for FPM.
|
||||
*/
|
||||
class SessionCookieJar extends CookieJar
|
||||
{
|
||||
/**
|
||||
* @var string session key
|
||||
*/
|
||||
private $sessionKey;
|
||||
|
||||
/**
|
||||
* @var bool control whether to persist session cookies or not
|
||||
*/
|
||||
private $storeSessionCookies;
|
||||
|
||||
/**
|
||||
* Create a new SessionCookieJar object.
|
||||
*
|
||||
* @param string $sessionKey Session key name to store the cookie
|
||||
* data in session
|
||||
* @param bool $storeSessionCookies set to true to store session cookies
|
||||
* @param bool $storeSessionCookies Control whether to persist session cookies or not.
|
||||
* set to true to store session cookies
|
||||
* in the cookie jar
|
||||
*/
|
||||
public function __construct($sessionKey, $storeSessionCookies = false)
|
||||
public function __construct(private string $sessionKey, private bool $storeSessionCookies = false)
|
||||
{
|
||||
$this->sessionKey = $sessionKey;
|
||||
$this->storeSessionCookies = $storeSessionCookies;
|
||||
$this->load();
|
||||
}
|
||||
|
||||
@ -52,7 +41,7 @@ class SessionCookieJar extends CookieJar
|
||||
/**
|
||||
* Save cookies to the client session.
|
||||
*/
|
||||
public function save()
|
||||
public function save(): void
|
||||
{
|
||||
$json = [];
|
||||
foreach ($this as $cookie) {
|
||||
@ -68,7 +57,7 @@ class SessionCookieJar extends CookieJar
|
||||
/**
|
||||
* Load the contents of the client session into the data array.
|
||||
*/
|
||||
protected function load()
|
||||
protected function load(): void
|
||||
{
|
||||
if (! isset($_SESSION[$this->sessionKey])) {
|
||||
return;
|
||||
|
@ -11,13 +11,15 @@ declare(strict_types=1);
|
||||
*/
|
||||
namespace Hyperf\HttpMessage\Cookie;
|
||||
|
||||
use Hyperf\Contract\Arrayable;
|
||||
use Stringable;
|
||||
|
||||
/**
|
||||
* Set-Cookie object.
|
||||
*/
|
||||
class SetCookie
|
||||
class SetCookie implements Stringable, Arrayable
|
||||
{
|
||||
/** @var array */
|
||||
private static $defaults = [
|
||||
private static array $defaults = [
|
||||
'Name' => null,
|
||||
'Value' => null,
|
||||
'Domain' => null,
|
||||
@ -29,8 +31,10 @@ class SetCookie
|
||||
'HttpOnly' => false,
|
||||
];
|
||||
|
||||
/** @var array Cookie data */
|
||||
private $data;
|
||||
/**
|
||||
* Array of cookie data provided by a Cookie parser.
|
||||
*/
|
||||
private array $data;
|
||||
|
||||
/**
|
||||
* @param array $data Array of cookie data provided by a Cookie parser
|
||||
@ -67,10 +71,8 @@ class SetCookie
|
||||
* Create a new SetCookie object from a string.
|
||||
*
|
||||
* @param string $cookie Set-Cookie header string
|
||||
*
|
||||
* @return self
|
||||
*/
|
||||
public static function fromString($cookie)
|
||||
public static function fromString(string $cookie): self
|
||||
{
|
||||
// Create the default return array
|
||||
$data = self::$defaults;
|
||||
@ -107,17 +109,15 @@ class SetCookie
|
||||
return new self($data);
|
||||
}
|
||||
|
||||
public function toArray()
|
||||
public function toArray(): array
|
||||
{
|
||||
return $this->data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the cookie name.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getName()
|
||||
public function getName(): ?string
|
||||
{
|
||||
return $this->data['Name'];
|
||||
}
|
||||
@ -127,17 +127,15 @@ class SetCookie
|
||||
*
|
||||
* @param string $name Cookie name
|
||||
*/
|
||||
public function setName($name)
|
||||
public function setName(?string $name)
|
||||
{
|
||||
$this->data['Name'] = $name;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the cookie value.
|
||||
*
|
||||
* @return null|string
|
||||
*/
|
||||
public function getValue()
|
||||
public function getValue(): ?string
|
||||
{
|
||||
return $this->data['Value'];
|
||||
}
|
||||
@ -147,17 +145,15 @@ class SetCookie
|
||||
*
|
||||
* @param null|string $value Cookie value
|
||||
*/
|
||||
public function setValue($value)
|
||||
public function setValue(?string $value)
|
||||
{
|
||||
$this->data['Value'] = $value;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the domain.
|
||||
*
|
||||
* @return null|string
|
||||
*/
|
||||
public function getDomain()
|
||||
public function getDomain(): ?string
|
||||
{
|
||||
return $this->data['Domain'];
|
||||
}
|
||||
@ -167,17 +163,15 @@ class SetCookie
|
||||
*
|
||||
* @param string $domain
|
||||
*/
|
||||
public function setDomain($domain)
|
||||
public function setDomain(?string $domain)
|
||||
{
|
||||
$this->data['Domain'] = $domain;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the path.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getPath()
|
||||
public function getPath(): string
|
||||
{
|
||||
return $this->data['Path'];
|
||||
}
|
||||
@ -187,17 +181,15 @@ class SetCookie
|
||||
*
|
||||
* @param string $path Path of the cookie
|
||||
*/
|
||||
public function setPath($path)
|
||||
public function setPath(string $path)
|
||||
{
|
||||
$this->data['Path'] = $path;
|
||||
}
|
||||
|
||||
/**
|
||||
* Maximum lifetime of the cookie in seconds.
|
||||
*
|
||||
* @return null|int
|
||||
*/
|
||||
public function getMaxAge()
|
||||
public function getMaxAge(): ?int
|
||||
{
|
||||
return $this->data['Max-Age'];
|
||||
}
|
||||
@ -205,9 +197,9 @@ class SetCookie
|
||||
/**
|
||||
* Set the max-age of the cookie.
|
||||
*
|
||||
* @param int $maxAge Max age of the cookie in seconds
|
||||
* @param null|int $maxAge Max age of the cookie in seconds
|
||||
*/
|
||||
public function setMaxAge($maxAge)
|
||||
public function setMaxAge(?int $maxAge)
|
||||
{
|
||||
$this->data['Max-Age'] = $maxAge;
|
||||
}
|
||||
@ -225,7 +217,7 @@ class SetCookie
|
||||
/**
|
||||
* Set the unix timestamp for which the cookie will expire.
|
||||
*
|
||||
* @param int|string $timestamp Unix timestamp
|
||||
* @param float|int|string $timestamp Unix timestamp
|
||||
*/
|
||||
public function setExpires($timestamp)
|
||||
{
|
||||
@ -235,61 +227,55 @@ class SetCookie
|
||||
}
|
||||
|
||||
/**
|
||||
* Get whether or not this is a secure cookie.
|
||||
*
|
||||
* @return null|bool
|
||||
* Get whether this is a secure cookie.
|
||||
*/
|
||||
public function getSecure()
|
||||
public function getSecure(): bool
|
||||
{
|
||||
return $this->data['Secure'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Set whether or not the cookie is secure.
|
||||
* Set whether the cookie is secure.
|
||||
*
|
||||
* @param bool $secure Set to true or false if secure
|
||||
*/
|
||||
public function setSecure($secure)
|
||||
public function setSecure(bool $secure)
|
||||
{
|
||||
$this->data['Secure'] = $secure;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get whether or not this is a session cookie.
|
||||
*
|
||||
* @return null|bool
|
||||
* Get whether this is a session cookie.
|
||||
*/
|
||||
public function getDiscard()
|
||||
public function getDiscard(): bool
|
||||
{
|
||||
return $this->data['Discard'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Set whether or not this is a session cookie.
|
||||
* Set whether this is a session cookie.
|
||||
*
|
||||
* @param bool $discard Set to true or false if this is a session cookie
|
||||
*/
|
||||
public function setDiscard($discard)
|
||||
public function setDiscard(bool $discard)
|
||||
{
|
||||
$this->data['Discard'] = $discard;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get whether or not this is an HTTP only cookie.
|
||||
*
|
||||
* @return bool
|
||||
* Get whether this is an HTTP only cookie.
|
||||
*/
|
||||
public function getHttpOnly()
|
||||
public function getHttpOnly(): bool
|
||||
{
|
||||
return $this->data['HttpOnly'];
|
||||
}
|
||||
|
||||
/**
|
||||
* Set whether or not this is an HTTP only cookie.
|
||||
* Set whether this is an HTTP only cookie.
|
||||
*
|
||||
* @param bool $httpOnly Set to true or false if this is HTTP only
|
||||
*/
|
||||
public function setHttpOnly($httpOnly)
|
||||
public function setHttpOnly(bool $httpOnly)
|
||||
{
|
||||
$this->data['HttpOnly'] = $httpOnly;
|
||||
}
|
||||
@ -308,10 +294,8 @@ class SetCookie
|
||||
* path is a %x2F ("/") character.
|
||||
*
|
||||
* @param string $requestPath Path to check against
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function matchesPath($requestPath)
|
||||
public function matchesPath(string $requestPath): bool
|
||||
{
|
||||
$cookiePath = $this->getPath();
|
||||
|
||||
@ -321,7 +305,7 @@ class SetCookie
|
||||
}
|
||||
|
||||
// Ensure that the cookie-path is a prefix of the request path.
|
||||
if (strpos($requestPath, $cookiePath) !== 0) {
|
||||
if (! str_starts_with($requestPath, $cookiePath)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -338,10 +322,8 @@ class SetCookie
|
||||
* Check if the cookie matches a domain value.
|
||||
*
|
||||
* @param string $domain Domain to check against
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function matchesDomain($domain)
|
||||
public function matchesDomain(string $domain): bool
|
||||
{
|
||||
// Remove the leading '.' as per spec in RFC 6265.
|
||||
// http://tools.ietf.org/html/rfc6265#section-5.2.3
|
||||
@ -363,10 +345,8 @@ class SetCookie
|
||||
|
||||
/**
|
||||
* Check if the cookie is expired.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function isExpired()
|
||||
public function isExpired(): bool
|
||||
{
|
||||
return $this->getExpires() && time() > $this->getExpires();
|
||||
}
|
||||
@ -376,7 +356,7 @@ class SetCookie
|
||||
*
|
||||
* @return bool|string Returns true if valid or an error message if invalid
|
||||
*/
|
||||
public function validate()
|
||||
public function validate(): bool|string
|
||||
{
|
||||
// Names must not be empty, but can be 0
|
||||
$name = $this->getName();
|
||||
|
@ -17,20 +17,14 @@ use RuntimeException;
|
||||
class HttpException extends RuntimeException
|
||||
{
|
||||
/**
|
||||
* @var int HTTP status
|
||||
*/
|
||||
public $statusCode;
|
||||
|
||||
/**
|
||||
* @param int $status HTTP status
|
||||
* @param int $statusCode HTTP status
|
||||
* @param null|string $message error message
|
||||
* @param int $code error code
|
||||
*/
|
||||
public function __construct($status, $message = '', $code = 0, \Throwable $previous = null)
|
||||
public function __construct(public int $statusCode, $message = '', $code = 0, \Throwable $previous = null)
|
||||
{
|
||||
$this->statusCode = $status;
|
||||
if (is_null($message)) {
|
||||
$message = Response::getReasonPhraseByCode($status);
|
||||
$message = Response::getReasonPhraseByCode($statusCode);
|
||||
}
|
||||
|
||||
parent::__construct($message, $code, $previous);
|
||||
|
@ -20,62 +20,41 @@ use Hyperf\Utils\ApplicationContext;
|
||||
use Psr\Http\Message\RequestInterface;
|
||||
use Psr\Http\Message\ServerRequestInterface;
|
||||
use Psr\Http\Message\UploadedFileInterface;
|
||||
use Psr\Http\Message\UriInterface;
|
||||
use Swoole;
|
||||
|
||||
class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestInterface
|
||||
{
|
||||
/**
|
||||
* @var \Swoole\Http\Request
|
||||
*/
|
||||
protected $swooleRequest;
|
||||
protected ?Swoole\Http\Request $swooleRequest = null;
|
||||
|
||||
/**
|
||||
* @var null|RequestParserInterface
|
||||
*/
|
||||
protected static $parser;
|
||||
protected static ?RequestParserInterface $parser = null;
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private $attributes = [];
|
||||
private array $attributes = [];
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private $cookieParams = [];
|
||||
private array $cookieParams = [];
|
||||
|
||||
/**
|
||||
* @var null|array|object
|
||||
*/
|
||||
private $parsedBody;
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private $queryParams = [];
|
||||
private array $queryParams = [];
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private $serverParams = [];
|
||||
private array $serverParams = [];
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private $uploadedFiles = [];
|
||||
private array $uploadedFiles = [];
|
||||
|
||||
/**
|
||||
* the body of parser.
|
||||
*
|
||||
* @var mixed
|
||||
*/
|
||||
private $bodyParams;
|
||||
private mixed $bodyParams;
|
||||
|
||||
/**
|
||||
* Load a swoole request, and transfer to a psr-7 request object.
|
||||
*
|
||||
* @return \Hyperf\HttpMessage\Server\Request
|
||||
*/
|
||||
public static function loadFromSwooleRequest(\Swoole\Http\Request $swooleRequest)
|
||||
public static function loadFromSwooleRequest(Swoole\Http\Request $swooleRequest)
|
||||
{
|
||||
$server = $swooleRequest->server;
|
||||
$method = $server['request_method'] ?? 'GET';
|
||||
@ -108,10 +87,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
|
||||
/**
|
||||
* Return an instance with the specified server params.
|
||||
*
|
||||
* @return static
|
||||
*/
|
||||
public function withServerParams(array $serverParams)
|
||||
public function withServerParams(array $serverParams): static
|
||||
{
|
||||
$clone = clone $this;
|
||||
$clone->serverParams = $serverParams;
|
||||
@ -141,9 +118,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
* updated cookie values.
|
||||
*
|
||||
* @param array $cookies array of key/value pairs representing cookies
|
||||
* @return static
|
||||
*/
|
||||
public function withCookieParams(array $cookies)
|
||||
public function withCookieParams(array $cookies): static
|
||||
{
|
||||
$clone = clone $this;
|
||||
$clone->cookieParams = $cookies;
|
||||
@ -168,10 +144,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
*
|
||||
* @param string $name the name of param
|
||||
* @param mixed $value the value of param
|
||||
*
|
||||
* @return static
|
||||
*/
|
||||
public function addQueryParam(string $name, $value)
|
||||
public function addQueryParam(string $name, mixed $value): static
|
||||
{
|
||||
$clone = clone $this;
|
||||
$clone->queryParams[$name] = $value;
|
||||
@ -260,10 +234,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
*
|
||||
* @param string $name the name of param
|
||||
* @param mixed $value the value of param
|
||||
*
|
||||
* @return static
|
||||
*/
|
||||
public function addParserBody(string $name, $value)
|
||||
public function addParserBody(string $name, mixed $value): static
|
||||
{
|
||||
if (is_array($this->parsedBody)) {
|
||||
$clone = clone $this;
|
||||
@ -276,10 +248,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
|
||||
/**
|
||||
* return parser result of body.
|
||||
*
|
||||
* @return mixed
|
||||
*/
|
||||
public function getBodyParams()
|
||||
public function getBodyParams(): mixed
|
||||
{
|
||||
return $this->bodyParams;
|
||||
}
|
||||
@ -316,12 +286,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
|
||||
/**
|
||||
* init body params from parser result.
|
||||
*
|
||||
* @param mixed $data
|
||||
*
|
||||
* @return static
|
||||
*/
|
||||
public function withBodyParams($data)
|
||||
public function withBodyParams(mixed $data): static
|
||||
{
|
||||
$clone = clone $this;
|
||||
$clone->bodyParams = $data;
|
||||
@ -407,10 +373,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
|
||||
/**
|
||||
* Get the URL (no query string) for the request.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function url()
|
||||
public function url(): string
|
||||
{
|
||||
return rtrim(preg_replace('/\?.*/', '', (string) $this->getUri()), '/');
|
||||
}
|
||||
@ -427,10 +391,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
|
||||
/**
|
||||
* Determine if the request is the result of an ajax call.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
public function isAjax()
|
||||
public function isAjax(): bool
|
||||
{
|
||||
return $this->isXmlHttpRequest();
|
||||
}
|
||||
@ -445,26 +407,23 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
*
|
||||
* @return bool true if the request is an XMLHttpRequest, false otherwise
|
||||
*/
|
||||
public function isXmlHttpRequest()
|
||||
public function isXmlHttpRequest(): bool
|
||||
{
|
||||
return $this->getHeaderLine('X-Requested-With') == 'XMLHttpRequest';
|
||||
}
|
||||
|
||||
public function getSwooleRequest(): \Swoole\Http\Request
|
||||
public function getSwooleRequest(): Swoole\Http\Request
|
||||
{
|
||||
return $this->swooleRequest;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return $this
|
||||
*/
|
||||
public function setSwooleRequest(\Swoole\Http\Request $swooleRequest)
|
||||
public function setSwooleRequest(Swoole\Http\Request $swooleRequest): static
|
||||
{
|
||||
$this->swooleRequest = $swooleRequest;
|
||||
return $this;
|
||||
}
|
||||
|
||||
protected static function normalizeParsedBody(array $data = [], ?RequestInterface $request = null)
|
||||
protected static function normalizeParsedBody(array $data = [], ?RequestInterface $request = null): array
|
||||
{
|
||||
if (! $request) {
|
||||
return $data;
|
||||
@ -510,9 +469,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
*
|
||||
* @param array $files A array which respect $_FILES structure
|
||||
* @throws \InvalidArgumentException for unrecognized values
|
||||
* @return array
|
||||
*/
|
||||
private static function normalizeFiles(array $files)
|
||||
private static function normalizeFiles(array $files): array
|
||||
{
|
||||
$normalized = [];
|
||||
|
||||
@ -523,7 +481,6 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
$normalized[$key] = self::createUploadedFileFromSpec($value);
|
||||
} elseif (is_array($value)) {
|
||||
$normalized[$key] = self::normalizeFiles($value);
|
||||
continue;
|
||||
} else {
|
||||
throw new BadRequestHttpException('Invalid value in files specification');
|
||||
}
|
||||
@ -538,9 +495,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
* delegate to normalizeNestedFileSpec() and return that return value.
|
||||
*
|
||||
* @param array $value $_FILES struct
|
||||
* @return array|UploadedFileInterface
|
||||
*/
|
||||
private static function createUploadedFileFromSpec(array $value)
|
||||
private static function createUploadedFileFromSpec(array $value): array|UploadedFileInterface
|
||||
{
|
||||
if (is_array($value['tmp_name'])) {
|
||||
return self::normalizeNestedFileSpec($value);
|
||||
@ -556,7 +512,7 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
*
|
||||
* @return UploadedFileInterface[]
|
||||
*/
|
||||
private static function normalizeNestedFileSpec(array $files = [])
|
||||
private static function normalizeNestedFileSpec(array $files = []): array
|
||||
{
|
||||
$normalizedFiles = [];
|
||||
|
||||
@ -577,9 +533,8 @@ class Request extends \Hyperf\HttpMessage\Base\Request implements ServerRequestI
|
||||
/**
|
||||
* Get a Uri populated with values from $swooleRequest->server.
|
||||
* @throws \InvalidArgumentException
|
||||
* @return \Psr\Http\Message\UriInterface
|
||||
*/
|
||||
private static function getUriFromGlobals(\Swoole\Http\Request $swooleRequest)
|
||||
private static function getUriFromGlobals(Swoole\Http\Request $swooleRequest): UriInterface
|
||||
{
|
||||
$server = $swooleRequest->server;
|
||||
$header = $swooleRequest->header;
|
||||
|
@ -18,15 +18,9 @@ use InvalidArgumentException;
|
||||
|
||||
class JsonParser implements RequestParserInterface
|
||||
{
|
||||
/**
|
||||
* @var bool
|
||||
*/
|
||||
public $asArray = true;
|
||||
public bool $asArray = true;
|
||||
|
||||
/**
|
||||
* @var bool
|
||||
*/
|
||||
public $throwException = true;
|
||||
public bool $throwException = true;
|
||||
|
||||
public function parse(string $rawBody, string $contentType): array
|
||||
{
|
||||
|
@ -15,7 +15,7 @@ use Hyperf\HttpMessage\Server\RequestParserInterface;
|
||||
|
||||
class Parser implements RequestParserInterface
|
||||
{
|
||||
protected $parsers = [];
|
||||
protected array $parsers = [];
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
|
@ -18,7 +18,7 @@ use InvalidArgumentException;
|
||||
|
||||
class XmlParser implements RequestParserInterface
|
||||
{
|
||||
public $throwException = true;
|
||||
public bool $throwException = true;
|
||||
|
||||
public function parse(string $rawBody, string $contentType): array
|
||||
{
|
||||
|
@ -16,20 +16,14 @@ use Hyperf\HttpMessage\Stream\SwooleStream;
|
||||
|
||||
class Response extends \Hyperf\HttpMessage\Base\Response
|
||||
{
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
protected $cookies = [];
|
||||
protected array $cookies = [];
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
protected $trailers = [];
|
||||
protected array $trailers = [];
|
||||
|
||||
/**
|
||||
* Returns an instance with body content.
|
||||
*/
|
||||
public function withContent(string $content): self
|
||||
public function withContent(string $content): static
|
||||
{
|
||||
$new = clone $this;
|
||||
$new->stream = new SwooleStream($content);
|
||||
@ -39,7 +33,7 @@ class Response extends \Hyperf\HttpMessage\Base\Response
|
||||
/**
|
||||
* Returns an instance with specified cookies.
|
||||
*/
|
||||
public function withCookie(Cookie $cookie): self
|
||||
public function withCookie(Cookie $cookie): static
|
||||
{
|
||||
$clone = clone $this;
|
||||
$clone->cookies[$cookie->getDomain()][$cookie->getPath()][$cookie->getName()] = $cookie;
|
||||
@ -58,7 +52,7 @@ class Response extends \Hyperf\HttpMessage\Base\Response
|
||||
* Returns an instance with specified trailer.
|
||||
* @param string $value
|
||||
*/
|
||||
public function withTrailer(string $key, $value): self
|
||||
public function withTrailer(string $key, $value): static
|
||||
{
|
||||
$new = clone $this;
|
||||
$new->trailers[$key] = $value;
|
||||
|
@ -18,10 +18,7 @@ use RuntimeException;
|
||||
|
||||
trait ResponseProxyTrait
|
||||
{
|
||||
/**
|
||||
* @var null|ResponseInterface
|
||||
*/
|
||||
protected $response;
|
||||
protected ?ResponseInterface $response = null;
|
||||
|
||||
public function setResponse(ResponseInterface $response)
|
||||
{
|
||||
@ -115,7 +112,7 @@ trait ResponseProxyTrait
|
||||
/**
|
||||
* Returns an instance with specified cookies.
|
||||
*/
|
||||
public function withCookie(Cookie $cookie): self
|
||||
public function withCookie(Cookie $cookie): static
|
||||
{
|
||||
$response = $this->getResponse();
|
||||
if (! method_exists($response, 'withCookie')) {
|
||||
@ -142,7 +139,7 @@ trait ResponseProxyTrait
|
||||
* Returns an instance with specified trailer.
|
||||
* @param string $value
|
||||
*/
|
||||
public function withTrailer(string $key, $value): self
|
||||
public function withTrailer(string $key, $value): static
|
||||
{
|
||||
$response = $this->getResponse();
|
||||
if (! method_exists($response, 'withTrailer')) {
|
||||
|
@ -13,18 +13,14 @@ namespace Hyperf\HttpMessage\Stream;
|
||||
|
||||
use Hyperf\HttpServer\Exception\Http\FileException;
|
||||
use Psr\Http\Message\StreamInterface;
|
||||
use SplFileInfo;
|
||||
use Stringable;
|
||||
|
||||
class SwooleFileStream implements StreamInterface, FileInterface
|
||||
class SwooleFileStream implements StreamInterface, FileInterface, Stringable
|
||||
{
|
||||
/**
|
||||
* @var int
|
||||
*/
|
||||
protected $size;
|
||||
protected int $size;
|
||||
|
||||
/**
|
||||
* @var \SplFileInfo
|
||||
*/
|
||||
protected $file;
|
||||
protected SplFileInfo $file;
|
||||
|
||||
/**
|
||||
* SwooleFileStream constructor.
|
||||
@ -33,8 +29,8 @@ class SwooleFileStream implements StreamInterface, FileInterface
|
||||
*/
|
||||
public function __construct($file)
|
||||
{
|
||||
if (! $file instanceof \SplFileInfo) {
|
||||
$file = new \SplFileInfo($file);
|
||||
if (! $file instanceof SplFileInfo) {
|
||||
$file = new SplFileInfo($file);
|
||||
}
|
||||
if (! $file->isReadable()) {
|
||||
throw new FileException('File must be readable.');
|
||||
@ -58,7 +54,7 @@ class SwooleFileStream implements StreamInterface, FileInterface
|
||||
{
|
||||
try {
|
||||
return $this->getContents();
|
||||
} catch (\Throwable $e) {
|
||||
} catch (\Throwable) {
|
||||
return '';
|
||||
}
|
||||
}
|
||||
@ -117,7 +113,7 @@ class SwooleFileStream implements StreamInterface, FileInterface
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether or not the stream is seekable.
|
||||
* Returns whether the stream is seekable.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
@ -158,7 +154,7 @@ class SwooleFileStream implements StreamInterface, FileInterface
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether or not the stream is writable.
|
||||
* Returns whether the stream is writable.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
@ -180,7 +176,7 @@ class SwooleFileStream implements StreamInterface, FileInterface
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether or not the stream is readable.
|
||||
* Returns whether the stream is readable.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
@ -192,8 +188,8 @@ class SwooleFileStream implements StreamInterface, FileInterface
|
||||
/**
|
||||
* Read data from the stream.
|
||||
*
|
||||
* @param int $length Read up to $length bytes from the object and return
|
||||
* them. Fewer than $length bytes may be returned if underlying stream
|
||||
* @param int $length Read up to $length bytes from the object and return them.
|
||||
* Fewer than $length bytes may be returned if underlying stream
|
||||
* call returns fewer bytes.
|
||||
* @throws \RuntimeException if an error occurs
|
||||
* @return string returns the data read from the stream, or an empty string
|
||||
|
@ -12,30 +12,19 @@ declare(strict_types=1);
|
||||
namespace Hyperf\HttpMessage\Stream;
|
||||
|
||||
use Psr\Http\Message\StreamInterface;
|
||||
use Stringable;
|
||||
|
||||
class SwooleStream implements StreamInterface
|
||||
class SwooleStream implements StreamInterface, Stringable
|
||||
{
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
protected $contents;
|
||||
protected int $size;
|
||||
|
||||
/**
|
||||
* @var int
|
||||
*/
|
||||
protected $size;
|
||||
|
||||
/**
|
||||
* @var bool
|
||||
*/
|
||||
protected $writable;
|
||||
protected bool $writable;
|
||||
|
||||
/**
|
||||
* SwooleStream constructor.
|
||||
*/
|
||||
public function __construct(string $contents = '')
|
||||
public function __construct(protected string $contents = '')
|
||||
{
|
||||
$this->contents = $contents;
|
||||
$this->size = strlen($this->contents);
|
||||
$this->writable = true;
|
||||
}
|
||||
@ -55,7 +44,7 @@ class SwooleStream implements StreamInterface
|
||||
{
|
||||
try {
|
||||
return $this->getContents();
|
||||
} catch (\Throwable $e) {
|
||||
} catch (\Throwable) {
|
||||
return '';
|
||||
}
|
||||
}
|
||||
|
@ -20,7 +20,7 @@ class UploadedFile extends \SplFileInfo implements UploadedFileInterface
|
||||
/**
|
||||
* @var int[]
|
||||
*/
|
||||
private static $errors = [
|
||||
private static array $errors = [
|
||||
UPLOAD_ERR_OK,
|
||||
UPLOAD_ERR_INI_SIZE,
|
||||
UPLOAD_ERR_FORM_SIZE,
|
||||
@ -31,53 +31,30 @@ class UploadedFile extends \SplFileInfo implements UploadedFileInterface
|
||||
UPLOAD_ERR_EXTENSION,
|
||||
];
|
||||
|
||||
/**
|
||||
* @var null|string
|
||||
*/
|
||||
private $clientFilename;
|
||||
private ?string $tmpFile = null;
|
||||
|
||||
/**
|
||||
* @var null|string
|
||||
*/
|
||||
private $clientMediaType;
|
||||
|
||||
/**
|
||||
* @var int
|
||||
*/
|
||||
private $error;
|
||||
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
private $tmpFile;
|
||||
|
||||
/**
|
||||
* @var bool
|
||||
*/
|
||||
private $moved = false;
|
||||
|
||||
/**
|
||||
* @var null|int
|
||||
*/
|
||||
private $size;
|
||||
private bool $moved = false;
|
||||
|
||||
/**
|
||||
* @var null|string
|
||||
*/
|
||||
private $mimeType;
|
||||
|
||||
/**
|
||||
* @param int $size The file size
|
||||
* @param int $error The error associated with the uploaded file
|
||||
* @param null|string $clientFilename The filename sent by the client
|
||||
* @param null|string $clientMediaType The media type sent by the client
|
||||
*/
|
||||
public function __construct(
|
||||
string $tmpFile,
|
||||
?int $size,
|
||||
int $errorStatus,
|
||||
?string $clientFilename = null,
|
||||
?string $clientMediaType = null
|
||||
private int $size,
|
||||
private int $error,
|
||||
private ?string $clientFilename = null,
|
||||
private ?string $clientMediaType = null
|
||||
) {
|
||||
$this->setError($errorStatus)
|
||||
->setSize($size)
|
||||
->setClientFilename($clientFilename)
|
||||
->setClientMediaType($clientMediaType);
|
||||
$this->isOk() && $this->setFile($tmpFile);
|
||||
$this->checkError($this->error);
|
||||
$this->isOk() && $this->tmpFile = $tmpFile;
|
||||
parent::__construct($tmpFile);
|
||||
}
|
||||
|
||||
@ -120,7 +97,7 @@ class UploadedFile extends \SplFileInfo implements UploadedFileInterface
|
||||
}
|
||||
|
||||
/**
|
||||
* Determire if the temp file is moved.
|
||||
* Determine if the temp file is moved.
|
||||
*/
|
||||
public function isMoved(): bool
|
||||
{
|
||||
@ -200,7 +177,7 @@ class UploadedFile extends \SplFileInfo implements UploadedFileInterface
|
||||
*
|
||||
* @return null|int the file size in bytes or null if unknown
|
||||
*/
|
||||
public function getSize(): ?int
|
||||
public function getSize()
|
||||
{
|
||||
return $this->size;
|
||||
}
|
||||
@ -264,71 +241,11 @@ class UploadedFile extends \SplFileInfo implements UploadedFileInterface
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Depending on the value set file or stream variable.
|
||||
*/
|
||||
private function setFile(string $file): self
|
||||
{
|
||||
$this->tmpFile = $file;
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws \InvalidArgumentException If invalid error status for UploadedFile
|
||||
*/
|
||||
private function setError(int $error): self
|
||||
private function checkError(int $error): void
|
||||
{
|
||||
if (in_array($error, UploadedFile::$errors) === false) {
|
||||
throw new \InvalidArgumentException('Invalid error status for UploadedFile');
|
||||
}
|
||||
|
||||
$this->error = $error;
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param null|int $size the file size in bytes or null if unknown
|
||||
* @throws \InvalidArgumentException if the size is not a interger
|
||||
*/
|
||||
private function setSize($size): self
|
||||
{
|
||||
if (is_int($size) === false) {
|
||||
throw new \InvalidArgumentException('Upload file size must be an integer');
|
||||
}
|
||||
|
||||
$this->size = $size;
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws \InvalidArgumentException
|
||||
*/
|
||||
private function setClientFilename(?string $clientFilename): self
|
||||
{
|
||||
if ($this->isStringOrNull($clientFilename) === false) {
|
||||
throw new \InvalidArgumentException('Upload file client filename must be a string or null');
|
||||
}
|
||||
|
||||
$this->clientFilename = $clientFilename;
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws \InvalidArgumentException
|
||||
*/
|
||||
private function setClientMediaType(?string $clientMediaType): self
|
||||
{
|
||||
if ($this->isStringOrNull($clientMediaType) === false) {
|
||||
throw new \InvalidArgumentException('Upload file client media type must be a string or null');
|
||||
}
|
||||
|
||||
$this->clientMediaType = $clientMediaType;
|
||||
return $this;
|
||||
}
|
||||
|
||||
private function isStringOrNull($param): bool
|
||||
{
|
||||
return in_array(gettype($param), ['string', 'NULL']);
|
||||
}
|
||||
|
||||
private function isStringNotEmpty($param): bool
|
||||
|
@ -12,8 +12,9 @@ declare(strict_types=1);
|
||||
namespace Hyperf\HttpMessage\Uri;
|
||||
|
||||
use Psr\Http\Message\UriInterface;
|
||||
use Stringable;
|
||||
|
||||
class Uri implements UriInterface
|
||||
class Uri implements UriInterface, Stringable
|
||||
{
|
||||
/**
|
||||
* Absolute http and https URIs require a host per RFC 7230 Section 2.7
|
||||
@ -23,71 +24,58 @@ class Uri implements UriInterface
|
||||
*/
|
||||
public const DEFAULT_HTTP_HOST = 'localhost';
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private static $defaultPorts = [
|
||||
private static array $defaultPorts = [
|
||||
'http' => 80,
|
||||
'https' => 443,
|
||||
];
|
||||
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
private static $charUnreserved = 'a-zA-Z0-9_\-\.~';
|
||||
private static string $charUnreserved = 'a-zA-Z0-9_\-\.~';
|
||||
|
||||
private static string $charSubDelims = '!\$&\'\(\)\*\+,;=';
|
||||
|
||||
private static array $replaceQuery = ['=' => '%3D', '&' => '%26'];
|
||||
|
||||
/**
|
||||
* @var string
|
||||
* uri scheme.
|
||||
*/
|
||||
private static $charSubDelims = '!\$&\'\(\)\*\+,;=';
|
||||
private string $scheme = '';
|
||||
|
||||
/**
|
||||
* @var array
|
||||
* uri user info.
|
||||
*/
|
||||
private static $replaceQuery = ['=' => '%3D', '&' => '%26'];
|
||||
private string $userInfo = '';
|
||||
|
||||
/**
|
||||
* @var string uri scheme
|
||||
* uri host.
|
||||
*/
|
||||
private $scheme = '';
|
||||
private string $host = '';
|
||||
|
||||
/**
|
||||
* @var string uri user info
|
||||
* uri port.
|
||||
*/
|
||||
private $userInfo = '';
|
||||
private ?int $port = null;
|
||||
|
||||
/**
|
||||
* @var string uri host
|
||||
* uri path.
|
||||
*/
|
||||
private $host = '';
|
||||
private string $path = '';
|
||||
|
||||
/**
|
||||
* @var null|int uri port
|
||||
* uri query string.
|
||||
*/
|
||||
private $port;
|
||||
private string $query = '';
|
||||
|
||||
/**
|
||||
* @var string uri path
|
||||
* uri fragment.
|
||||
*/
|
||||
private $path = '';
|
||||
|
||||
/**
|
||||
* @var string uri query string
|
||||
*/
|
||||
private $query = '';
|
||||
|
||||
/**
|
||||
* @var string uri fragment
|
||||
*/
|
||||
private $fragment = '';
|
||||
private string $fragment = '';
|
||||
|
||||
/**
|
||||
* @param string $uri URI to parse
|
||||
*/
|
||||
public function __construct($uri = '')
|
||||
public function __construct(string $uri = '')
|
||||
{
|
||||
// weak type check to also accept null until we can add scalar type hints
|
||||
if ($uri != '') {
|
||||
if ($uri) {
|
||||
$parts = parse_url($uri);
|
||||
if ($parts === false) {
|
||||
throw new \InvalidArgumentException("Unable to parse URI: {$uri}");
|
||||
@ -253,7 +241,7 @@ class Uri implements UriInterface
|
||||
* The value returned MUST be percent-encoded, but MUST NOT double-encode
|
||||
* any characters. To determine what characters to encode, please refer to
|
||||
* RFC 3986, Sections 2 and 3.4.
|
||||
* As an example, if a value in a key/value pair of the query string should
|
||||
* As an example, if a value in a key/value a pair of the query string should
|
||||
* include an ampersand ("&") not intended as a delimiter between values,
|
||||
* that value MUST be passed in encoded form (e.g., "%26") to the instance.
|
||||
*
|
||||
@ -318,7 +306,7 @@ class Uri implements UriInterface
|
||||
* user; an empty string for the user is equivalent to removing user
|
||||
* information.
|
||||
*
|
||||
* @param string $user the user name to use for authority
|
||||
* @param string $user the username to use for authority
|
||||
* @param null|string $password the password associated with $user
|
||||
* @return static a new instance with the specified user information
|
||||
*/
|
||||
@ -448,9 +436,8 @@ class Uri implements UriInterface
|
||||
* @param UriInterface $uri URI to use as a base
|
||||
* @param string $key key to set
|
||||
* @param null|string $value Value to set
|
||||
* @return UriInterface
|
||||
*/
|
||||
public static function withQueryValue(UriInterface $uri, $key, $value)
|
||||
public static function withQueryValue(UriInterface $uri, string $key, ?string $value): UriInterface
|
||||
{
|
||||
$current = $uri->getQuery();
|
||||
|
||||
@ -512,18 +499,12 @@ class Uri implements UriInterface
|
||||
* `file:///` is the more common syntax for the file scheme anyway (Chrome for example redirects to
|
||||
* that format).
|
||||
*
|
||||
* @param string $scheme
|
||||
* @param string $authority
|
||||
* @param string $path
|
||||
* @param string $query
|
||||
* @param string $fragment
|
||||
* @return string
|
||||
* @see https://tools.ietf.org/html/rfc3986#section-5.3
|
||||
*/
|
||||
public static function composeComponents($scheme, $authority, $path, $query, $fragment)
|
||||
public static function composeComponents(string $scheme, string $authority, string $path, string $query, string $fragment)
|
||||
{
|
||||
$uri = '';
|
||||
// weak type checks to also accept null until we can add scalar type hints
|
||||
if ($scheme != '') {
|
||||
$uri .= $scheme . ':';
|
||||
}
|
||||
@ -553,10 +534,8 @@ class Uri implements UriInterface
|
||||
|
||||
/**
|
||||
* Get default port of the current scheme.
|
||||
*
|
||||
* @return null|int
|
||||
*/
|
||||
public function getDefaultPort()
|
||||
public function getDefaultPort(): ?int
|
||||
{
|
||||
return self::$defaultPorts[$this->getScheme()] ?? null;
|
||||
}
|
||||
@ -570,10 +549,10 @@ class Uri implements UriInterface
|
||||
$this->host = self::DEFAULT_HTTP_HOST;
|
||||
}
|
||||
if ($this->getAuthority() === '') {
|
||||
if (strpos($this->path, '//') === 0) {
|
||||
if (str_starts_with($this->path, '//')) {
|
||||
throw new \InvalidArgumentException('The path of a URI without an authority must not start with two slashes "//"');
|
||||
}
|
||||
if ($this->scheme === '' && strpos(explode('/', $this->path, 2)[0], ':') !== false) {
|
||||
if ($this->scheme === '' && str_contains(explode('/', $this->path, 2)[0], ':')) {
|
||||
throw new \InvalidArgumentException('A relative URI must not have a path beginning with a segment containing a colon');
|
||||
}
|
||||
} elseif (isset($this->path[0]) && $this->path[0] !== '/') {
|
||||
@ -589,7 +568,7 @@ class Uri implements UriInterface
|
||||
private function applyParts(array $parts)
|
||||
{
|
||||
$this->scheme = isset($parts['scheme']) ? $this->filterScheme($parts['scheme']) : '';
|
||||
$this->userInfo = isset($parts['user']) ? $parts['user'] : '';
|
||||
$this->userInfo = $parts['user'] ?? '';
|
||||
$this->host = isset($parts['host']) ? $this->filterHost($parts['host']) : '';
|
||||
$this->port = isset($parts['port']) ? $this->filterPort($parts['port']) : null;
|
||||
$this->path = isset($parts['path']) ? $this->filterPath($parts['path']) : '';
|
||||
@ -602,40 +581,20 @@ class Uri implements UriInterface
|
||||
$this->removeDefaultPort();
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $scheme
|
||||
* @throws \InvalidArgumentException if the scheme is invalid
|
||||
* @return string
|
||||
*/
|
||||
private function filterScheme($scheme)
|
||||
private function filterScheme(string $scheme): string
|
||||
{
|
||||
if (! is_string($scheme)) {
|
||||
throw new \InvalidArgumentException('Scheme must be a string');
|
||||
}
|
||||
|
||||
return strtolower($scheme);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $host
|
||||
* @throws \InvalidArgumentException if the host is invalid
|
||||
* @return string
|
||||
*/
|
||||
private function filterHost($host)
|
||||
private function filterHost(string $host): string
|
||||
{
|
||||
if (! is_string($host)) {
|
||||
throw new \InvalidArgumentException('Host must be a string');
|
||||
}
|
||||
|
||||
return strtolower($host);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param null|int|string $port
|
||||
* @throws \InvalidArgumentException if the port is invalid
|
||||
* @return null|int
|
||||
*/
|
||||
private function filterPort($port)
|
||||
private function filterPort($port): ?int
|
||||
{
|
||||
if ($port === null) {
|
||||
return null;
|
||||
@ -652,7 +611,7 @@ class Uri implements UriInterface
|
||||
/**
|
||||
* Remove the port property when the property is a default port.
|
||||
*/
|
||||
private function removeDefaultPort()
|
||||
private function removeDefaultPort(): void
|
||||
{
|
||||
if ($this->port !== null && $this->isDefaultPort()) {
|
||||
$this->port = null;
|
||||
@ -662,16 +621,10 @@ class Uri implements UriInterface
|
||||
/**
|
||||
* Filters the path of a URI.
|
||||
*
|
||||
* @param string $path
|
||||
* @throws \InvalidArgumentException if the path is invalid
|
||||
* @return string
|
||||
*/
|
||||
private function filterPath($path)
|
||||
private function filterPath(string $path): string
|
||||
{
|
||||
if (! is_string($path)) {
|
||||
throw new \InvalidArgumentException('Path must be a string');
|
||||
}
|
||||
|
||||
return preg_replace_callback(
|
||||
'/(?:[^' . self::$charUnreserved . self::$charSubDelims . '%:@\/]++|%(?![A-Fa-f0-9]{2}))/',
|
||||
[
|
||||
@ -684,17 +637,9 @@ class Uri implements UriInterface
|
||||
|
||||
/**
|
||||
* Filters the query string or fragment of a URI.
|
||||
*
|
||||
* @param string $str
|
||||
* @throws \InvalidArgumentException if the query or fragment is invalid
|
||||
* @return string
|
||||
*/
|
||||
private function filterQueryAndFragment($str)
|
||||
private function filterQueryAndFragment(string $str): string
|
||||
{
|
||||
if (! is_string($str)) {
|
||||
throw new \InvalidArgumentException('Query and fragment must be a string');
|
||||
}
|
||||
|
||||
return preg_replace_callback(
|
||||
'/(?:[^' . self::$charUnreserved . self::$charSubDelims . '%:@\/\?]++|%(?![A-Fa-f0-9]{2}))/',
|
||||
[
|
||||
@ -705,10 +650,7 @@ class Uri implements UriInterface
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string
|
||||
*/
|
||||
private function rawurlencodeMatchZero(array $match)
|
||||
private function rawurlencodeMatchZero(array $match): string
|
||||
{
|
||||
return rawurlencode($match[0]);
|
||||
}
|
||||
|
@ -17,7 +17,7 @@ use Psr\Http\Message\RequestInterface;
|
||||
|
||||
class RequestStub extends Request
|
||||
{
|
||||
public static function normalizeParsedBody(array $data = [], ?RequestInterface $request = null)
|
||||
public static function normalizeParsedBody(array $data = [], ?RequestInterface $request = null): array
|
||||
{
|
||||
return parent::normalizeParsedBody($data, $request);
|
||||
}
|
||||
|
@ -11,7 +11,8 @@ declare(strict_types=1);
|
||||
*/
|
||||
namespace Hyperf\Utils\Contracts;
|
||||
|
||||
interface Arrayable
|
||||
use Hyperf\Contract;
|
||||
|
||||
interface Arrayable extends Contract\Arrayable
|
||||
{
|
||||
public function toArray(): array;
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user