2

我正在使用 PSR-7(使用 Zend Expressive)进行开发。我想出了方法

ServerRequestInterface::withAttribute()

我想知道为什么对象 Response 没有。我想在“响应端”处理后通过中间件传递元数据。是否有某种方式可以在 Response 上传递“属性”以进行后处理?遵循架构指南,实现这一目标的最佳方式是什么?

4

3 回答 3

3

最佳实践是使用请求对象在中间件之间传递数据。响应是发送给客户端的内容,您希望保持干净。该请求仅存在于服务器上,您可以添加(敏感数据)属性以传递。如果出现问题或者您在删除自定义数据之前提前返回响应,那么这并不重要,因为您的响应是“干净的”。

此外,如果您需要传递数据:中间件始终按照从配置中获取的顺序执行。这样可以确保 MiddlewareX 中的请求对象包含 MiddlewareY 设置的数据。

更新:有关如何通过请求传递数据的示例。

中间件 2 设置了一个信使对象,中间件 4 可以使用该对象来设置再次输出所需的数据。

<?php

namespace Middleware;

use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Message\ResponseInterface;

class Middleware2
{
    public function __invoke(ServerRequestInterface $request, ResponseInterface $response, callable $next)
    {
        $messenger = new Messenger();

        // Do something else before next middleware

        if ($next) {
            $response = $next($request->withAttribute(Messenger::class, $messenger), $response);
        }  

        // Do something with the Response after it got back
        // At this point the $messenger object contains the updated data from Middleware4

        return $response->withHeader('Content-Language', $locale);
    }
}

中间件 4 获取信使对象并更新其值。

<?php

namespace Middleware;

use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Message\ResponseInterface;

class Middleware4
{
    public function __invoke(ServerRequestInterface $request, ResponseInterface $response, callable $next)
    {
        $messenger = $request->getAttribute(Messenger::class);
        $messenger->info('going in');
        // Do something else before next middleware

        if ($next) {
            $response = $next($request->withAttribute(FlashMessenger::class, $messenger), $response);
        }  

        // Do something with the Response after it got back
        $messenger->info('going out');

        return $response->withHeader('Content-Language', $locale);
    }
}
于 2016-07-17T07:20:15.737 回答
2

PSR-7规范仅为服务器请求定义属性。它们主要用于存储从传入请求推导出的元数据,以便稍后在您到达域层时使用它们。

另一方面,响应通常在域层中创建,并在实际发送到客户端之前遍历所有中间件堆栈。因此,添加到响应中的元数据将没有实际使用的地方。

我想如果您想将数据从内部中间件传递到外部中间件,最好的方法是使用响应标头。

于 2016-07-09T13:26:01.160 回答
0

不确定这是否是“最佳实践”,但另一种可能性是将您的数据对象简单地注入中间件。

中间件 2 注入了一个 messenger 对象并在其上设置了一些数据:

<?php

namespace Middleware;

use Interop\Http\Server\MiddlewareInterface;
use Interop\Http\Server\RequestHandlerInterface;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;

class Middleware2
{

    private $messenger;

    public function __construct(Messenger $messenger)
    {
        $this->messenger = $messenger;
    }

    public function process(
        ServerRequestInterface $request,
        RequestHandlerInterface $handler
        ): ResponseInterface {
            $this->messenger->foo = 'bar';
            $response = $handler->handle($request);
            if ($this->messenger->foo = 'baz') {
                return $response->withHeader('Really-Important-Header', 'Baz');
            }
            return $response;
        }
}

中间件 4 改变数据:

<?php

namespace Middleware;

use Interop\Http\Server\MiddlewareInterface;
use Interop\Http\Server\RequestHandlerInterface;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;

class Middleware4
{

    private $messenger;

    public function __construct(Messenger $messenger)
    {
        $this->messenger = $messenger;
    }

    public function process(
        ServerRequestInterface $request,
        RequestHandlerInterface $handler
        ): ResponseInterface {
            $this->messenger->foo = 'baz';
            return $handler->handle($request);
        }
}

您甚至可以使用其中一个中间件作为信使。

警告:您必须确保使用相同的信使对象构造两个类。但大多数依赖注入容器似乎都是这种情况。

于 2017-12-22T14:03:18.177 回答