1<?php 2 3/** 4 * webtrees: online genealogy 5 * Copyright (C) 2019 webtrees development team 6 * This program is free software: you can redistribute it and/or modify 7 * it under the terms of the GNU General Public License as published by 8 * the Free Software Foundation, either version 3 of the License, or 9 * (at your option) any later version. 10 * This program is distributed in the hope that it will be useful, 11 * but WITHOUT ANY WARRANTY; without even the implied warranty of 12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13 * GNU General Public License for more details. 14 * You should have received a copy of the GNU General Public License 15 * along with this program. If not, see <http://www.gnu.org/licenses/>. 16 */ 17 18declare(strict_types=1); 19 20namespace Fisharebest\Webtrees; 21 22use Closure; 23use ErrorException; 24use Fisharebest\Webtrees\Http\Middleware\BadBotBlocker; 25use Fisharebest\Webtrees\Http\Middleware\BootModules; 26use Fisharebest\Webtrees\Http\Middleware\CheckForMaintenanceMode; 27use Fisharebest\Webtrees\Http\Middleware\ClientIp; 28use Fisharebest\Webtrees\Http\Middleware\DoHousekeeping; 29use Fisharebest\Webtrees\Http\Middleware\EmitResponse; 30use Fisharebest\Webtrees\Http\Middleware\HandleExceptions; 31use Fisharebest\Webtrees\Http\Middleware\LoadRoutes; 32use Fisharebest\Webtrees\Http\Middleware\NoRouteFound; 33use Fisharebest\Webtrees\Http\Middleware\PhpEnvironment; 34use Fisharebest\Webtrees\Http\Middleware\ReadConfigIni; 35use Fisharebest\Webtrees\Http\Middleware\RegisterFactories; 36use Fisharebest\Webtrees\Http\Middleware\Router; 37use Fisharebest\Webtrees\Http\Middleware\SecurityHeaders; 38use Fisharebest\Webtrees\Http\Middleware\UpdateDatabaseSchema; 39use Fisharebest\Webtrees\Http\Middleware\UseCache; 40use Fisharebest\Webtrees\Http\Middleware\UseDatabase; 41use Fisharebest\Webtrees\Http\Middleware\UseDebugbar; 42use Fisharebest\Webtrees\Http\Middleware\UseFilesystem; 43use Fisharebest\Webtrees\Http\Middleware\UseLanguage; 44use Fisharebest\Webtrees\Http\Middleware\UseSession; 45use Fisharebest\Webtrees\Http\Middleware\UseTheme; 46use Fisharebest\Webtrees\Http\Middleware\UseTransaction; 47use Fisharebest\Webtrees\Http\Middleware\BaseUrl; 48use Nyholm\Psr7\Factory\Psr17Factory; 49use Psr\Http\Message\ResponseFactoryInterface; 50use Psr\Http\Message\ServerRequestFactoryInterface; 51use Psr\Http\Message\StreamFactoryInterface; 52use Psr\Http\Message\UploadedFileFactoryInterface; 53use Psr\Http\Message\UriFactoryInterface; 54 55use function app; 56use function error_reporting; 57use function set_error_handler; 58 59use const E_USER_DEPRECATED; 60 61/** 62 * Definitions for the webtrees application. 63 */ 64class Webtrees 65{ 66 // The root folder of this installation 67 public const ROOT_DIR = __DIR__ . '/../'; 68 69 // This is the location of system data, such as temporary and cache files. 70 // The system files are always in this location. 71 // It is also the default location of user data, such as media and GEDCOM files. 72 // The user files could be anywhere supported by Flysystem. 73 public const DATA_DIR = self::ROOT_DIR . 'data/'; 74 75 // Location of the file containing the database connection details. 76 public const CONFIG_FILE = self::DATA_DIR . 'config.ini.php'; 77 78 // Location of the file that triggers maintenance mode. 79 public const OFFLINE_FILE = self::DATA_DIR . 'offline.txt'; 80 81 // Location of our modules. 82 public const MODULES_PATH = 'modules_v4/'; 83 public const MODULES_DIR = self::ROOT_DIR . self::MODULES_PATH; 84 85 // Enable debugging on development builds. 86 public const DEBUG = self::STABILITY !== ''; 87 88 // We want to know about all PHP errors during development, and fewer in production. 89 public const ERROR_REPORTING = self::DEBUG ? E_ALL | E_STRICT : E_ALL & !E_USER_DEPRECATED & !E_USER_DEPRECATED; 90 91 // The name of the application. 92 public const NAME = 'webtrees'; 93 94 // Required version of database tables/columns/indexes/etc. 95 public const SCHEMA_VERSION = 44; 96 97 // e.g. "dev", "alpha", "beta", etc. 98 public const STABILITY = 'dev'; 99 100 // Version number 101 public const VERSION = '2.0.5' . (self::STABILITY === '' ? '' : '-') . self::STABILITY; 102 103 // Project website. 104 public const URL = 'https://webtrees.net/'; 105 106 // FAQ linnks 107 public const URL_FAQ_EMAIL = 'https://webtrees.net/faq/email'; 108 109 // Project website. 110 public const GEDCOM_PDF = 'https://webtrees.net/downloads/gedcom-551.pdf'; 111 112 private const MIDDLEWARE = [ 113 PhpEnvironment::class, 114 EmitResponse::class, 115 SecurityHeaders::class, 116 ReadConfigIni::class, 117 BaseUrl::class, 118 HandleExceptions::class, 119 ClientIp::class, 120 UseCache::class, 121 BadBotBlocker::class, 122 UseDatabase::class, 123 UseDebugbar::class, 124 UpdateDatabaseSchema::class, 125 UseFilesystem::class, 126 UseSession::class, 127 UseLanguage::class, 128 RegisterFactories::class, 129 CheckForMaintenanceMode::class, 130 UseTheme::class, 131 DoHousekeeping::class, 132 UseTransaction::class, 133 LoadRoutes::class, 134 BootModules::class, 135 Router::class, 136 NoRouteFound::class, 137 ]; 138 139 /** 140 * Initialise the application. 141 * 142 * @return void 143 */ 144 public function bootstrap(): void 145 { 146 // Show all errors and warnings in development, fewer in production. 147 error_reporting(self::ERROR_REPORTING); 148 149 set_error_handler($this->phpErrorHandler()); 150 } 151 152 /** 153 * An error handler that can be passed to set_error_handler(). 154 * 155 * @return Closure 156 */ 157 private function phpErrorHandler(): Closure 158 { 159 return static function (int $errno, string $errstr, string $errfile, int $errline): bool { 160 // Ignore errors that are silenced with '@' 161 if (error_reporting() & $errno) { 162 throw new ErrorException($errstr, 0, $errno, $errfile, $errline); 163 } 164 165 return true; 166 }; 167 } 168 169 /** 170 * We can use any PSR-7 / PSR-17 compatible message factory. 171 * 172 * @return void 173 */ 174 public function selectMessageFactory(): void 175 { 176 app()->bind(ResponseFactoryInterface::class, Psr17Factory::class); 177 app()->bind(ServerRequestFactoryInterface::class, Psr17Factory::class); 178 app()->bind(StreamFactoryInterface::class, Psr17Factory::class); 179 app()->bind(UploadedFileFactoryInterface::class, Psr17Factory::class); 180 app()->bind(UriFactoryInterface::class, Psr17Factory::class); 181 } 182 183 /** 184 * The webtrees application is built from middleware. 185 * 186 * @return string[] 187 */ 188 public function middleware(): array 189 { 190 return self::MIDDLEWARE; 191 } 192} 193