|
30 | 30 | use React\Promise\Deferred; |
31 | 31 | use React\Promise\Promise; |
32 | 32 | use React\Promise\PromiseInterface; |
| 33 | +use React\Socket\ConnectionInterface; |
| 34 | +use React\Socket\Connector; |
33 | 35 | use ReflectionMethod; |
34 | 36 | use ReflectionProperty; |
35 | 37 | use function React\Async\await; |
@@ -718,6 +720,88 @@ public function testRunWillRestartLoopUntilSocketIsClosed(): void |
718 | 720 | $app->run(); |
719 | 721 | } |
720 | 722 |
|
| 723 | + public function testRunWillListenForHttpRequestAndSendBackHttpResponseOverSocket(): void |
| 724 | + { |
| 725 | + $socket = stream_socket_server('127.0.0.1:0'); |
| 726 | + assert(is_resource($socket)); |
| 727 | + $addr = stream_socket_get_name($socket, false); |
| 728 | + assert(is_string($addr)); |
| 729 | + fclose($socket); |
| 730 | + |
| 731 | + $container = new Container([ |
| 732 | + 'X_LISTEN' => $addr |
| 733 | + ]); |
| 734 | + |
| 735 | + $app = new App($container); |
| 736 | + |
| 737 | + Loop::futureTick(function () use ($addr): void { |
| 738 | + $connector = new Connector(); |
| 739 | + $connector->connect($addr)->then(function (ConnectionInterface $connection): void { |
| 740 | + $connection->on('data', function (string $data): void { |
| 741 | + $this->assertStringStartsWith("HTTP/1.0 404 Not Found\r\n", $data); |
| 742 | + }); |
| 743 | + |
| 744 | + // lovely: remove socket server on client connection close to terminate loop |
| 745 | + $connection->on('close', function (): void { |
| 746 | + $resources = get_resources(); |
| 747 | + end($resources); |
| 748 | + prev($resources); |
| 749 | + $socket = prev($resources); |
| 750 | + assert(is_resource($socket)); |
| 751 | + |
| 752 | + Loop::removeReadStream($socket); |
| 753 | + fclose($socket); |
| 754 | + |
| 755 | + Loop::stop(); |
| 756 | + }); |
| 757 | + |
| 758 | + $connection->write("GET /unknown HTTP/1.0\r\nHost: localhost\r\n\r\n"); |
| 759 | + }); |
| 760 | + }); |
| 761 | + |
| 762 | + $this->expectOutputRegex('/' . preg_quote('Listening on http://' . $addr . PHP_EOL, '/') . '.*/'); |
| 763 | + $app->run(); |
| 764 | + } |
| 765 | + |
| 766 | + public function testRunWillReportHttpErrorForInvalidClientRequest(): void |
| 767 | + { |
| 768 | + $socket = stream_socket_server('127.0.0.1:0'); |
| 769 | + assert(is_resource($socket)); |
| 770 | + $addr = stream_socket_get_name($socket, false); |
| 771 | + assert(is_string($addr)); |
| 772 | + fclose($socket); |
| 773 | + |
| 774 | + $container = new Container([ |
| 775 | + 'X_LISTEN' => $addr |
| 776 | + ]); |
| 777 | + |
| 778 | + $app = new App($container); |
| 779 | + |
| 780 | + Loop::futureTick(function () use ($addr): void { |
| 781 | + $connector = new Connector(); |
| 782 | + $connector->connect($addr)->then(function (ConnectionInterface $connection): void { |
| 783 | + $connection->write("not a valid HTTP request\r\n\r\n"); |
| 784 | + |
| 785 | + // lovely: remove socket server on client connection close to terminate loop |
| 786 | + $connection->on('close', function (): void { |
| 787 | + $resources = get_resources(); |
| 788 | + end($resources); |
| 789 | + prev($resources); |
| 790 | + $socket = prev($resources); |
| 791 | + assert(is_resource($socket)); |
| 792 | + |
| 793 | + Loop::removeReadStream($socket); |
| 794 | + fclose($socket); |
| 795 | + |
| 796 | + Loop::stop(); |
| 797 | + }); |
| 798 | + }); |
| 799 | + }); |
| 800 | + |
| 801 | + $this->expectOutputRegex('/HTTP error: .*' . PHP_EOL . '$/'); |
| 802 | + $app->run(); |
| 803 | + } |
| 804 | + |
721 | 805 | /** |
722 | 806 | * @requires function pcntl_signal |
723 | 807 | * @requires function posix_kill |
|
0 commit comments