<?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;
    }
}