Initial version
This commit is contained in:
parent
072f9b2fa0
commit
57d3efe1df
8 changed files with 375 additions and 0 deletions
8
.woodpecker/code-style.yml
Normal file
8
.woodpecker/code-style.yml
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
pipeline:
|
||||||
|
code-style:
|
||||||
|
image: composer
|
||||||
|
commands:
|
||||||
|
- composer global config repositories.repo-name vcs https://git.digital-competence.de/Packages/php-codesniffer
|
||||||
|
- composer global config --no-plugins allow-plugins.dealerdirect/phpcodesniffer-composer-installer true
|
||||||
|
- composer global require digicomp/php-codesniffer:@dev
|
||||||
|
- composer global exec -- phpcs --runtime-set ignore_warnings_on_exit 1 --standard=DigiComp Classes/ Tests/
|
28
.woodpecker/functional-tests.yml
Normal file
28
.woodpecker/functional-tests.yml
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
workspace:
|
||||||
|
base: /woodpecker
|
||||||
|
path: package
|
||||||
|
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- FLOW_VERSION: 7.3
|
||||||
|
PHP_VERSION: 8.1
|
||||||
|
- FLOW_VERSION: 8.3
|
||||||
|
PHP_VERSION: 8.2
|
||||||
|
|
||||||
|
pipeline:
|
||||||
|
functional-tests:
|
||||||
|
image: "thecodingmachine/php:${PHP_VERSION}-v4-cli"
|
||||||
|
environment:
|
||||||
|
# Enable the PDO_SQLITE extension
|
||||||
|
- "PHP_EXTENSION_PDO_SQLITE=1"
|
||||||
|
- "FLOW_VERSION=${FLOW_VERSION}"
|
||||||
|
- "NEOS_BUILD_DIR=/woodpecker/Build-${FLOW_VERSION}"
|
||||||
|
commands:
|
||||||
|
- "sudo mkdir $NEOS_BUILD_DIR"
|
||||||
|
- "sudo chown -R docker:docker $NEOS_BUILD_DIR"
|
||||||
|
- "cd $NEOS_BUILD_DIR"
|
||||||
|
- "composer create-project --no-install neos/flow-base-distribution:^$FLOW_VERSION ."
|
||||||
|
- "composer config repositories.repo-name path /woodpecker/package"
|
||||||
|
- "composer remove --dev --no-update neos/behat || composer remove --no-update neos/behat"
|
||||||
|
- "composer require digicomp/attachment-view-utility:@dev"
|
||||||
|
- "bin/phpunit --configuration Build/BuildEssentials/PhpUnit/FunctionalTests.xml Packages/Application/DigiComp.AttachmentViewUtility/Tests/Functional"
|
61
Classes/AttachmentViewTrait.php
Normal file
61
Classes/AttachmentViewTrait.php
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DigiComp\AttachmentViewUtility;
|
||||||
|
|
||||||
|
use cardinalby\ContentDisposition\ContentDisposition;
|
||||||
|
use GuzzleHttp\Psr7\Response;
|
||||||
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
use Psr\Http\Message\StreamInterface;
|
||||||
|
|
||||||
|
trait AttachmentViewTrait
|
||||||
|
{
|
||||||
|
abstract protected function getAttachmentName(): string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string|resource|StreamInterface|null
|
||||||
|
*/
|
||||||
|
abstract protected function getAttachmentContent();
|
||||||
|
|
||||||
|
abstract protected function getAttachmentMimeType(): string;
|
||||||
|
|
||||||
|
protected function addOptionsForAttachment(): void
|
||||||
|
{
|
||||||
|
if (\property_exists($this, 'supportedOptions')) {
|
||||||
|
$this->supportedOptions['attachmentCharset'] = [
|
||||||
|
'utf-8',
|
||||||
|
'Charset of the content or FALSE if you want to suppress the information in header',
|
||||||
|
'string|false',
|
||||||
|
];
|
||||||
|
$this->supportedOptions['attachmentDisposition'] = [
|
||||||
|
'attachment',
|
||||||
|
'One of "inline" or "attachment"',
|
||||||
|
'string',
|
||||||
|
];
|
||||||
|
} else {
|
||||||
|
throw new \RuntimeException('supported option could not be set', 1697552694);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function render(): ResponseInterface
|
||||||
|
{
|
||||||
|
if ($this->options['attachmentCharset'] === false) {
|
||||||
|
$charset = '';
|
||||||
|
} else {
|
||||||
|
$charset = '; charset=' . $this->options['attachmentCharset'];
|
||||||
|
}
|
||||||
|
return new Response(
|
||||||
|
200,
|
||||||
|
[
|
||||||
|
'Content-Disposition' => ContentDisposition::create(
|
||||||
|
$this->getAttachmentName(),
|
||||||
|
true,
|
||||||
|
$this->options['attachmentDisposition']
|
||||||
|
)->format(),
|
||||||
|
'Content-Type' => $this->getAttachmentMimeType() . $charset,
|
||||||
|
],
|
||||||
|
$this->getAttachmentContent()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
66
Classes/EelFilenameTrait.php
Normal file
66
Classes/EelFilenameTrait.php
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DigiComp\AttachmentViewUtility;
|
||||||
|
|
||||||
|
use Neos\Eel\CompilingEvaluator;
|
||||||
|
use Neos\Eel\Context;
|
||||||
|
use Neos\Eel\Helper\ArrayHelper;
|
||||||
|
use Neos\Eel\Helper\StringHelper;
|
||||||
|
use Neos\Flow\Annotations as Flow;
|
||||||
|
use Neos\Flow\I18n\EelHelper\TranslationHelper;
|
||||||
|
use Neos\FluidAdaptor\View\TemplateView;
|
||||||
|
use Neos\Utility\TypeHandling;
|
||||||
|
|
||||||
|
trait EelFilenameTrait
|
||||||
|
{
|
||||||
|
protected string $eelExpressionOptionKey = 'filenameEelExpression';
|
||||||
|
|
||||||
|
protected function addOptionFilenameEelExpression(): void
|
||||||
|
{
|
||||||
|
if (\property_exists($this, 'supportedOptions')) {
|
||||||
|
$this->supportedOptions['filenameEelExpression'] = [
|
||||||
|
null,
|
||||||
|
'Callable which creates a filename from variables',
|
||||||
|
'string',
|
||||||
|
true,
|
||||||
|
];
|
||||||
|
} else {
|
||||||
|
throw new \RuntimeException('supported option could not be set', 1697552694);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getAttachmentName(): string
|
||||||
|
{
|
||||||
|
if ($this instanceof TemplateView) {
|
||||||
|
$variables = $this->getRenderingContext()->getVariableProvider()->getAll();
|
||||||
|
} elseif (\property_exists($this, 'variables')) {
|
||||||
|
$variables = $this->variables;
|
||||||
|
} else {
|
||||||
|
throw new \RuntimeException(
|
||||||
|
'No variables can be detected for this kind of view: ' . TypeHandling::getTypeForValue($this),
|
||||||
|
1697550214
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (!\property_exists($this, 'options')) {
|
||||||
|
throw new \RuntimeException('Your view options could not be found', 1697550440);
|
||||||
|
}
|
||||||
|
$expression = $this->options[$this->eelExpressionOptionKey];
|
||||||
|
$context = new Context(\array_merge($variables, [
|
||||||
|
'Array' => new ArrayHelper(),
|
||||||
|
'String' => new StringHelper(),
|
||||||
|
'Translation' => new TranslationHelper(),
|
||||||
|
]));
|
||||||
|
$this->emitFilenameEelExpressionContext($context);
|
||||||
|
return (new CompilingEvaluator())->evaluate($expression, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @SuppressWarnings("unused")
|
||||||
|
*/
|
||||||
|
#[Flow\Signal]
|
||||||
|
protected function emitFilenameEelExpressionContext(Context $context): void
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
58
README.md
Normal file
58
README.md
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
# DigiComp.AttachmentViewUtility
|
||||||
|
|
||||||
|
This package helps with recurring tasks of creating views for `neos/flow` projects.
|
||||||
|
|
||||||
|
It delivers two traits:
|
||||||
|
1. `AttachmentViewTrait` creates views which returns downloadable resources
|
||||||
|
2. `EelFilenameTrait` allows to set the filename with an eel expression, which got the view variables as context
|
||||||
|
|
||||||
|
They can be used together or standalone - but the filename trait does not make sense, if your view is not delivered by attachment.
|
||||||
|
|
||||||
|
The `EelContext` contains the `String`, `Array` and `Translation` - Helpers. If you need more, you can connect to the `YourView::filenameEelExpressionContext` signal, which gets the context as argument. That way you can extend the context with whatever you need.
|
||||||
|
|
||||||
|
## Example
|
||||||
|
|
||||||
|
SimpleAttachmentView.php:
|
||||||
|
```php
|
||||||
|
class SimpleAttachmentView extends AbstractView
|
||||||
|
{
|
||||||
|
use AttachmentViewTrait;
|
||||||
|
use EelFilenameTrait;
|
||||||
|
|
||||||
|
public function __construct(array $options = [])
|
||||||
|
{
|
||||||
|
$this->addOptionFilenameEelExpression();
|
||||||
|
$this->addOptionsForAttachment();
|
||||||
|
parent::__construct($options);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getAttachmentContent()
|
||||||
|
{
|
||||||
|
return 'Hello ' . $this->variables['name'];
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getAttachmentMimeType(): string
|
||||||
|
{
|
||||||
|
return 'text/plain';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Controller:
|
||||||
|
```php
|
||||||
|
class DefaultController extends ActionController {
|
||||||
|
public function downloadAction()
|
||||||
|
{
|
||||||
|
$this->view->assign('name', 'World');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Views.yaml:
|
||||||
|
```yaml
|
||||||
|
-
|
||||||
|
requestFilter: "isController('Default') && isAction('download')"
|
||||||
|
viewObjectName: "Acme\\Vendor\\SimpleAttachmentView"
|
||||||
|
options:
|
||||||
|
filenameEelExpression: "'hello-' + name + '.txt"
|
||||||
|
```
|
82
Tests/Functional/AttachmentViewTraitTest.php
Normal file
82
Tests/Functional/AttachmentViewTraitTest.php
Normal file
|
@ -0,0 +1,82 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DigiComp\AttachmentViewUtility\Tests\Functional;
|
||||||
|
|
||||||
|
use DigiComp\AttachmentViewUtility\Tests\Functional\Fixtures\SimpleAttachmentTemplateView;
|
||||||
|
use Neos\Eel\Context;
|
||||||
|
use Neos\Flow\SignalSlot\Dispatcher;
|
||||||
|
use Neos\Flow\Tests\FunctionalTestCase;
|
||||||
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
|
||||||
|
class AttachmentViewTraitTest extends FunctionalTestCase
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @test
|
||||||
|
*/
|
||||||
|
public function itCreatesAttachmentResponses(): void
|
||||||
|
{
|
||||||
|
$view = new SimpleAttachmentTemplateView([
|
||||||
|
'templateSource' => 'TollesTemplate! {testVar}',
|
||||||
|
'filenameEelExpression' => 'testVar + ".txt"',
|
||||||
|
'attachmentCharset' => 'iso-8859-1'
|
||||||
|
]);
|
||||||
|
$view->assign('testVar', '£ and € rates');
|
||||||
|
|
||||||
|
$result = $view->render();
|
||||||
|
static::assertInstanceOf(ResponseInterface::class, $result);
|
||||||
|
static::assertEquals(
|
||||||
|
'attachment; filename="£ and ? rates.txt"; filename*=UTF-8\'\'%C2%A3%20and%20%E2%82%AC%20rates.txt',
|
||||||
|
$result->getHeaderLine('Content-Disposition')
|
||||||
|
);
|
||||||
|
static::assertEquals('text/plain; charset=iso-8859-1', $result->getHeaderLine('Content-Type'));
|
||||||
|
static::assertEquals('TollesTemplate! £ and € rates', (string)$result->getBody());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @test
|
||||||
|
*/
|
||||||
|
public function itIsPossibleToSuppressCharset(): void
|
||||||
|
{
|
||||||
|
$view = new SimpleAttachmentTemplateView([
|
||||||
|
'templateSource' => 'TollesTemplate! {testVar}',
|
||||||
|
'filenameEelExpression' => 'testVar + ".txt"',
|
||||||
|
'attachmentCharset' => false
|
||||||
|
]);
|
||||||
|
$view->assign('testVar', 'WORLD');
|
||||||
|
$result = $view->render();
|
||||||
|
static::assertEquals('text/plain', $result->getHeaderLine('Content-Type'));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @test
|
||||||
|
*/
|
||||||
|
public function itIsPossibleToExtendTheContextBySignalSlot(): void
|
||||||
|
{
|
||||||
|
$view = new SimpleAttachmentTemplateView([
|
||||||
|
'templateSource' => 'TollesTemplate! {testVar}',
|
||||||
|
'filenameEelExpression' => 'greet(testVar) + ".txt"',
|
||||||
|
'attachmentCharset' => false
|
||||||
|
]);
|
||||||
|
$view->assign('testVar', 'WORLD');
|
||||||
|
$dispatcher = $this->objectManager->get(Dispatcher::class);
|
||||||
|
$dispatcher->connect(
|
||||||
|
SimpleAttachmentTemplateView::class,
|
||||||
|
'filenameEelExpressionContext',
|
||||||
|
function (Context $eelContext) {
|
||||||
|
$eelContext->push(
|
||||||
|
function ($name) {
|
||||||
|
return 'Hello ' . $name;
|
||||||
|
},
|
||||||
|
'greet'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
$result = $view->render();
|
||||||
|
static::assertEquals(
|
||||||
|
'attachment; filename="Hello WORLD.txt"',
|
||||||
|
$result->getHeaderLine('Content-Disposition')
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
33
Tests/Functional/Fixtures/SimpleAttachmentTemplateView.php
Normal file
33
Tests/Functional/Fixtures/SimpleAttachmentTemplateView.php
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DigiComp\AttachmentViewUtility\Tests\Functional\Fixtures;
|
||||||
|
|
||||||
|
use DigiComp\AttachmentViewUtility\AttachmentViewTrait;
|
||||||
|
use DigiComp\AttachmentViewUtility\EelFilenameTrait;
|
||||||
|
use Neos\Flow\Mvc\View\SimpleTemplateView;
|
||||||
|
|
||||||
|
class SimpleAttachmentTemplateView extends SimpleTemplateView
|
||||||
|
{
|
||||||
|
use AttachmentViewTrait;
|
||||||
|
use EelFilenameTrait;
|
||||||
|
|
||||||
|
public function __construct(array $options = [])
|
||||||
|
{
|
||||||
|
$this->addOptionFilenameEelExpression();
|
||||||
|
$this->addOptionsForAttachment();
|
||||||
|
parent::__construct($options);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getAttachmentContent()
|
||||||
|
{
|
||||||
|
/** @noinspection MissUsingParentKeywordInspection */
|
||||||
|
return parent::render();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getAttachmentMimeType(): string
|
||||||
|
{
|
||||||
|
return 'text/plain';
|
||||||
|
}
|
||||||
|
}
|
39
composer.json
Normal file
39
composer.json
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
{
|
||||||
|
"name": "digicomp/attachment-view-utility",
|
||||||
|
"type": "neos-package",
|
||||||
|
"description": "",
|
||||||
|
"require": {
|
||||||
|
"cardinalby/content-disposition": "^1.1",
|
||||||
|
"neos/eel": "^7.3 | ^8.3",
|
||||||
|
"neos/flow": "^7.3 | ^8.3",
|
||||||
|
"php": "^8.0"
|
||||||
|
},
|
||||||
|
"autoload": {
|
||||||
|
"psr-4": {
|
||||||
|
"DigiComp\\AttachmentViewUtility\\": "Classes"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"extra": {
|
||||||
|
"branch-alias": {
|
||||||
|
"dev-develop": "1.0.x-dev"
|
||||||
|
},
|
||||||
|
"applied-flow-migrations": [
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Ferdinand Kuhl",
|
||||||
|
"email": "f.kuhl@digital-competence.de",
|
||||||
|
"homepage": "https://www.digital-competence.de",
|
||||||
|
"role": "Developer"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"homepage": "https://git.digital-competence.de/Packages/DigiComp.AttachmentViewUtility",
|
||||||
|
"keywords": [
|
||||||
|
"Neos",
|
||||||
|
"Flow",
|
||||||
|
"view",
|
||||||
|
"mvc"
|
||||||
|
]
|
||||||
|
}
|
Loading…
Reference in a new issue