9

大家好,

我已经在这个问题上苦苦挣扎了一个多星期,最后决定寻求帮助,希望有人知道答案。

我正在开发一个应用程序,它使用Google 的协议缓冲区作为数据交换格式。我正在使用 DrSlump 的 PHP实现,它让您使用数据填充类实例,然后将它们序列化为二进制字符串(或将二进制字符串解码为 PHP 对象)。

我已经设法实现了我的自定义ProtobufStrategy,如果selectRenderer(ViewEvent $e)事件ProtobufRenderer包含ProtobufModel. 渲染器然后通过调用从模型中提取我的自定义参数$model->getOptions()以确定需要将哪些消息发送回客户端,序列化数据并将二进制字符串输出到php://output

为了使其更有意义,让我们看一下以下示例消息:

message SearchRequest {
    required string query = 1;
    optional int32 page_number = 2;
    optional int32 result_per_page = 3;
}

如果我想用这条消息回复客户,我会从我的操作中返回如下内容:

public function getSearchRequestAction()
{
    [..]
    $data = array(
        'query'           => 'my query',
        'page_number'     => 3,
        'result_per_page' => 20,
    );
    return new ProtobufModel($data, array(
        'message' => 'MyNamespace\Protobuf\SearchRequest',
    ));
}

如您所见,我正在利用ViewModel的第二个参数$options来判断需要序列化的消息。然后,如前所述,可以通过调用在渲染器中提取$model->getOptions().

到现在为止还挺好。我的控制器操作按预期输出二进制数据。

但是,我在处理异常时遇到了问题。我的计划是捕获所有异常并使用我的异常消息实例响应客户端,如下所示:

message Exception {
    optional string message = 1;
    optional int32 code = 2;
    optional string file = 3;
    optional uint32 line = 4;
    optional string trace = 5;
    optional Exception previous = 6;
}

从理论上讲,它应该开箱即用,但事实并非如此。问题是Zend\Mvc\View\Http\ExceptionStrategy::prepareExceptionViewModel(MvcEvent $e)返回一个实例ViewModel,它显然不包含我需要的额外$options信息。

它还返回ViewModeland not ProtobufModel,这意味着 Zend 调用默认值ViewPhpRenderer并将异常输出为 HTML 页面。

我想要做的是用我自己的类替换默认值ExceptionStrategy(最终也是RouteNotFoundStrategy),这将返回如下内容:

$data = array(
    'message'  => $e->getMessage(),
    'code'     => $e->getCode(),
    'file'     => $e->getFile(),
    'line'     => $e->getLine(),
    'trace'    => $e->getTraceAsString(),
    'previous' => $e->getPrevious(),
);
return new ProtobufModel($data, array(
    'message' => 'MyNamespace\Protobuf\Exception',
));

……我找不到办法……

我尝试创建自己的ExceptionStrategy类并将其别名为现有的ExceptionStrategy服务,但 Zend 抱怨已经存在具有此类名称的服务。

我怀疑我在自定义策略扩展的正确路径上找不到覆盖默认策略的方法。

我注意到默认ExceptionStrategy和控制台一个在Zend/Mvc/View/Http/ViewManager. 我希望我不必添加自定义视图管理器来实现这么简单的事情,但是如果我错了,请纠正我。

任何帮助将不胜感激!

4

1 回答 1

10

最简单的方法是做一个小小的捏造。

首先,注册你的监听器以比 ExceptionStrategy 更高的优先级运行;因为它以默认优先级注册,这意味着任何高于 1 的优先级。

然后,在您的侦听器中,在您返回之前,确保将 MvcEvent 中的“错误”设置为虚假值:

$e->setError(false);

一旦你这样做了,默认的 ExceptionStrategy 会说,“这里没什么可做的,继续前进”并提前返回,然后再对 ViewModel 做任何事情。

当您使用它时,您还应该确保更改事件中的结果实例:

$e->setResult($yourProtobufModel)

因为这将确保这是其他听众检查的内容。

于 2012-10-02T16:16:38.427 回答