Skip to content

Commit

Permalink
Merge pull request #547 from flightphp/stream-response
Browse files Browse the repository at this point in the history
added streaming responses. Fixed JSONP.
  • Loading branch information
fadrian06 authored Feb 21, 2024
2 parents 6861388 + 65c4509 commit 4f37a48
Show file tree
Hide file tree
Showing 14 changed files with 200 additions and 41 deletions.
1 change: 1 addition & 0 deletions .gitattributes
Original file line number Diff line number Diff line change
Expand Up @@ -5,4 +5,5 @@
/.gitignore export-ignore
/phpcs.xml export-ignore
/phpstan.neon export-ignore
/phpstan-baseline.neon export-ignore
/phpunit.xml export-ignore
2 changes: 0 additions & 2 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,5 @@ composer.phar
composer.lock
.phpunit.result.cache
coverage/
.vscode/settings.json
*.sublime*
.vscode/
clover.xml
5 changes: 5 additions & 0 deletions .vscode/settings.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"php.suggest.basic": false,
"editor.detectIndentation": false,
"editor.insertSpaces": true
}
68 changes: 46 additions & 22 deletions flight/Engine.php
Original file line number Diff line number Diff line change
Expand Up @@ -364,49 +364,55 @@ public function path(string $dir): void
/**
* Processes each routes middleware.
*
* @param array<int, callable> $middleware Middleware attached to the route.
* @param array<mixed> $params `$route->params`.
* @param Route $route The route to process the middleware for.
* @param string $event_name If this is the before or after method.
*/
protected function processMiddleware(array $middleware, array $params, string $event_name): bool
protected function processMiddleware(Route $route, string $event_name): bool
{
$at_least_one_middleware_failed = false;

foreach ($middleware as $middleware) {
$middlewares = $event_name === Dispatcher::FILTER_BEFORE ? $route->middleware : array_reverse($route->middleware);
$params = $route->params;

foreach ($middlewares as $middleware) {
$middleware_object = false;

if ($event_name === 'before') {
if ($event_name === Dispatcher::FILTER_BEFORE) {
// can be a callable or a class
$middleware_object = (is_callable($middleware) === true
? $middleware
: (method_exists($middleware, 'before') === true
? [$middleware, 'before']
: (method_exists($middleware, Dispatcher::FILTER_BEFORE) === true
? [$middleware, Dispatcher::FILTER_BEFORE]
: false
)
);
} elseif ($event_name === 'after') {
} elseif ($event_name === Dispatcher::FILTER_AFTER) {
// must be an object. No functions allowed here
if (
is_object($middleware) === true
&& !($middleware instanceof Closure)
&& method_exists($middleware, 'after') === true
&& method_exists($middleware, Dispatcher::FILTER_AFTER) === true
) {
$middleware_object = [$middleware, 'after'];
$middleware_object = [$middleware, Dispatcher::FILTER_AFTER];
}
}

if ($middleware_object === false) {
continue;
}

if ($this->response()->v2_output_buffering === false) {
$use_v3_output_buffering =
$this->response()->v2_output_buffering === false &&
$route->is_streamed === false;

if ($use_v3_output_buffering === true) {
ob_start();
}

// It's assumed if you don't declare before, that it will be assumed as the before method
$middleware_result = $middleware_object($params);

if ($this->response()->v2_output_buffering === false) {
if ($use_v3_output_buffering === true) {
$this->response()->write(ob_get_clean());
}

Expand Down Expand Up @@ -462,16 +468,36 @@ public function _start(): void
$params[] = $route;
}

// If this route is to be streamed, we need to output the headers now
if ($route->is_streamed === true) {
$response->status($route->streamed_headers['status']);
unset($route->streamed_headers['status']);
$response->header('X-Accel-Buffering', 'no');
$response->header('Connection', 'close');
foreach ($route->streamed_headers as $header => $value) {
$response->header($header, $value);
}

// We obviously don't know the content length right now. This must be false.
$response->content_length = false;
$response->sendHeaders();
$response->markAsSent();
}

// Run any before middlewares
if (count($route->middleware) > 0) {
$at_least_one_middleware_failed = $this->processMiddleware($route->middleware, $route->params, 'before');
$at_least_one_middleware_failed = $this->processMiddleware($route, 'before');
if ($at_least_one_middleware_failed === true) {
$failed_middleware_check = true;
break;
}
}

if ($response->v2_output_buffering === false) {
$use_v3_output_buffering =
$this->response()->v2_output_buffering === false &&
$route->is_streamed === false;

if ($use_v3_output_buffering === true) {
ob_start();
}

Expand All @@ -481,18 +507,14 @@ public function _start(): void
$params
);

if ($response->v2_output_buffering === false) {
if ($use_v3_output_buffering === true) {
$response->write(ob_get_clean());
}

// Run any before middlewares
if (count($route->middleware) > 0) {
// process the middleware in reverse order now
$at_least_one_middleware_failed = $this->processMiddleware(
array_reverse($route->middleware),
$route->params,
'after'
);
$at_least_one_middleware_failed = $this->processMiddleware($route, 'after');

if ($at_least_one_middleware_failed === true) {
$failed_middleware_check = true;
Expand Down Expand Up @@ -774,8 +796,10 @@ public function _jsonp(
$this->response()
->status($code)
->header('Content-Type', 'application/javascript; charset=' . $charset)
->write($callback . '(' . $json . ');')
->send();
->write($callback . '(' . $json . ');');
if ($this->response()->v2_output_buffering === true) {
$this->response()->send();
}
}

/**
Expand Down
8 changes: 8 additions & 0 deletions flight/net/Response.php
Original file line number Diff line number Diff line change
Expand Up @@ -386,6 +386,14 @@ public function sent(): bool
return $this->sent;
}

/**
* Marks the response as sent.
*/
public function markAsSent(): void
{
$this->sent = true;
}

/**
* Sends a HTTP response.
*/
Expand Down
33 changes: 28 additions & 5 deletions flight/net/Route.php
Original file line number Diff line number Diff line change
Expand Up @@ -63,10 +63,20 @@ class Route
/**
* The middleware to be applied to the route
*
* @var array<int,callable|object>
* @var array<int, callable|object>
*/
public array $middleware = [];

/** Whether the response for this route should be streamed. */
public bool $is_streamed = false;

/**
* If this route is streamed, the headers to be sent before the response.
*
* @var array<string, mixed>
*/
public array $streamed_headers = [];

/**
* Constructor.
*
Expand Down Expand Up @@ -179,7 +189,7 @@ public function matchAlias(string $alias): bool
/**
* Hydrates the route url with the given parameters
*
* @param array<string,mixed> $params the parameters to pass to the route
* @param array<string, mixed> $params the parameters to pass to the route
*/
public function hydrateUrl(array $params = []): string
{
Expand Down Expand Up @@ -212,9 +222,7 @@ public function setAlias(string $alias): self
/**
* Sets the route middleware
*
* @param array<int,callable>|callable $middleware
*
* @return self
* @param array<int, callable>|callable $middleware
*/
public function addMiddleware($middleware): self
{
Expand All @@ -225,4 +233,19 @@ public function addMiddleware($middleware): self
}
return $this;
}

/**
* This will allow the response for this route to be streamed.
*
* @param array<string, mixed> $headers a key value of headers to set before the stream starts.
*
* @return $this
*/
public function streamWithHeaders(array $headers): self
{
$this->is_streamed = true;
$this->streamed_headers = $headers;

return $this;
}
}
6 changes: 6 additions & 0 deletions phpstan-baseline.neon
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
parameters:
ignoreErrors:
-
message: "#^Parameter \\#2 \\$callback of method flight\\\\core\\\\Dispatcher\\:\\:set\\(\\) expects Closure\\(\\)\\: mixed, array\\{\\$this\\(flight\\\\Engine\\), literal\\-string&non\\-falsy\\-string\\} given\\.$#"
count: 1
path: flight/Engine.php
1 change: 1 addition & 0 deletions phpstan.neon
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
includes:
- vendor/phpstan/phpstan/conf/bleedingEdge.neon
- phpstan-baseline.neon

parameters:
level: 6
Expand Down
32 changes: 32 additions & 0 deletions tests/EngineTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
namespace tests;

use Exception;
use Flight;
use flight\Engine;
use flight\net\Response;
use PHPUnit\Framework\TestCase;
Expand Down Expand Up @@ -307,6 +308,16 @@ public function testJson()
{
$engine = new Engine();
$engine->json(['key1' => 'value1', 'key2' => 'value2']);
$this->assertEquals('application/json; charset=utf-8', $engine->response()->headers()['Content-Type']);
$this->assertEquals(200, $engine->response()->status());
$this->assertEquals('{"key1":"value1","key2":"value2"}', $engine->response()->getBody());
}

public function testJsonV2OutputBuffering()
{
$engine = new Engine();
$engine->response()->v2_output_buffering = true;
$engine->json(['key1' => 'value1', 'key2' => 'value2']);
$this->expectOutputString('{"key1":"value1","key2":"value2"}');
$this->assertEquals('application/json; charset=utf-8', $engine->response()->headers()['Content-Type']);
$this->assertEquals(200, $engine->response()->status());
Expand All @@ -317,6 +328,17 @@ public function testJsonP()
$engine = new Engine();
$engine->request()->query['jsonp'] = 'whatever';
$engine->jsonp(['key1' => 'value1', 'key2' => 'value2']);
$this->assertEquals('application/javascript; charset=utf-8', $engine->response()->headers()['Content-Type']);
$this->assertEquals(200, $engine->response()->status());
$this->assertEquals('whatever({"key1":"value1","key2":"value2"});', $engine->response()->getBody());
}

public function testJsonPV2OutputBuffering()
{
$engine = new Engine();
$engine->response()->v2_output_buffering = true;
$engine->request()->query['jsonp'] = 'whatever';
$engine->jsonp(['key1' => 'value1', 'key2' => 'value2']);
$this->expectOutputString('whatever({"key1":"value1","key2":"value2"});');
$this->assertEquals('application/javascript; charset=utf-8', $engine->response()->headers()['Content-Type']);
$this->assertEquals(200, $engine->response()->status());
Expand All @@ -326,6 +348,16 @@ public function testJsonpBadParam()
{
$engine = new Engine();
$engine->jsonp(['key1' => 'value1', 'key2' => 'value2']);
$this->assertEquals('({"key1":"value1","key2":"value2"});', $engine->response()->getBody());
$this->assertEquals('application/javascript; charset=utf-8', $engine->response()->headers()['Content-Type']);
$this->assertEquals(200, $engine->response()->status());
}

public function testJsonpBadParamV2OutputBuffering()
{
$engine = new Engine();
$engine->response()->v2_output_buffering = true;
$engine->jsonp(['key1' => 'value1', 'key2' => 'value2']);
$this->expectOutputString('({"key1":"value1","key2":"value2"});');
$this->assertEquals('application/javascript; charset=utf-8', $engine->response()->headers()['Content-Type']);
$this->assertEquals(200, $engine->response()->status());
Expand Down
26 changes: 26 additions & 0 deletions tests/FlightTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -278,4 +278,30 @@ public function testHookOutputBufferingV2OutputBuffering()
Flight::start();
$this->assertEquals('hooked before starttest', Flight::response()->getBody());
}

public function testStreamRoute()
{
$response_mock = new class extends Response {
public function setRealHeader(string $header_string, bool $replace = true, int $response_code = 0): Response
{
return $this;
}
};
$mock_response_class_name = get_class($response_mock);
Flight::register('response', $mock_response_class_name);
Flight::route('/stream', function () {
echo 'stream';
})->streamWithHeaders(['Content-Type' => 'text/plain', 'X-Test' => 'test', 'status' => 200 ]);
Flight::request()->url = '/stream';
$this->expectOutputString('stream');
Flight::start();
$this->assertEquals('', Flight::response()->getBody());
$this->assertEquals([
'Content-Type' => 'text/plain',
'X-Test' => 'test',
'X-Accel-Buffering' => 'no',
'Connection' => 'close'
], Flight::response()->getHeaders());
$this->assertEquals(200, Flight::response()->status());
}
}
9 changes: 9 additions & 0 deletions tests/run_all_tests.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
#!/bin/bash

# Run all tests
composer lint
composer beautify
composer phpcs
composer test-coverage
xdg-open http://localhost:8000
composer test-server
Loading

0 comments on commit 4f37a48

Please sign in to comment.