-
-
Notifications
You must be signed in to change notification settings - Fork 38
Commit
- Loading branch information
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,122 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Flow\Bridge\Symfony\HttpFoundation; | ||
|
||
use Flow\ETL\{Extractor, Transformation, Transformations}; | ||
use Symfony\Component\HttpFoundation\{HeaderUtils, Response}; | ||
|
||
/** | ||
* FlowStreamedResponse builder. | ||
*/ | ||
final class DataStream | ||
{ | ||
/** | ||
* @var array<string, string> | ||
*/ | ||
private array $headers = [ | ||
'Cache-Control' => 'no-store, no-cache, must-revalidate, private', | ||
'X-Accel-Buffering' => 'no', // provides support for Nginx | ||
'Pragma' => 'no-cache', // Backward compatibility for HTTP/1.0 | ||
]; | ||
|
||
private ?Output $output = null; | ||
|
||
private int $status = Response::HTTP_OK; | ||
|
||
/** | ||
* @var array<Transformation> | ||
*/ | ||
private array $transformations = []; | ||
|
||
public function __construct(private readonly Extractor $extractor) | ||
{ | ||
} | ||
|
||
public static function open(Extractor $extractor) : self | ||
{ | ||
return new self($extractor); | ||
} | ||
|
||
/** | ||
* Send the data stream to the output. | ||
*/ | ||
public function sendTo(Output $output) : FlowStreamedResponse | ||
{ | ||
$this->output = $output; | ||
|
||
$this->headers['Content-Type'] = $this->output->type()->toContentTypeHeader(); | ||
|
||
return new FlowStreamedResponse( | ||
$this->extractor, | ||
$this->output, | ||
\count($this->transformations) ? new Transformations(...$this->transformations) : new Transformations(), | ||
$this->status, | ||
$this->headers | ||
); | ||
} | ||
|
||
/** | ||
* Apply transformations to the data stream. | ||
* Transformations are applied in the order they are passed. | ||
* Transformations are applied on the fly, while streaming the data, this means | ||
* that any resource expensive transformations like for example aggregations or sorting | ||
* might significantly slow down the streaming process or even cause out of memory errors. | ||
*/ | ||
public function transform(Transformation ...$transformations) : self | ||
Check warning on line 67 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
{ | ||
$this->transformations = $transformations; | ||
Check warning on line 69 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
|
||
return $this; | ||
Check warning on line 71 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
} | ||
|
||
/** | ||
* Set the filename for the response. | ||
* If the attachment flag is set to true, the response will be treated as an attachment meaning that | ||
* the browser will prompt the user to download the file. | ||
*/ | ||
public function underFilename(string $name, bool $attachment = true) : self | ||
{ | ||
$this->headers['Content-Disposition'] = HeaderUtils::makeDisposition( | ||
$attachment ? HeaderUtils::DISPOSITION_ATTACHMENT : HeaderUtils::DISPOSITION_INLINE, | ||
$name | ||
); | ||
|
||
return $this; | ||
} | ||
|
||
/** | ||
* Set additional headers. | ||
* Headers are merged with the default headers. | ||
*/ | ||
public function withHeaders(array $headers) : self | ||
Check warning on line 93 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
{ | ||
$this->headers = array_merge($this->headers, $headers); | ||
Check warning on line 95 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
|
||
return $this; | ||
Check warning on line 97 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
} | ||
|
||
/** | ||
* Remove a specific header if it exists. | ||
* If the header does not exist, nothing happens. | ||
*/ | ||
public function withoutHeader(string $name) : self | ||
Check warning on line 104 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
{ | ||
if (\array_key_exists($name, $this->headers)) { | ||
unset($this->headers[$name]); | ||
Check warning on line 107 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
} | ||
|
||
return $this; | ||
Check warning on line 110 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
} | ||
|
||
/** | ||
* Set the HTTP status code. Default is 200. | ||
*/ | ||
public function withStatus(int $status) : self | ||
Check warning on line 116 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
{ | ||
$this->status = $status; | ||
Check warning on line 118 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
|
||
return $this; | ||
Check warning on line 120 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/DataStream.php
|
||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Flow\Bridge\Symfony\HttpFoundation\Transformation; | ||
|
||
use function Flow\ETL\DSL\int_entry; | ||
use Flow\Bridge\Symfony\HttpFoundation\Transformation\AddRowIndex\StartFrom; | ||
use Flow\ETL\{DataFrame, Row, Transformation}; | ||
|
||
final readonly class AddRowIndex implements Transformation | ||
{ | ||
public function __construct(private string $indexColumn = 'index', private StartFrom $startFrom = StartFrom::ZERO) | ||
{ | ||
} | ||
|
||
public function transform(DataFrame $dataFrame) : DataFrame | ||
{ | ||
$index = $this->startFrom === StartFrom::ZERO ? 0 : 1; | ||
|
||
return $dataFrame->map(function (Row $row) use (&$index) { | ||
$row = $row->add(int_entry($this->indexColumn, $index)); | ||
$index++; | ||
|
||
return $row; | ||
}); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Flow\Bridge\Symfony\HttpFoundation\Transformation\AddRowIndex; | ||
|
||
enum StartFrom | ||
{ | ||
case ONE; | ||
case ZERO; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Flow\Bridge\Symfony\HttpFoundation\Transformation; | ||
|
||
use Flow\ETL\{DataFrame, Transformation}; | ||
use Flow\Filesystem\Exception\InvalidArgumentException; | ||
|
||
/** | ||
* Sets batch size for DataFrame. | ||
* Small batch size can be useful when processing large data sets since only one row is processed at a time. | ||
* This means that while processing large data sets, memory usage is kept low. | ||
* | ||
* Normally flow allows to use batch size -1 (which means no batches) but it defeats the purpose of using this transformation on | ||
* Data Streams. | ||
*/ | ||
final readonly class BatchSize implements Transformation | ||
{ | ||
/** | ||
* @param int<1, max> $size | ||
*/ | ||
public function __construct(private int $size) | ||
{ | ||
if ($size < 1) { | ||
throw new InvalidArgumentException('Batch size must be greater than 0'); | ||
Check warning on line 26 in src/bridge/symfony/http-foundation/src/Flow/Bridge/Symfony/HttpFoundation/Transformation/BatchSize.php
|
||
} | ||
} | ||
|
||
public function transform(DataFrame $dataFrame) : DataFrame | ||
{ | ||
return $dataFrame->batchSize($this->size); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,65 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Flow\Bridge\Symfony\HttpFoundation\Tests\Unit\Transformation; | ||
|
||
use function Flow\ETL\DSL\{df, from_array}; | ||
use Flow\Bridge\Symfony\HttpFoundation\Transformation\AddRowIndex; | ||
use Flow\Bridge\Symfony\HttpFoundation\Transformation\AddRowIndex\StartFrom; | ||
use Flow\ETL\Tests\FlowTestCase; | ||
|
||
final class AddRowIndexTest extends FlowTestCase | ||
{ | ||
public function test_adding_row_index_to_each_row() : void | ||
{ | ||
$rows = df() | ||
->read(from_array( | ||
[ | ||
['id' => 1, 'name' => 'John Doe', 'salary' => 7000, 'currency' => 'USD'], | ||
['id' => 2, 'name' => 'Jane Doe', 'salary' => 8000, 'currency' => 'USD'], | ||
['id' => 3, 'name' => 'John Smith', 'salary' => 9000, 'currency' => 'USD'], | ||
['id' => 4, 'name' => 'Jane Smith', 'salary' => 10000, 'currency' => 'USD'], | ||
] | ||
)) | ||
->with(new AddRowIndex()) | ||
->fetch() | ||
->toArray(); | ||
|
||
self::assertEquals( | ||
[ | ||
['index' => 0, 'id' => 1, 'name' => 'John Doe', 'salary' => 7000, 'currency' => 'USD'], | ||
['index' => 1, 'id' => 2, 'name' => 'Jane Doe', 'salary' => 8000, 'currency' => 'USD'], | ||
['index' => 2, 'id' => 3, 'name' => 'John Smith', 'salary' => 9000, 'currency' => 'USD'], | ||
['index' => 3, 'id' => 4, 'name' => 'Jane Smith', 'salary' => 10000, 'currency' => 'USD'], | ||
], | ||
$rows | ||
); | ||
} | ||
|
||
public function test_adding_row_index_to_each_row_starting_from_1() : void | ||
{ | ||
$rows = df() | ||
->read(from_array( | ||
[ | ||
['id' => 1, 'name' => 'John Doe', 'salary' => 7000, 'currency' => 'USD'], | ||
['id' => 2, 'name' => 'Jane Doe', 'salary' => 8000, 'currency' => 'USD'], | ||
['id' => 3, 'name' => 'John Smith', 'salary' => 9000, 'currency' => 'USD'], | ||
['id' => 4, 'name' => 'Jane Smith', 'salary' => 10000, 'currency' => 'USD'], | ||
] | ||
)) | ||
->with(new AddRowIndex(startFrom: StartFrom::ONE)) | ||
->fetch() | ||
->toArray(); | ||
|
||
self::assertEquals( | ||
[ | ||
['index' => 1, 'id' => 1, 'name' => 'John Doe', 'salary' => 7000, 'currency' => 'USD'], | ||
['index' => 2, 'id' => 2, 'name' => 'Jane Doe', 'salary' => 8000, 'currency' => 'USD'], | ||
['index' => 3, 'id' => 3, 'name' => 'John Smith', 'salary' => 9000, 'currency' => 'USD'], | ||
['index' => 4, 'id' => 4, 'name' => 'Jane Smith', 'salary' => 10000, 'currency' => 'USD'], | ||
], | ||
$rows | ||
); | ||
} | ||
} |