ServerRequestFactory.php 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. <?php
  2. declare(strict_types=1);
  3. namespace PhpMyAdmin\Http\Factory;
  4. use GuzzleHttp\Psr7\HttpFactory;
  5. use Laminas\Diactoros\ServerRequestFactory as LaminasServerRequestFactory;
  6. use Laminas\Diactoros\UriFactory as LaminasUriFactory;
  7. use Nyholm\Psr7\Factory\Psr17Factory;
  8. use PhpMyAdmin\Http\ServerRequest;
  9. use Psr\Http\Message\ServerRequestFactoryInterface;
  10. use Psr\Http\Message\ServerRequestInterface;
  11. use Psr\Http\Message\UriFactoryInterface;
  12. use Psr\Http\Message\UriInterface;
  13. use Slim\Psr7\Factory\ServerRequestFactory as SlimServerRequestFactory;
  14. use Slim\Psr7\Factory\UriFactory as SlimUriFactory;
  15. use function class_exists;
  16. use function count;
  17. use function current;
  18. use function explode;
  19. use function function_exists;
  20. use function getallheaders;
  21. use function in_array;
  22. use function is_numeric;
  23. use function is_string;
  24. use function parse_url;
  25. use function preg_match;
  26. use function strpos;
  27. use function strstr;
  28. use function substr;
  29. use const PHP_URL_QUERY;
  30. class ServerRequestFactory
  31. {
  32. /** @var ServerRequestFactoryInterface */
  33. private $serverRequestFactory;
  34. /** @var UriFactoryInterface */
  35. private $uriFactory;
  36. public function __construct(
  37. ?ServerRequestFactoryInterface $serverRequestFactory = null,
  38. ?UriFactoryInterface $uriFactory = null
  39. ) {
  40. $this->serverRequestFactory = $serverRequestFactory ?? $this->createServerRequestFactory();
  41. $this->uriFactory = $uriFactory ?? $this->createUriFactory();
  42. }
  43. private function createServerRequestFactory(): ServerRequestFactoryInterface
  44. {
  45. if (class_exists(Psr17Factory::class)) {
  46. /** @var ServerRequestFactoryInterface $factory */
  47. $factory = new Psr17Factory();
  48. } elseif (class_exists(HttpFactory::class)) {
  49. /** @var ServerRequestFactoryInterface $factory */
  50. $factory = new HttpFactory();
  51. } elseif (class_exists(LaminasServerRequestFactory::class)) {
  52. /** @var ServerRequestFactoryInterface $factory */
  53. $factory = new LaminasServerRequestFactory();
  54. } else {
  55. $factory = new SlimServerRequestFactory();
  56. }
  57. return $factory;
  58. }
  59. private function createUriFactory(): UriFactoryInterface
  60. {
  61. if (class_exists(Psr17Factory::class)) {
  62. /** @var UriFactoryInterface $factory */
  63. $factory = new Psr17Factory();
  64. } elseif (class_exists(HttpFactory::class)) {
  65. /** @var UriFactoryInterface $factory */
  66. $factory = new HttpFactory();
  67. } elseif (class_exists(LaminasUriFactory::class)) {
  68. /** @var UriFactoryInterface $factory */
  69. $factory = new LaminasUriFactory();
  70. } else {
  71. $factory = new SlimUriFactory();
  72. }
  73. return $factory;
  74. }
  75. public static function createFromGlobals(): ServerRequest
  76. {
  77. if (class_exists(SlimServerRequestFactory::class)) {
  78. /** @psalm-suppress InternalMethod */
  79. $serverRequest = SlimServerRequestFactory::createFromGlobals();
  80. } elseif (class_exists(LaminasServerRequestFactory::class)) {
  81. /** @var ServerRequestInterface $serverRequest */
  82. $serverRequest = LaminasServerRequestFactory::fromGlobals();
  83. } else {
  84. $creator = new self();
  85. $serverRequest = self::createServerRequestFromGlobals($creator);
  86. }
  87. return new ServerRequest($serverRequest);
  88. }
  89. /**
  90. * @return array<string, string>
  91. */
  92. protected function getallheaders(): array
  93. {
  94. /** @var array<string, string> $headers */
  95. $headers = function_exists('getallheaders') ? getallheaders() : [];
  96. return $headers;
  97. }
  98. private static function createServerRequestFromGlobals(self $creator): ServerRequestInterface
  99. {
  100. $serverRequest = $creator->serverRequestFactory->createServerRequest(
  101. $_SERVER['REQUEST_METHOD'] ?? 'GET',
  102. $creator->createUriFromGlobals($_SERVER),
  103. $_SERVER
  104. );
  105. foreach ($creator->getallheaders() as $name => $value) {
  106. $serverRequest = $serverRequest->withAddedHeader($name, $value);
  107. }
  108. $serverRequest = $serverRequest->withQueryParams($_GET);
  109. if ($serverRequest->getMethod() !== 'POST') {
  110. return $serverRequest;
  111. }
  112. $contentType = '';
  113. foreach ($serverRequest->getHeader('Content-Type') as $headerValue) {
  114. $contentType = current(explode(';', $headerValue));
  115. }
  116. if (in_array($contentType, ['application/x-www-form-urlencoded', 'multipart/form-data'], true)) {
  117. return $serverRequest->withParsedBody($_POST);
  118. }
  119. return $serverRequest;
  120. }
  121. /**
  122. * Create new Uri from environment.
  123. *
  124. * Initially based on the \Slim\Psr7\Factory\UriFactory::createFromGlobals() implementation.
  125. *
  126. * @param mixed[] $server
  127. */
  128. private function createUriFromGlobals(array $server): UriInterface
  129. {
  130. $uri = $this->uriFactory->createUri('');
  131. $uri = $uri->withScheme(! isset($server['HTTPS']) || $server['HTTPS'] === 'off' ? 'http' : 'https');
  132. if (isset($server['PHP_AUTH_USER']) && is_string($server['PHP_AUTH_USER']) && $server['PHP_AUTH_USER'] !== '') {
  133. $uri = $uri->withUserInfo(
  134. $server['PHP_AUTH_USER'],
  135. isset($server['PHP_AUTH_PW']) && is_string($server['PHP_AUTH_PW']) ? $server['PHP_AUTH_PW'] : null
  136. );
  137. }
  138. if (isset($server['HTTP_HOST']) && is_string($server['HTTP_HOST'])) {
  139. $uri = $uri->withHost($server['HTTP_HOST']);
  140. } elseif (isset($server['SERVER_NAME']) && is_string($server['SERVER_NAME'])) {
  141. $uri = $uri->withHost($server['SERVER_NAME']);
  142. }
  143. if (isset($server['SERVER_PORT']) && is_numeric($server['SERVER_PORT']) && $server['SERVER_PORT'] >= 1) {
  144. $uri = $uri->withPort((int) $server['SERVER_PORT']);
  145. } else {
  146. $uri = $uri->withPort($uri->getScheme() === 'https' ? 443 : 80);
  147. }
  148. if (preg_match('/^(\[[a-fA-F0-9:.]+])(:\d+)?\z/', $uri->getHost(), $matches)) {
  149. $uri = $uri->withHost($matches[1]);
  150. if (isset($matches[2])) {
  151. $uri = $uri->withPort((int) substr($matches[2], 1));
  152. }
  153. } else {
  154. $pos = strpos($uri->getHost(), ':');
  155. if ($pos !== false) {
  156. $port = (int) substr($uri->getHost(), $pos + 1);
  157. $host = (string) strstr($uri->getHost(), ':', true);
  158. $uri = $uri->withHost($host)->withPort($port);
  159. }
  160. }
  161. if (isset($server['QUERY_STRING']) && is_string($server['QUERY_STRING'])) {
  162. $uri = $uri->withQuery($server['QUERY_STRING']);
  163. }
  164. if (isset($server['REQUEST_URI']) && is_string($server['REQUEST_URI'])) {
  165. $uriFragments = explode('?', $server['REQUEST_URI']);
  166. $uri = $uri->withPath($uriFragments[0]);
  167. if ($uri->getQuery() === '' && count($uriFragments) > 1) {
  168. $query = parse_url('https://www.example.com' . $server['REQUEST_URI'], PHP_URL_QUERY);
  169. if (is_string($query) && $query !== '') {
  170. $uri = $uri->withQuery($query);
  171. }
  172. }
  173. }
  174. return $uri;
  175. }
  176. }