8

我正在用 Zend 开发一个 Rest Controller,我对 url 到路由器的映射感到困惑。

基本上我阅读了有关Zend Router的信息,但我无法计划我的 url 以满足上述路由。

这些是我的一些应该映射到路由器的 url。

  1. http://localhost/api/v1/tags.xml

  2. http://localhost/api/v1/tags.xml?abc=true (参数:abc=true)

  3. http://localhost/api/v1/tags/123456.xml (参数:123456.xml)

  4. http://localhost/api/v1/tags/123456/pings.xml (参数:123456,pings.xml)

  5. http://localhost/api/v1/tags/123456/pings.xml?a=1&b=2(参数:123456,pings.xml,a=1,b=2)

  6. http://localhost/api/v1/tags/123456/pings/count.xml(参数:123456,pings,count.xml)

我计划这样对于 url 模式 1 到 3,“标签”应该是控制器,对于 url 模式 4 到 6,“pings”应该是控制器。

现在我不确定如何配置路由器以使上述方案起作用。请注意,我无法更改这些网址。我可以为好的答案提供 100 分的声誉分数。

4

2 回答 2

6

前两个 URL 可以组合到一个路由器。

$r = new Zend_Controller_Router_Route_Regex('api/v1/tags.xml',
                array('controller' => 'tags', 'action' => 'index'));
$router->addRoute('route1', $r);

要区分前两个路由,请检查标签控制器中是否存在 abc 参数。在您的标签控制器中添加以下内容,索引操作。

if($this->_getParam('abc') == "true")
{
//route 2
} else {
// route 1
}

类似地,路线 4 和 5 可以合并为一条路线。

我已经解释了路线 6。对于路线 3,您可以使用相同的逻辑。

$r = new Zend_Controller_Router_Route_Regex('api/v1/tags/(.*)/pings/(.*)',
                array('controller' => 'pings', 'action' => 'index'),
array(1 => 'param1',2=>'param2')
);
$router->addRoute('route6', $r);

然后可以在 pings 控制器中访问参数,如下所示。

$this->_getParam('param1') and $this->_getParam('param2')

路线 5:

$r = new Zend_Controller_Router_Route_Regex('api/v1/tags/(.*)/pings.xml',
                array('controller' => 'pings', 'action' => 'index'),
array(1 => 'param1')
);
$router->addRoute('route5', $r);

参数(在?之后的 URL 的一部分)将不会在路由器中处理。默认情况下,它们将被传递给您的控制器。

要获取 URL 中传递的特定参数值,请在控制器中使用以下内容。

$this->_getParam('a');

逻辑是在您的路由中使用 (.*) 并为其分配参数名称并在您的控制器中访问它们

于 2011-03-01T12:40:01.693 回答
4

这是一个算法的开始,它从请求中提取控制器、索引参数和扩展,您可以将其合并到扩展版本中Zend_Rest_Route::match()

public function match( $request )
{
    $path = $request->getPathInfo();

    // distill extension (if any) and the remaining path
    preg_match( '~(?U:(?<path>.*))(?:\.(?<extension>[^\.]*))?$~', $path, $matches );
    $this->_values[ '_extension' ] = isset( $matches[ 'extension' ] ) ? $matches[ 'extension' ] : null;
    $path = isset( $matches[ 'path' ] ) ? $matches[ 'path' ] : '';

    // split the path into segments
    $pathSegments = preg_split( '~' . self::URI_DELIMITER . '~', $path, -1, PREG_SPLIT_NO_EMPTY );

    // leave if no path segments found? up to you to decide, but I put it in anyway
    if( 0 == ( $length = count( $pathSegments ) ) )
    {
        return false;
    }

    // initialize some vars
    $params = array();
    $controller = null;

    // start finding the controller 
    // (presumes controller found at segment 0, 2, 4, etc...)
    for( $i = 0; $i < $length; $i += 2 )
    {
        // you should probably check here if this is a valid REST controller 
        // (see Zend_Rest_Route::_checkRestfulController() )
        $controller = $params[] = $pathSegments[ $i ];
        if( isset( $pathSegments[ $i + 1 ] ) )
        {
            $params[] = $pathSegments[ $i + 1 ];
        }
    }
    // remove the param which is the actual controller
    array_splice( $params, $i - 2, 1 );

    // set the controller
    $this->_values[ 'controller' ] = $controller;

    // merge the params and defaults
    $this->_values = array_merge( $this->_values, $params, $this->_defaults );

    return $this->_values;
}

它几乎没有经过测试,因此当然不是生产材料。但它应该让你开始。

到目前为止,它给你的是:
控制器
、扩展
、索引参数

这没有给你的是:
动作(发布、放置、删除等。算法已经在里面Zend_Rest_Route::match()
命名参数(Zend_Controller_Request_Http已经处理好了)

编辑
我意识到到目前为止这个答案可能被认为有点模糊。关键是将这个算法与 的match()算法合并Zend_Rest_Route。但是上面这段代码还是需要多多注意的;您可能也想考虑模块(就像 一样Zend_Rest_Route),甚至可能是可选的 baseUrl (不确定 ZF 实际上是如何在内部处理这个问题的)。

于 2011-03-01T12:55:36.350 回答