2013-06-27 12:45:18 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Icinga\Web;
|
|
|
|
|
|
|
|
use Icinga\Application\Icinga;
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Url class that provides convenient access to parameters, allows to modify query parameters and
|
|
|
|
* returns Urls reflecting all changes made to the url and to the parameters.
|
|
|
|
*
|
|
|
|
* Direct instantiation is prohibited and should be done either with @see Url::fromRequest() or
|
|
|
|
* @see Url::fromUrlString()
|
|
|
|
*
|
|
|
|
* Currently, protocol, host and port are ignored and will be implemented when required
|
|
|
|
*
|
|
|
|
*/
|
2013-06-27 12:45:18 +02:00
|
|
|
class Url
|
|
|
|
{
|
2013-08-08 16:22:22 +02:00
|
|
|
/**
|
|
|
|
* Rather dirty hack as the ApplicationBootstrap isn't an interface right now and can't be mocked
|
|
|
|
* overwrite this to use a specific request for all Urls (so only in tests)
|
|
|
|
*
|
|
|
|
* @var null
|
|
|
|
*/
|
|
|
|
public static $overwrittenRequest = null;
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* An array of all parameters stored in this Url
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private $params = array();
|
2013-06-27 12:45:18 +02:00
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* The relative path of this Url, without query parameters
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private $path = '';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The baseUrl that will be appended to @see Url::$path in order to
|
|
|
|
* create an absolute Url
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private $baseUrl = '/';
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private function __construct()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new Url class representing the current request
|
|
|
|
*
|
|
|
|
* If $params are given, those will be added to the request's parameters
|
|
|
|
* and overwrite any existing parameters
|
|
|
|
*
|
|
|
|
* @param string $url The string representation of the Url to parse
|
|
|
|
* @param array $params Parameters that should additionally be considered for the Url
|
|
|
|
* @param Zend_Request $request A request to use instead of the default one
|
|
|
|
*
|
|
|
|
* @return Url
|
|
|
|
*/
|
|
|
|
public static function fromRequest(array $params = array(), $request = null)
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-12 12:06:30 +02:00
|
|
|
if ($request === null) {
|
2013-08-08 16:22:22 +02:00
|
|
|
$request = self::getRequest();
|
2013-07-12 12:06:30 +02:00
|
|
|
}
|
2013-07-31 10:59:34 +02:00
|
|
|
|
|
|
|
$urlObject = new Url();
|
|
|
|
$urlObject->setPath($request->getPathInfo());
|
|
|
|
$urlObject->setParams(array_merge($request->getQuery(), $params));
|
|
|
|
$urlObject->setBaseUrl($request->getBaseUrl());
|
|
|
|
return $urlObject;
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
|
2013-08-08 16:22:22 +02:00
|
|
|
/**
|
|
|
|
* Return a request object that should be used for determining the URL
|
|
|
|
*
|
|
|
|
* @return Zend_Abstract_Request
|
|
|
|
*/
|
|
|
|
private static function getRequest()
|
|
|
|
{
|
|
|
|
if (self::$overwrittenRequest) {
|
|
|
|
return self::$overwrittenRequest;
|
|
|
|
}
|
|
|
|
return Icinga::app()->getFrontController()->getRequest();
|
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Create a new Url class representing the given url
|
|
|
|
*
|
|
|
|
* If $params are given, those will be added to the urls parameters
|
|
|
|
* and overwrite any existing parameters
|
|
|
|
*
|
|
|
|
* @param string $url The string representation of the Url to parse
|
|
|
|
* @param array $params An array of parameters that should additionally be considered for the Url
|
|
|
|
* @param Zend_Request $request A request to use instead of the default one
|
|
|
|
*
|
|
|
|
* @return Url
|
|
|
|
*/
|
|
|
|
public static function fromPath($url, array $params = array(), $request = null)
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-31 10:59:34 +02:00
|
|
|
$urlObject = new Url();
|
|
|
|
if ($request === null) {
|
2013-08-08 16:22:22 +02:00
|
|
|
$request = self::getRequest();
|
2013-07-31 10:59:34 +02:00
|
|
|
}
|
|
|
|
$urlObject->setBaseUrl($request->getBaseUrl());
|
|
|
|
|
|
|
|
$urlParts = parse_url($url);
|
|
|
|
if (isset($urlParts["path"])) {
|
|
|
|
$urlObject->setPath($urlParts["path"]);
|
|
|
|
}
|
|
|
|
if (isset($urlParts["query"])) {
|
2013-08-01 15:54:38 +02:00
|
|
|
$urlParams = array();
|
2013-07-31 10:59:34 +02:00
|
|
|
parse_str($urlParts["query"], $urlParams);
|
|
|
|
$params = array_merge($urlParams, $params);
|
|
|
|
}
|
|
|
|
|
|
|
|
$urlObject->setParams($params);
|
|
|
|
return $urlObject;
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Overwrite the baseUrl.
|
|
|
|
*
|
|
|
|
* If an empty Url is given '/' is used as the base
|
|
|
|
*
|
|
|
|
* @param string $baseUrl The url path to use as the Url Base
|
|
|
|
* @return $this
|
|
|
|
*/
|
2013-06-27 12:45:18 +02:00
|
|
|
public function setBaseUrl($baseUrl)
|
|
|
|
{
|
2013-07-31 10:59:34 +02:00
|
|
|
if (trim($baseUrl) == '') {
|
|
|
|
$baseUrl = '/';
|
|
|
|
}
|
2013-06-27 12:45:18 +02:00
|
|
|
$this->baseUrl = $baseUrl;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2013-08-06 11:53:42 +02:00
|
|
|
/**
|
|
|
|
* Return the baseUrl set for this Url
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getBaseUrl()
|
|
|
|
{
|
|
|
|
return $this->baseUrl;
|
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Set the relative path of this url, without query parameters
|
|
|
|
*
|
|
|
|
* @param string $path The path to set
|
|
|
|
*/
|
|
|
|
public function setPath($path)
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-31 10:59:34 +02:00
|
|
|
$this->path = $path;
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Return the relative path of this Url, without query parameters
|
|
|
|
*
|
|
|
|
* If you want the relative path with query parameters use getRelativeUrl
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2013-07-12 12:06:30 +02:00
|
|
|
public function getPath()
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-12 12:06:30 +02:00
|
|
|
return $this->path;
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Return the relative url with query parameters as a string
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getRelativeUrl()
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-08-01 15:54:38 +02:00
|
|
|
if (empty($this->params)) {
|
|
|
|
return ltrim($this->path, '/');
|
|
|
|
}
|
|
|
|
return ltrim($this->path, '/').'?'.http_build_query($this->params);
|
2013-07-31 10:59:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the absolute url with query parameters as a string
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getAbsoluteUrl()
|
|
|
|
{
|
|
|
|
$url = $this->getRelativeUrl();
|
2013-08-08 16:22:22 +02:00
|
|
|
return preg_replace('/\/{2,}/', '/', '/'.$this->baseUrl.'/'.$url);
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
2013-08-01 15:54:38 +02:00
|
|
|
* Add a set of parameters to the query part if the keys don't exist yet
|
2013-07-31 10:59:34 +02:00
|
|
|
*
|
|
|
|
* @param array $params The parameters to add
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function addParams(array $params)
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
|
|
|
$this->params += $params;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Overwrite the parameters used in the query part
|
|
|
|
*
|
|
|
|
* @param array $params The new parameters to use for the query part
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setParams(array $params)
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-31 10:59:34 +02:00
|
|
|
$this->params = $params;
|
2013-06-27 12:45:18 +02:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Return all parameters that will be used in the query part
|
|
|
|
*
|
|
|
|
* @return array An associative key => value array containing all parameters
|
|
|
|
*/
|
2013-07-12 12:06:30 +02:00
|
|
|
public function getParams()
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-12 12:06:30 +02:00
|
|
|
return $this->params;
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Return true if the Urls' query parameter with $key exists, otherwise false
|
|
|
|
*
|
|
|
|
* @param $key A key to check for existing
|
|
|
|
* @return bool
|
|
|
|
*/
|
2013-06-27 12:45:18 +02:00
|
|
|
public function hasParam($key)
|
|
|
|
{
|
|
|
|
return array_key_exists($key, $this->params);
|
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Return the Url's query parameter with the name $key if exists, otherwise $default
|
|
|
|
*
|
|
|
|
* @param $key A query parameter name to return if existing
|
|
|
|
* @param mixed $default A value to return when the parameter doesn't exist
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2013-06-27 12:45:18 +02:00
|
|
|
public function getParam($key, $default = null)
|
|
|
|
{
|
|
|
|
if ($this->hasParam($key)) {
|
|
|
|
return $this->params[$key];
|
|
|
|
}
|
|
|
|
return $default;
|
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Set a single parameter $key, overwriting existing ones with the same key
|
|
|
|
*
|
|
|
|
* @param string $key A string representing the key of the parameter
|
|
|
|
* @param array|string $value An array or string to set as the parameter value
|
|
|
|
* @return $this
|
|
|
|
*/
|
2013-07-12 12:06:30 +02:00
|
|
|
public function setParam($key, $value)
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-12 12:06:30 +02:00
|
|
|
$this->params[$key] = $value;
|
|
|
|
return $this;
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Remove provided key (if string) or keys (if array of string) from the query parameter array
|
|
|
|
*
|
|
|
|
* @param string|array $keyOrArrayOfKeys An array of strings or a string representing the key(s)
|
|
|
|
* of the parameters to be removed
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function remove($keyOrArrayOfKeys)
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-31 10:59:34 +02:00
|
|
|
if (is_array($keyOrArrayOfKeys)) {
|
|
|
|
$this->removeKeys($keyOrArrayOfKeys);
|
|
|
|
} else {
|
|
|
|
$this->removeKey($keyOrArrayOfKeys);
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Remove all parameters with the parameter names in the $keys array
|
|
|
|
*
|
|
|
|
* @param array $keys An array of strings containing parameter names to remove
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function removeKeys(array $keys)
|
2013-07-12 12:06:30 +02:00
|
|
|
{
|
2013-07-31 10:59:34 +02:00
|
|
|
foreach ($keys as $key) {
|
|
|
|
$this->removeKey($key);
|
|
|
|
}
|
|
|
|
return $this;
|
2013-07-12 12:06:30 +02:00
|
|
|
}
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Remove a single parameter with the provided parameter name $key
|
|
|
|
*
|
|
|
|
* @param string $key The key to remove from the Url
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function removeKey($key)
|
2013-06-27 12:45:18 +02:00
|
|
|
{
|
2013-07-31 10:59:34 +02:00
|
|
|
if (isset($this->params[$key])) {
|
|
|
|
unset($this->params[$key]);
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
2013-07-31 10:59:34 +02:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a copy of this url without the parameter given
|
|
|
|
*
|
|
|
|
* The argument can either a single query parameter name or an array of parameter names to
|
|
|
|
* remove from the query list
|
|
|
|
*
|
|
|
|
* @param string|array $keyOrArrayOfKeys A single string or an array containing parameter names
|
|
|
|
* @return Url
|
|
|
|
*/
|
|
|
|
public function getUrlWithout($keyOrArrayOfKeys)
|
|
|
|
{
|
|
|
|
$url = clone($this);
|
|
|
|
$url->remove($keyOrArrayOfKeys);
|
|
|
|
return $url;
|
|
|
|
}
|
|
|
|
|
2013-08-06 11:53:42 +02:00
|
|
|
|
|
|
|
|
2013-07-31 10:59:34 +02:00
|
|
|
/**
|
|
|
|
* Alias for @see Url::getAbsoluteUrl()
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function __toString() {
|
|
|
|
return $this->getAbsoluteUrl();
|
2013-06-27 12:45:18 +02:00
|
|
|
}
|
|
|
|
}
|