2019-04-08 15:12:10 -04:00
|
|
|
<?php
|
2020-02-08 11:16:42 -05:00
|
|
|
/**
|
2023-01-01 09:36:24 -05:00
|
|
|
* @copyright Copyright (C) 2010-2023, the Friendica project
|
2020-02-08 11:16:42 -05:00
|
|
|
*
|
|
|
|
* @license GNU AGPL version 3 or any later version
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
2019-04-08 15:12:10 -04:00
|
|
|
|
2019-08-15 11:23:00 -04:00
|
|
|
namespace Friendica\App;
|
2019-04-08 15:12:10 -04:00
|
|
|
|
2021-10-26 15:44:29 -04:00
|
|
|
use Friendica\Core\Config\Capability\IManageConfigValues;
|
2019-12-03 15:11:42 -05:00
|
|
|
use Friendica\Core\System;
|
2019-08-15 11:23:00 -04:00
|
|
|
use Friendica\Util\Network;
|
|
|
|
use Friendica\Util\Strings;
|
2019-12-03 15:11:42 -05:00
|
|
|
use Friendica\Network\HTTPException;
|
2019-04-08 15:12:10 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A class which checks and contains the basic
|
2019-04-08 17:12:34 -04:00
|
|
|
* environment for the BaseURL (url, urlpath, ssl_policy, hostname, scheme)
|
2019-04-08 15:12:10 -04:00
|
|
|
*/
|
|
|
|
class BaseURL
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* No SSL necessary
|
|
|
|
*/
|
|
|
|
const SSL_POLICY_NONE = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SSL is necessary
|
|
|
|
*/
|
|
|
|
const SSL_POLICY_FULL = 1;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SSL is optional, but preferred
|
|
|
|
*/
|
|
|
|
const SSL_POLICY_SELFSIGN = 2;
|
|
|
|
|
2019-04-08 17:12:34 -04:00
|
|
|
/**
|
|
|
|
* Define the Default SSL scheme
|
|
|
|
*/
|
|
|
|
const DEFAULT_SSL_SCHEME = self::SSL_POLICY_SELFSIGN;
|
|
|
|
|
2019-04-08 15:12:10 -04:00
|
|
|
/**
|
|
|
|
* The Friendica Config
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2021-10-26 15:44:29 -04:00
|
|
|
* @var IManageConfigValues
|
2019-04-08 15:12:10 -04:00
|
|
|
*/
|
|
|
|
private $config;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The server side variables
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private $server;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The hostname of the Base URL
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private $hostname;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The SSL_POLICY of the Base URL
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
private $sslPolicy;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The URL sub-path of the Base URL
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private $urlPath;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The full URL
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private $url;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The current scheme of this call
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private $scheme;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the hostname of this node
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @return string
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function getHostname(): string
|
2019-04-08 15:12:10 -04:00
|
|
|
{
|
|
|
|
return $this->hostname;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the current scheme of this call
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @return string
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function getScheme(): string
|
2019-04-08 15:12:10 -04:00
|
|
|
{
|
|
|
|
return $this->scheme;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the SSL policy of this node
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @return int
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function getSSLPolicy(): int
|
2019-04-08 15:12:10 -04:00
|
|
|
{
|
|
|
|
return $this->sslPolicy;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the sub-path of this URL
|
2019-08-15 11:23:00 -04:00
|
|
|
*
|
2019-04-08 15:12:10 -04:00
|
|
|
* @return string
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function getUrlPath(): string
|
2019-04-08 15:12:10 -04:00
|
|
|
{
|
|
|
|
return $this->urlPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the full URL of this call
|
|
|
|
*
|
|
|
|
* Note: $ssl parameter value doesn't directly correlate with the resulting protocol
|
|
|
|
*
|
|
|
|
* @param bool $ssl True, if ssl should get used
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function get(bool $ssl = false): string
|
2019-04-08 15:12:10 -04:00
|
|
|
{
|
2019-04-08 17:12:34 -04:00
|
|
|
if ($this->sslPolicy === self::SSL_POLICY_SELFSIGN && $ssl) {
|
|
|
|
return Network::switchScheme($this->url);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->url;
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save current parts of the base Url
|
|
|
|
*
|
|
|
|
* @param string? $hostname
|
|
|
|
* @param int? $sslPolicy
|
|
|
|
* @param string? $urlPath
|
|
|
|
*
|
|
|
|
* @return bool true, if successful
|
2022-06-16 10:49:43 -04:00
|
|
|
* @TODO Find proper types
|
2019-04-08 15:12:10 -04:00
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function save($hostname = null, $sslPolicy = null, $urlPath = null): bool
|
2019-04-08 15:12:10 -04:00
|
|
|
{
|
2023-01-04 02:14:00 -05:00
|
|
|
$currUrl = $this->url;
|
|
|
|
|
|
|
|
$configTransaction = $this->config->beginTransaction();
|
2019-04-08 15:12:10 -04:00
|
|
|
|
2019-04-09 02:48:04 -04:00
|
|
|
if (!empty($hostname) && $hostname !== $this->hostname) {
|
2023-01-04 02:14:00 -05:00
|
|
|
$configTransaction->set('config', 'hostname', $hostname);
|
|
|
|
$this->hostname = $hostname;
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|
|
|
|
|
2019-04-09 02:48:04 -04:00
|
|
|
if (isset($sslPolicy) && $sslPolicy !== $this->sslPolicy) {
|
2023-01-04 02:14:00 -05:00
|
|
|
$configTransaction->set('system', 'ssl_policy', $sslPolicy);
|
|
|
|
$this->sslPolicy = $sslPolicy;
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|
|
|
|
|
2019-04-09 02:48:04 -04:00
|
|
|
if (isset($urlPath) && $urlPath !== $this->urlPath) {
|
2023-01-04 02:14:00 -05:00
|
|
|
$configTransaction->set('system', 'urlpath', $urlPath);
|
|
|
|
$this->urlPath = $urlPath;
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
$this->determineBaseUrl();
|
2023-01-03 17:55:51 -05:00
|
|
|
if ($this->url !== $currUrl) {
|
2023-01-04 02:14:00 -05:00
|
|
|
$configTransaction->set('system', 'url', $this->url);
|
|
|
|
}
|
|
|
|
|
2023-01-04 13:55:22 -05:00
|
|
|
$configTransaction->commit();
|
2019-04-08 15:12:10 -04:00
|
|
|
|
2019-04-10 14:38:39 -04:00
|
|
|
return true;
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save the current url as base URL
|
|
|
|
*
|
2022-06-16 10:49:43 -04:00
|
|
|
* @param string $url
|
2019-04-08 15:12:10 -04:00
|
|
|
*
|
|
|
|
* @return bool true, if the save was successful
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function saveByURL(string $url): bool
|
2019-04-08 15:12:10 -04:00
|
|
|
{
|
|
|
|
$parsed = @parse_url($url);
|
|
|
|
|
2021-06-16 09:23:07 -04:00
|
|
|
if (empty($parsed) || empty($parsed['host'])) {
|
2019-04-08 15:12:10 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$hostname = $parsed['host'];
|
|
|
|
if (!empty($hostname) && !empty($parsed['port'])) {
|
|
|
|
$hostname .= ':' . $parsed['port'];
|
|
|
|
}
|
|
|
|
|
|
|
|
$urlPath = null;
|
|
|
|
if (!empty($parsed['path'])) {
|
|
|
|
$urlPath = trim($parsed['path'], '\\/');
|
|
|
|
}
|
|
|
|
|
2019-04-08 17:12:34 -04:00
|
|
|
$sslPolicy = null;
|
|
|
|
if (!empty($parsed['scheme'])) {
|
|
|
|
if ($parsed['scheme'] == 'https') {
|
|
|
|
$sslPolicy = BaseURL::SSL_POLICY_FULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->save($hostname, $sslPolicy, $urlPath);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks, if a redirect to the HTTPS site would be necessary
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function checkRedirectHttps()
|
|
|
|
{
|
2019-08-15 11:23:00 -04:00
|
|
|
return $this->config->get('system', 'force_ssl') &&
|
|
|
|
($this->getScheme() == "http") &&
|
|
|
|
intval($this->getSSLPolicy()) == BaseURL::SSL_POLICY_FULL &&
|
|
|
|
strpos($this->get(), 'https://') === 0 &&
|
|
|
|
!empty($this->server['REQUEST_METHOD']) &&
|
|
|
|
$this->server['REQUEST_METHOD'] === 'GET';
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-10-26 16:09:11 -04:00
|
|
|
* @param IManageConfigValues $config The Friendica IConfiguration
|
|
|
|
* @param array $server The $_SERVER array
|
2019-04-08 15:12:10 -04:00
|
|
|
*/
|
2021-10-26 15:44:29 -04:00
|
|
|
public function __construct(IManageConfigValues $config, array $server)
|
2019-04-08 15:12:10 -04:00
|
|
|
{
|
2023-01-21 14:20:29 -05:00
|
|
|
$this->config = $config;
|
|
|
|
$this->server = $server;
|
2019-04-08 17:12:34 -04:00
|
|
|
$this->hostname = $this->config->get('config', 'hostname');
|
2023-01-21 14:20:29 -05:00
|
|
|
$this->urlPath = $this->config->get('system', 'urlpath') ?? '';
|
|
|
|
$this->sslPolicy = $this->config->get('system', 'ssl_policy') ?? static::DEFAULT_SSL_SCHEME;
|
2019-04-08 17:12:34 -04:00
|
|
|
$this->url = $this->config->get('system', 'url');
|
2019-04-08 15:12:10 -04:00
|
|
|
|
2023-01-21 15:06:06 -05:00
|
|
|
if (empty($this->hostname) || empty($this->url)) {
|
|
|
|
throw new \Exception('Invalid config - Missing system.url or config.hostname');
|
|
|
|
}
|
|
|
|
|
2023-01-21 14:20:29 -05:00
|
|
|
$this->determineSchema();
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine the full URL based on all parts
|
|
|
|
*/
|
|
|
|
private function determineBaseUrl()
|
|
|
|
{
|
|
|
|
$scheme = 'http';
|
|
|
|
|
|
|
|
if ($this->sslPolicy == self::SSL_POLICY_FULL) {
|
|
|
|
$scheme = 'https';
|
|
|
|
}
|
|
|
|
|
2019-08-15 11:23:00 -04:00
|
|
|
$this->url = $scheme . '://' . $this->hostname . (!empty($this->urlPath) ? '/' . $this->urlPath : '');
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine the scheme of the current used link
|
|
|
|
*/
|
|
|
|
private function determineSchema()
|
|
|
|
{
|
|
|
|
$this->scheme = 'http';
|
|
|
|
|
|
|
|
if (!empty($this->server['HTTPS']) ||
|
2019-08-15 11:23:00 -04:00
|
|
|
!empty($this->server['HTTP_FORWARDED']) && preg_match('/proto=https/', $this->server['HTTP_FORWARDED']) ||
|
|
|
|
!empty($this->server['HTTP_X_FORWARDED_PROTO']) && $this->server['HTTP_X_FORWARDED_PROTO'] == 'https' ||
|
|
|
|
!empty($this->server['HTTP_X_FORWARDED_SSL']) && $this->server['HTTP_X_FORWARDED_SSL'] == 'on' ||
|
|
|
|
!empty($this->server['FRONT_END_HTTPS']) && $this->server['FRONT_END_HTTPS'] == 'on' ||
|
|
|
|
!empty($this->server['SERVER_PORT']) && (intval($this->server['SERVER_PORT']) == 443) // XXX: reasonable assumption, but isn't this hardcoding too much?
|
2019-04-08 15:12:10 -04:00
|
|
|
) {
|
|
|
|
$this->scheme = 'https';
|
|
|
|
}
|
|
|
|
}
|
2019-08-15 11:23:00 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes the base url from an url. This avoids some mixed content problems.
|
|
|
|
*
|
|
|
|
* @param string $origURL
|
|
|
|
*
|
|
|
|
* @return string The cleaned url
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function remove(string $origURL): string
|
2019-08-15 11:23:00 -04:00
|
|
|
{
|
|
|
|
// Remove the hostname from the url if it is an internal link
|
|
|
|
$nurl = Strings::normaliseLink($origURL);
|
|
|
|
$base = Strings::normaliseLink($this->get());
|
|
|
|
$url = str_replace($base . '/', '', $nurl);
|
|
|
|
|
|
|
|
// if it is an external link return the orignal value
|
|
|
|
if ($url == Strings::normaliseLink($origURL)) {
|
|
|
|
return $origURL;
|
|
|
|
} else {
|
|
|
|
return $url;
|
|
|
|
}
|
|
|
|
}
|
2019-12-03 15:11:42 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Redirects to another module relative to the current Friendica base URL.
|
|
|
|
* If you want to redirect to a external URL, use System::externalRedirectTo()
|
|
|
|
*
|
|
|
|
* @param string $toUrl The destination URL (Default is empty, which is the default page of the Friendica node)
|
|
|
|
* @param bool $ssl if true, base URL will try to get called with https:// (works just for relative paths)
|
|
|
|
*
|
2022-07-07 15:47:39 -04:00
|
|
|
* @throws HTTPException\FoundException
|
|
|
|
* @throws HTTPException\MovedPermanentlyException
|
|
|
|
* @throws HTTPException\TemporaryRedirectException
|
|
|
|
*
|
2019-12-03 15:11:42 -05:00
|
|
|
* @throws HTTPException\InternalServerErrorException In Case the given URL is not relative to the Friendica node
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function redirect(string $toUrl = '', bool $ssl = false)
|
2019-12-03 15:11:42 -05:00
|
|
|
{
|
|
|
|
if (!empty(parse_url($toUrl, PHP_URL_SCHEME))) {
|
2023-02-04 19:18:05 -05:00
|
|
|
throw new HTTPException\InternalServerErrorException("$toUrl is not a relative path, please use System::externalRedirectTo");
|
2019-12-03 15:11:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
$redirectTo = $this->get($ssl) . '/' . ltrim($toUrl, '/');
|
|
|
|
System::externalRedirect($redirectTo);
|
|
|
|
}
|
2019-12-30 17:00:08 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the base url as string
|
|
|
|
*/
|
2022-06-16 10:49:43 -04:00
|
|
|
public function __toString(): string
|
2019-12-30 17:00:08 -05:00
|
|
|
{
|
2022-06-16 10:49:43 -04:00
|
|
|
return (string) $this->get();
|
2019-12-30 17:00:08 -05:00
|
|
|
}
|
2019-04-08 15:12:10 -04:00
|
|
|
}
|