2018-10-05 22:36:09 +02:00
|
|
|
<?php
|
2024-08-24 15:27:00 +02:00
|
|
|
|
|
|
|
// Copyright (C) 2010-2024, the Friendica project
|
|
|
|
// SPDX-FileCopyrightText: 2010-2024 the Friendica project
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-or-later
|
2018-10-05 22:36:09 +02:00
|
|
|
|
|
|
|
namespace Friendica\App;
|
|
|
|
|
2019-08-16 09:46:38 +02:00
|
|
|
use Detection\MobileDetect;
|
2023-01-15 16:12:25 +01:00
|
|
|
use Friendica\Core\Config\Capability\IManageConfigValues;
|
2019-07-21 01:22:10 +02:00
|
|
|
use Friendica\Database\Database;
|
2018-10-05 22:36:09 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Mode of the current Friendica Node
|
|
|
|
*
|
|
|
|
* @package Friendica\App
|
|
|
|
*/
|
|
|
|
class Mode
|
|
|
|
{
|
2019-07-21 14:40:50 +02:00
|
|
|
const LOCALCONFIGPRESENT = 1;
|
|
|
|
const DBAVAILABLE = 2;
|
|
|
|
const DBCONFIGAVAILABLE = 4;
|
2018-10-05 22:36:09 +02:00
|
|
|
const MAINTENANCEDISABLED = 8;
|
|
|
|
|
2021-01-01 19:35:29 +00:00
|
|
|
const UNDEFINED = 0;
|
|
|
|
const INDEX = 1;
|
|
|
|
const DAEMON = 2;
|
|
|
|
const WORKER = 3;
|
|
|
|
|
2020-10-02 20:14:57 +02:00
|
|
|
const BACKEND_CONTENT_TYPES = ['application/jrd+json', 'text/xml',
|
2020-10-02 10:55:42 +00:00
|
|
|
'application/rss+xml', 'application/atom+xml', 'application/activity+json'];
|
|
|
|
|
2021-11-19 22:47:49 +01:00
|
|
|
/**
|
|
|
|
* A list of modules, which are backend methods
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
const BACKEND_MODULES = [
|
|
|
|
'_well_known',
|
|
|
|
'api',
|
|
|
|
'dfrn_notify',
|
|
|
|
'feed',
|
|
|
|
'fetch',
|
|
|
|
'followers',
|
|
|
|
'following',
|
|
|
|
'hcard',
|
|
|
|
'hostxrd',
|
|
|
|
'inbox',
|
|
|
|
'manifest',
|
|
|
|
'nodeinfo',
|
|
|
|
'noscrape',
|
|
|
|
'objects',
|
|
|
|
'outbox',
|
|
|
|
'poco',
|
|
|
|
'receive',
|
|
|
|
'rsd_xml',
|
|
|
|
'statistics_json',
|
|
|
|
'xrd',
|
|
|
|
];
|
|
|
|
|
2018-10-05 22:36:09 +02:00
|
|
|
/***
|
2019-08-15 15:51:15 +02:00
|
|
|
* @var int The mode of this Application
|
2018-10-05 22:36:09 +02:00
|
|
|
*
|
|
|
|
*/
|
2018-10-06 16:27:20 +02:00
|
|
|
private $mode;
|
|
|
|
|
2021-01-01 19:35:29 +00:00
|
|
|
/***
|
|
|
|
* @var int Who executes this Application
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
private $executor = self::UNDEFINED;
|
|
|
|
|
2019-08-15 15:51:15 +02:00
|
|
|
/**
|
|
|
|
* @var bool True, if the call is a backend call
|
|
|
|
*/
|
|
|
|
private $isBackend;
|
|
|
|
|
2019-08-16 09:46:38 +02:00
|
|
|
/**
|
|
|
|
* @var bool True, if the call is a ajax call
|
|
|
|
*/
|
|
|
|
private $isAjax;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool True, if the call is from a mobile device
|
|
|
|
*/
|
|
|
|
private $isMobile;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool True, if the call is from a tablet device
|
|
|
|
*/
|
|
|
|
private $isTablet;
|
|
|
|
|
|
|
|
public function __construct(int $mode = 0, bool $isBackend = false, bool $isAjax = false, bool $isMobile = false, bool $isTablet = false)
|
2018-10-06 16:27:20 +02:00
|
|
|
{
|
2019-08-15 15:51:15 +02:00
|
|
|
$this->mode = $mode;
|
|
|
|
$this->isBackend = $isBackend;
|
2019-08-16 09:46:38 +02:00
|
|
|
$this->isAjax = $isAjax;
|
|
|
|
$this->isMobile = $isMobile;
|
|
|
|
$this->isTablet = $isTablet;
|
2018-10-06 16:27:20 +02:00
|
|
|
}
|
2018-10-05 22:36:09 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the App mode
|
|
|
|
*
|
|
|
|
* - App::MODE_INSTALL : Either the database connection can't be established or the config table doesn't exist
|
|
|
|
* - App::MODE_MAINTENANCE: The maintenance mode has been set
|
|
|
|
* - App::MODE_NORMAL : Normal run with all features enabled
|
|
|
|
*
|
2019-08-12 18:13:58 +02:00
|
|
|
* @return Mode returns the determined mode
|
2019-07-21 01:22:10 +02:00
|
|
|
*
|
2019-07-21 14:40:50 +02:00
|
|
|
* @throws \Exception
|
2018-10-05 22:36:09 +02:00
|
|
|
*/
|
2023-01-15 16:12:25 +01:00
|
|
|
public function determine(string $basePath, Database $database, IManageConfigValues $config): Mode
|
2018-10-05 22:36:09 +02:00
|
|
|
{
|
2019-08-12 18:13:58 +02:00
|
|
|
$mode = 0;
|
2018-10-06 16:27:20 +02:00
|
|
|
|
2023-01-15 13:46:01 +01:00
|
|
|
if (!file_exists($basePath . '/config/local.config.php') &&
|
|
|
|
!file_exists($basePath . '/config/local.ini.php') &&
|
|
|
|
!file_exists($basePath . '/.htconfig.php')) {
|
2019-08-12 18:13:58 +02:00
|
|
|
return new Mode($mode);
|
2018-10-05 22:36:09 +02:00
|
|
|
}
|
|
|
|
|
2019-08-12 18:13:58 +02:00
|
|
|
$mode |= Mode::LOCALCONFIGPRESENT;
|
2018-10-05 22:36:09 +02:00
|
|
|
|
2019-08-12 18:13:58 +02:00
|
|
|
if (!$database->connected()) {
|
|
|
|
return new Mode($mode);
|
2018-10-05 22:36:09 +02:00
|
|
|
}
|
|
|
|
|
2019-08-12 18:13:58 +02:00
|
|
|
$mode |= Mode::DBAVAILABLE;
|
2018-10-05 22:36:09 +02:00
|
|
|
|
2023-01-15 16:12:25 +01:00
|
|
|
if (!empty($config->get('system', 'maintenance'))) {
|
2019-08-12 18:13:58 +02:00
|
|
|
return new Mode($mode);
|
2018-10-05 22:36:09 +02:00
|
|
|
}
|
|
|
|
|
2019-08-12 18:13:58 +02:00
|
|
|
$mode |= Mode::MAINTENANCEDISABLED;
|
2019-07-21 01:22:10 +02:00
|
|
|
|
2019-08-16 09:46:38 +02:00
|
|
|
return new Mode($mode, $this->isBackend, $this->isAjax, $this->isMobile, $this->isTablet);
|
2019-08-15 15:51:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if the site is called via a backend process
|
|
|
|
*
|
2021-11-16 22:34:49 +01:00
|
|
|
* @param bool $isBackend True, if the call is from a backend script (daemon, worker, ...)
|
|
|
|
* @param array $server The $_SERVER variable
|
2021-11-19 22:47:49 +01:00
|
|
|
* @param Arguments $args The Friendica App arguments
|
2021-11-16 22:34:49 +01:00
|
|
|
* @param MobileDetect $mobileDetect The mobile detection library
|
2019-08-15 15:51:15 +02:00
|
|
|
*
|
|
|
|
* @return Mode returns the determined mode
|
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function determineRunMode(bool $isBackend, array $server, Arguments $args, MobileDetect $mobileDetect): Mode
|
2019-08-15 15:51:15 +02:00
|
|
|
{
|
2020-10-02 10:55:42 +00:00
|
|
|
foreach (self::BACKEND_CONTENT_TYPES as $type) {
|
2020-10-02 09:31:39 +00:00
|
|
|
if (strpos(strtolower($server['HTTP_ACCEPT'] ?? ''), $type) !== false) {
|
|
|
|
$isBackend = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-19 22:47:49 +01:00
|
|
|
$isBackend = $isBackend || in_array($args->getModuleName(), static::BACKEND_MODULES);
|
2019-08-16 09:46:38 +02:00
|
|
|
$isMobile = $mobileDetect->isMobile();
|
|
|
|
$isTablet = $mobileDetect->isTablet();
|
|
|
|
$isAjax = strtolower($server['HTTP_X_REQUESTED_WITH'] ?? '') == 'xmlhttprequest';
|
2019-08-15 15:51:15 +02:00
|
|
|
|
2019-08-16 09:46:38 +02:00
|
|
|
return new Mode($this->mode, $isBackend, $isAjax, $isMobile, $isTablet);
|
2018-10-05 22:36:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks, if the Friendica Node has the given mode
|
|
|
|
*
|
|
|
|
* @param int $mode A mode to test
|
|
|
|
*
|
|
|
|
* @return bool returns true, if the mode is set
|
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function has(int $mode): bool
|
2018-10-05 22:36:09 +02:00
|
|
|
{
|
2018-10-06 16:27:20 +02:00
|
|
|
return ($this->mode & $mode) > 0;
|
2018-10-05 22:36:09 +02:00
|
|
|
}
|
|
|
|
|
2021-01-01 19:35:29 +00:00
|
|
|
/**
|
|
|
|
* Set the execution mode
|
|
|
|
*
|
|
|
|
* @param integer $executor Execution Mode
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function setExecutor(int $executor)
|
|
|
|
{
|
|
|
|
$this->executor = $executor;
|
|
|
|
|
|
|
|
// Daemon and worker are always backend
|
|
|
|
if (in_array($executor, [self::DAEMON, self::WORKER])) {
|
|
|
|
$this->isBackend = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*isBackend = true;*
|
|
|
|
* get the execution mode
|
|
|
|
*
|
|
|
|
* @return int Execution Mode
|
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function getExecutor(): int
|
2021-01-01 19:35:29 +00:00
|
|
|
{
|
|
|
|
return $this->executor;
|
|
|
|
}
|
2018-10-05 22:36:09 +02:00
|
|
|
|
|
|
|
/**
|
2022-12-28 02:57:57 +01:00
|
|
|
* Install mode is when the local config file is missing or the database isn't available.
|
2018-10-05 22:36:09 +02:00
|
|
|
*
|
2022-06-16 16:49:43 +02:00
|
|
|
* @return bool Whether installation mode is active (local/database configuration files present or not)
|
2018-10-05 22:36:09 +02:00
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function isInstall(): bool
|
2018-10-05 22:36:09 +02:00
|
|
|
{
|
2018-10-06 16:27:20 +02:00
|
|
|
return !$this->has(Mode::LOCALCONFIGPRESENT) ||
|
2022-12-28 02:57:57 +01:00
|
|
|
!$this->has(MODE::DBAVAILABLE);
|
2018-10-05 22:36:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Normal mode is when the local config file is set, the DB schema is installed and the maintenance mode is off.
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function isNormal(): bool
|
2018-10-05 22:36:09 +02:00
|
|
|
{
|
2018-10-06 16:27:20 +02:00
|
|
|
return $this->has(Mode::LOCALCONFIGPRESENT) &&
|
2019-07-21 01:22:10 +02:00
|
|
|
$this->has(Mode::DBAVAILABLE) &&
|
|
|
|
$this->has(Mode::MAINTENANCEDISABLED);
|
2018-10-05 22:36:09 +02:00
|
|
|
}
|
2019-08-15 15:51:15 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true, if the call is from a backend node (f.e. from a worker)
|
|
|
|
*
|
|
|
|
* @return bool Is it a backend call
|
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function isBackend(): bool
|
2019-08-15 15:51:15 +02:00
|
|
|
{
|
|
|
|
return $this->isBackend;
|
|
|
|
}
|
2019-08-16 09:46:38 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if request was an AJAX (xmlhttprequest) request.
|
|
|
|
*
|
|
|
|
* @return bool true if it was an AJAX request
|
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function isAjax(): bool
|
2019-08-16 09:46:38 +02:00
|
|
|
{
|
|
|
|
return $this->isAjax;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if request was a mobile request.
|
|
|
|
*
|
|
|
|
* @return bool true if it was an mobile request
|
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function isMobile(): bool
|
2019-08-16 09:46:38 +02:00
|
|
|
{
|
|
|
|
return $this->isMobile;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if request was a tablet request.
|
|
|
|
*
|
|
|
|
* @return bool true if it was an tablet request
|
|
|
|
*/
|
2022-06-16 16:49:43 +02:00
|
|
|
public function isTablet(): bool
|
2019-08-16 09:46:38 +02:00
|
|
|
{
|
|
|
|
return $this->isTablet;
|
|
|
|
}
|
2019-07-21 14:40:50 +02:00
|
|
|
}
|