server.inc 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. <?php declare(strict_types=1);
  2. function http_server_skipif() {
  3. if (!function_exists('pcntl_fork')) die('skip pcntl_fork() not available');
  4. if (!function_exists('posix_kill')) die('skip posix_kill() not available');
  5. if (!stream_socket_server('tcp://localhost:0')) die('skip stream_socket_server() failed');
  6. }
  7. function http_server_init(&$output = null) {
  8. pcntl_alarm(60);
  9. $server = stream_socket_server('tcp://localhost:0', $errno, $errstr);
  10. if (!$server) {
  11. return false;
  12. }
  13. if ($output === null) {
  14. $output = tmpfile();
  15. if ($output === false) {
  16. return false;
  17. }
  18. }
  19. $pid = pcntl_fork();
  20. if ($pid == -1) {
  21. die('could not fork');
  22. } else if ($pid) {
  23. return [
  24. 'pid' => $pid,
  25. 'uri' => 'http://' . stream_socket_get_name($server, false),
  26. ];
  27. }
  28. return $server;
  29. }
  30. /* Minimal HTTP server with predefined responses.
  31. *
  32. * $socket_string is the socket to create and listen on (e.g. tcp://127.0.0.1:1234)
  33. * $files is an iterable of files or callable generator yielding files.
  34. * containing N responses for N expected requests. Server dies after N requests.
  35. * $output is a stream on which everything sent by clients is written to
  36. */
  37. function http_server($files, &$output = null) {
  38. if (!is_resource($server = http_server_init($output))) {
  39. return $server;
  40. }
  41. if (is_callable($files)) {
  42. $files = $files($server);
  43. }
  44. foreach($files as $file) {
  45. $sock = stream_socket_accept($server);
  46. if (!$sock) {
  47. exit(1);
  48. }
  49. // read headers
  50. $content_length = 0;
  51. stream_set_blocking($sock, false);
  52. while (!feof($sock)) {
  53. list($r, $w, $e) = array(array($sock), null, null);
  54. if (!stream_select($r, $w, $e, 1)) continue;
  55. $line = stream_get_line($sock, 8192, "\r\n");
  56. if ($line === '') {
  57. fwrite($output, "\r\n");
  58. break;
  59. } else if ($line !== false) {
  60. fwrite($output, "$line\r\n");
  61. if (preg_match('#^Content-Length\s*:\s*([[:digit:]]+)\s*$#i', $line, $matches)) {
  62. $content_length = (int) $matches[1];
  63. }
  64. }
  65. }
  66. stream_set_blocking($sock, true);
  67. // read content
  68. if ($content_length > 0) {
  69. stream_copy_to_stream($sock, $output, $content_length);
  70. }
  71. // send response
  72. $fd = fopen($file, 'rb');
  73. stream_copy_to_stream($fd, $sock);
  74. fclose($sock);
  75. }
  76. exit(0);
  77. }
  78. function http_server_sleep($micro_seconds = 500000)
  79. {
  80. if (!is_resource($server = http_server_init($output))) {
  81. return $server;
  82. }
  83. $sock = stream_socket_accept($server);
  84. if (!$sock) {
  85. exit(1);
  86. }
  87. usleep($micro_seconds);
  88. fclose($sock);
  89. exit(0);
  90. }
  91. function http_server_kill(int $pid) {
  92. posix_kill($pid, SIGTERM);
  93. pcntl_waitpid($pid, $status);
  94. }