291 lines
6.5 KiB
PHP
291 lines
6.5 KiB
PHP
<?php
|
|
|
|
namespace PicoFeed\Client;
|
|
|
|
/**
|
|
* URL class.
|
|
*
|
|
* @author Frederic Guillot
|
|
*/
|
|
class Url
|
|
{
|
|
/**
|
|
* URL.
|
|
*
|
|
* @var string
|
|
*/
|
|
private $url = '';
|
|
|
|
/**
|
|
* URL components.
|
|
*
|
|
* @var array
|
|
*/
|
|
private $components = array();
|
|
|
|
/**
|
|
* Constructor.
|
|
*
|
|
* @param string $url URL
|
|
*/
|
|
public function __construct($url)
|
|
{
|
|
$this->url = $url;
|
|
$this->components = parse_url($url) ?: array();
|
|
|
|
// Issue with PHP < 5.4.7 and protocol relative url
|
|
if (version_compare(PHP_VERSION, '5.4.7', '<') && $this->isProtocolRelative()) {
|
|
$pos = strpos($this->components['path'], '/', 2);
|
|
|
|
if ($pos === false) {
|
|
$pos = strlen($this->components['path']);
|
|
}
|
|
|
|
$this->components['host'] = substr($this->components['path'], 2, $pos - 2);
|
|
$this->components['path'] = substr($this->components['path'], $pos);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Shortcut method to get an absolute url from relative url.
|
|
*
|
|
* @static
|
|
*
|
|
* @param mixed $item_url Unknown url (can be relative or not)
|
|
* @param mixed $website_url Website url
|
|
*
|
|
* @return string
|
|
*/
|
|
public static function resolve($item_url, $website_url)
|
|
{
|
|
$link = is_string($item_url) ? new self($item_url) : $item_url;
|
|
$website = is_string($website_url) ? new self($website_url) : $website_url;
|
|
|
|
if ($link->isRelativeUrl()) {
|
|
if ($link->isRelativePath()) {
|
|
return $link->getAbsoluteUrl($website->getBaseUrl($website->getBasePath()));
|
|
}
|
|
|
|
return $link->getAbsoluteUrl($website->getBaseUrl());
|
|
} elseif ($link->isProtocolRelative()) {
|
|
$link->setScheme($website->getScheme());
|
|
}
|
|
|
|
return $link->getAbsoluteUrl();
|
|
}
|
|
|
|
/**
|
|
* Shortcut method to get a base url.
|
|
*
|
|
* @static
|
|
*
|
|
* @param string $url
|
|
*
|
|
* @return string
|
|
*/
|
|
public static function base($url)
|
|
{
|
|
$link = new self($url);
|
|
|
|
return $link->getBaseUrl();
|
|
}
|
|
|
|
/**
|
|
* Get the base URL.
|
|
*
|
|
* @param string $suffix Add a suffix to the url
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getBaseUrl($suffix = '')
|
|
{
|
|
return $this->hasHost() ? $this->getScheme('://').$this->getHost().$this->getPort(':').$suffix : '';
|
|
}
|
|
|
|
/**
|
|
* Get the absolute URL.
|
|
*
|
|
* @param string $base_url Use this url as base url
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getAbsoluteUrl($base_url = '')
|
|
{
|
|
if ($base_url) {
|
|
$base = new self($base_url);
|
|
$url = $base->getAbsoluteUrl().substr($this->getFullPath(), 1);
|
|
} else {
|
|
$url = $this->hasHost() ? $this->getBaseUrl().$this->getFullPath() : '';
|
|
}
|
|
|
|
return $url;
|
|
}
|
|
|
|
/**
|
|
* Return true if the url is relative.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function isRelativeUrl()
|
|
{
|
|
return !$this->hasScheme() && !$this->isProtocolRelative();
|
|
}
|
|
|
|
/**
|
|
* Return true if the path is relative.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function isRelativePath()
|
|
{
|
|
$path = $this->getPath();
|
|
|
|
return empty($path) || $path{0}
|
|
!== '/';
|
|
}
|
|
|
|
/**
|
|
* Filters the path of a URI.
|
|
*
|
|
* Imported from Guzzle library: https://github.com/guzzle/psr7/blob/master/src/Uri.php#L568-L582
|
|
*
|
|
* @param $path
|
|
*
|
|
* @return string
|
|
*/
|
|
public function filterPath($path, $charUnreserved = 'a-zA-Z0-9_\-\.~', $charSubDelims = '!\$&\'\(\)\*\+,;=')
|
|
{
|
|
return preg_replace_callback(
|
|
'/(?:[^'.$charUnreserved.$charSubDelims.':@\/%]+|%(?![A-Fa-f0-9]{2}))/',
|
|
function (array $matches) { return rawurlencode($matches[0]); },
|
|
$path
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Get the path.
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getPath()
|
|
{
|
|
return $this->filterPath(empty($this->components['path']) ? '' : $this->components['path']);
|
|
}
|
|
|
|
/**
|
|
* Get the base path.
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getBasePath()
|
|
{
|
|
$current_path = $this->getPath();
|
|
|
|
$path = $this->isRelativePath() ? '/' : '';
|
|
$path .= substr($current_path, -1) === '/' ? $current_path : dirname($current_path);
|
|
|
|
return preg_replace('/\\\\\/|\/\//', '/', $path.'/');
|
|
}
|
|
|
|
/**
|
|
* Get the full path (path + querystring + fragment).
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getFullPath()
|
|
{
|
|
$path = $this->isRelativePath() ? '/' : '';
|
|
$path .= $this->getPath();
|
|
$path .= empty($this->components['query']) ? '' : '?'.$this->components['query'];
|
|
$path .= empty($this->components['fragment']) ? '' : '#'.$this->components['fragment'];
|
|
|
|
return $path;
|
|
}
|
|
|
|
/**
|
|
* Get the hostname.
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getHost()
|
|
{
|
|
return empty($this->components['host']) ? '' : $this->components['host'];
|
|
}
|
|
|
|
/**
|
|
* Return true if the url has a hostname.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function hasHost()
|
|
{
|
|
return !empty($this->components['host']);
|
|
}
|
|
|
|
/**
|
|
* Get the scheme.
|
|
*
|
|
* @param string $suffix Suffix to add when there is a scheme
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getScheme($suffix = '')
|
|
{
|
|
return ($this->hasScheme() ? $this->components['scheme'] : 'http').$suffix;
|
|
}
|
|
|
|
/**
|
|
* Set the scheme.
|
|
*
|
|
* @param string $scheme Set a scheme
|
|
*
|
|
* @return string
|
|
*/
|
|
public function setScheme($scheme)
|
|
{
|
|
$this->components['scheme'] = $scheme;
|
|
}
|
|
|
|
/**
|
|
* Return true if the url has a scheme.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function hasScheme()
|
|
{
|
|
return !empty($this->components['scheme']);
|
|
}
|
|
|
|
/**
|
|
* Get the port.
|
|
*
|
|
* @param string $prefix Prefix to add when there is a port
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getPort($prefix = '')
|
|
{
|
|
return $this->hasPort() ? $prefix.$this->components['port'] : '';
|
|
}
|
|
|
|
/**
|
|
* Return true if the url has a port.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function hasPort()
|
|
{
|
|
return !empty($this->components['port']);
|
|
}
|
|
|
|
/**
|
|
* Return true if the url is protocol relative (start with //).
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function isProtocolRelative()
|
|
{
|
|
return strpos($this->url, '//') === 0;
|
|
}
|
|
}
|