2

我正在尝试在 Laravel Route 中创建 SabreDAV-Server。以下代码显示我尝试过:

Illuminate\Routing\Router::$verbs = [
    'GET',
    'HEAD',
    'POST',
    'PUT',
    'PATCH',
    'DELETE', 
    'PROPFIND',
    'PROPPATCH',
    'MKCOL',
    'COPY',
    'MOVE',
    'LOCK',
    'UNLOCK'
];


    Route::match(['GET', 'HEAD', 'POST', 'PUT', 'DELETE', 'PATCH', 'PROPFIND', 'PROPPATCH', 'MKCOL', 'COPY', 'MOVE', 'LOCK', 'UNLOCK'], 'carddav{test}', function()
{
        date_default_timezone_set('Europe/Berlin');

        $baseUri = '/carddav';

        $pdo = new PDO('mysql:host=localhost;dbname=dav', 'root', 'root');
        $pdo->setAttribute(PDO::ATTR_ERRMODE,PDO::ERRMODE_EXCEPTION);

        $authBackend = new \Sabre\DAV\Auth\Backend\PDO($pdo);
        $principalBackend = new \Sabre\DAVACL\PrincipalBackend\PDO($pdo);
        $carddavBackend = new \Sabre\CardDAV\Backend\PDO($pdo);

        $nodes = [
                new \Sabre\DAVACL\PrincipalCollection($principalBackend),
                new \Sabre\CardDAV\AddressBookRoot($principalBackend, $carddavBackend)
        ];

        $server = new \Sabre\DAV\Server($nodes);
        $server->setBaseUri($baseUri);

        $server->addPlugin(new \Sabre\DAV\Auth\Plugin($authBackend, 'SabreDAV'));
        $server->addPlugin(new \Sabre\DAV\Browser\Plugin());
        $server->addPlugin(new \Sabre\CardDAV\Plugin());
        $server->addPlugin(new \Sabre\DAVACL\Plugin());
        $server->addPlugin(new \Sabre\DAV\Sync\Plugin());

        $server->exec();
})->where('path', '(.)*';

但是,如果我尝试在浏览器中调用它,则会出现错误:

<?xml version="1.0" encoding="utf-8"?>
<d:error xmlns:d="DAV:" xmlns:s="http://sabredav.org/ns">
  <s:sabredav-version>2.0.4</s:sabredav-version>
  <s:exception>Sabre\DAV\Exception\NotAuthenticated</s:exception>
  <s:message>No digest authentication headers were found</s:message>
</d:error>

没有身份验证提示。

如果我尝试从 Evolution 连接,则会出现消息:“方法不允许”。

有人知道问题是什么吗?

谢谢,佩佩

4

1 回答 1

1

问题是发送的 HTTP 状态代码。无论来自 SabreDAV 的响应如何,Laravel 路由器总是将 HTTP 状态代码设置为 200,因此任何 CardDAV 客户端都不会知道他们必须授权请求——忽略基本身份验证挑战。

我的解决方案可能不是最优雅的解决方案,但它确实有效。只需将 $server->exec() 包装在 ob_start() 和 ob_end() 标签中,然后输出带有真实 Laravel 响应的内容:

ob_start();
$server->exec();

$status = $server->httpResponse->getStatus();
$content = ob_get_contents();
ob_end_clean();

return response($content, $status);

一般指导:

  • 使用“邮递员”(谷歌浏览器应用程序)测试请求,您会在预先发送授权标头时看到它们正在工作
  • 使用像“Charles”这样的网络调试代理来监控实际的请求和响应主体
于 2015-12-23T09:40:52.520 回答