xref: /webtrees/app/Services/ServerCheckService.php (revision 8d897fd11a9a8845d8456cb9eaad33b9c9d5a225)
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 */
17declare(strict_types=1);
18
19namespace Fisharebest\Webtrees\Services;
20
21use Fisharebest\Webtrees\I18N;
22use Illuminate\Database\Capsule\Manager as DB;
23use Illuminate\Support\Collection;
24use Illuminate\Support\Str;
25use SQLite3;
26use stdClass;
27use Throwable;
28
29use function array_map;
30use function class_exists;
31use function date;
32use function e;
33use function explode;
34use function extension_loaded;
35use function function_exists;
36use function in_array;
37use function preg_replace;
38use function strpos;
39use function strtolower;
40use function sys_get_temp_dir;
41use function trim;
42use function version_compare;
43
44use const PATH_SEPARATOR;
45use const PHP_MAJOR_VERSION;
46use const PHP_MINOR_VERSION;
47use const PHP_VERSION;
48
49/**
50 * Check if the server meets the minimum requirements for webtrees.
51 */
52class ServerCheckService
53{
54    private const PHP_SUPPORT_URL   = 'https://secure.php.net/supported-versions.php';
55    private const PHP_MINOR_VERSION = PHP_MAJOR_VERSION . '.' . PHP_MINOR_VERSION;
56    private const PHP_SUPPORT_DATES = [
57        '7.1' => '2019-12-01',
58        '7.2' => '2020-11-30',
59        '7.3' => '2021-12-06',
60    ];
61
62    // As required by illuminate/database 5.8
63    private const MINIMUM_SQLITE_VERSION = '3.7.11';
64
65    /**
66     * Things that may cause webtrees to break.
67     *
68     * @param string $driver
69     *
70     * @return Collection
71     */
72    public function serverErrors($driver = ''): Collection
73    {
74        $errors = Collection::make([
75            $this->databaseDriverErrors($driver),
76            $this->checkPhpExtension('mbstring'),
77            $this->checkPhpExtension('iconv'),
78            $this->checkPhpExtension('pcre'),
79            $this->checkPhpExtension('session'),
80            $this->checkPhpExtension('xml'),
81            $this->checkPhpFunction('parse_ini_file'),
82        ]);
83
84        return $errors
85            ->flatten()
86            ->filter();
87    }
88
89    /**
90     * Things that should be fixed, but which won't stop completely webtrees from running.
91     *
92     * @param string $driver
93     *
94     * @return Collection
95     */
96    public function serverWarnings($driver = ''): Collection
97    {
98        $warnings = Collection::make([
99            $this->databaseDriverWarnings($driver),
100            $this->databaseEngineWarnings(),
101            $this->checkPhpExtension('curl'),
102            $this->checkPhpExtension('gd'),
103            $this->checkPhpExtension('zip'),
104            $this->checkPhpExtension('simplexml'),
105            $this->checkPhpIni('file_uploads', true),
106            $this->checkSystemTemporaryFolder(),
107            $this->checkPhpVersion(),
108        ]);
109
110        return $warnings
111            ->flatten()
112            ->filter();
113    }
114
115    /**
116     * Check if a PHP extension is loaded.
117     *
118     * @param string $extension
119     *
120     * @return string
121     */
122    private function checkPhpExtension(string $extension): string
123    {
124        if (!extension_loaded($extension)) {
125            return I18N::translate('The PHP extension “%s” is not installed.', $extension);
126        }
127
128        return '';
129    }
130
131    /**
132     * Check if a PHP setting is correct.
133     *
134     * @param string $varname
135     * @param bool   $expected
136     *
137     * @return string
138     */
139    private function checkPhpIni(string $varname, bool $expected): string
140    {
141        $ini_get = (bool) ini_get($varname);
142
143        if ($expected && $ini_get !== $expected) {
144            return I18N::translate('The PHP.INI setting “%1$s” is disabled.', $varname);
145        }
146
147        if (!$expected && $ini_get !== $expected) {
148            return I18N::translate('The PHP.INI setting “%1$s” is enabled.', $varname);
149        }
150
151        return '';
152    }
153
154    /**
155     * Check if a PHP function is in the list of disabled functions.
156     *
157     * @param string $function
158     *
159     * @return bool
160     */
161    public function isFunctionDisabled(string $function): bool
162    {
163        $disable_functions = explode(',', ini_get('disable_functions'));
164        $disable_functions = array_map(static function (string $func): string {
165            return strtolower(trim($func));
166        }, $disable_functions);
167
168        $function = strtolower($function);
169
170        return in_array($function, $disable_functions, true) || !function_exists($function);
171    }
172
173    /**
174     * Create a warning message for a disabled function.
175     *
176     * @param string $function
177     *
178     * @return string
179     */
180    private function checkPhpFunction(string $function): string
181    {
182        if ($this->isFunctionDisabled($function)) {
183            return I18N::translate('The PHP function “%1$s” is disabled.', $function . '()');
184        }
185
186        return '';
187    }
188
189    /**
190     * Some servers configure their temporary folder in an unaccessible place.
191     */
192    private function checkPhpVersion(): string
193    {
194        $today = date('Y-m-d');
195
196        foreach (self::PHP_SUPPORT_DATES as $version => $end_date) {
197            if ($today > $end_date && version_compare(self::PHP_MINOR_VERSION, $version) <= 0) {
198                return I18N::translate('Your web server is using PHP version %s, which is no longer receiving security updates. You should upgrade to a later version as soon as possible.', PHP_VERSION) . ' <a href="' . e(self::PHP_SUPPORT_URL) . '">' . e(self::PHP_SUPPORT_URL) . '</a>';
199            }
200        }
201
202        return '';
203    }
204
205    /**
206     * Check the
207     *
208     * @return string
209     */
210    private function checkSqliteVersion(): string
211    {
212        if (class_exists(SQLite3::class)) {
213            $sqlite_version = SQLite3::version()['versionString'];
214
215            if (version_compare($sqlite_version, self::MINIMUM_SQLITE_VERSION) < 0) {
216                return I18N::translate('SQLite version %s is installed. SQLite version %s or later is required.', $sqlite_version, self::MINIMUM_SQLITE_VERSION);
217            }
218        }
219
220        return '';
221    }
222
223    /**
224     * Some servers configure their temporary folder in an unaccessible place.
225     */
226    private function checkSystemTemporaryFolder(): string
227    {
228        $open_basedir = ini_get('open_basedir');
229
230        if ($open_basedir === '') {
231            // open_basedir not used.
232            return '';
233        }
234
235        $open_basedirs = explode(PATH_SEPARATOR, $open_basedir);
236
237        $sys_temp_dir = sys_get_temp_dir();
238        $sys_temp_dir = $this->normalizeFolder($sys_temp_dir);
239
240        foreach ($open_basedirs as $dir) {
241            $dir = $this->normalizeFolder($dir);
242
243            if (strpos($sys_temp_dir, $dir) === 0) {
244                return '';
245            }
246        }
247
248        $message = I18N::translate('The server’s temporary folder cannot be accessed.');
249        $message .= '<br>sys_get_temp_dir() = "' . e($sys_temp_dir) . '"';
250        $message .= '<br>ini_get("open_basedir") = "' . e($open_basedir) . '"';
251
252        return $message;
253    }
254
255    /**
256     * Convert a folder name to a canonical form:
257     * - forward slashes.
258     * - trailing slash.
259     * We can't use realpath() as this can trigger open_basedir restrictions,
260     * and we are using this code to find out whether open_basedir will affect us.
261     *
262     * @param string $path
263     *
264     * @return string
265     */
266    private function normalizeFolder(string $path): string
267    {
268        $path = preg_replace('/[\\/]+/', '/', $path);
269        $path = Str::finish($path, '/');
270
271        return $path;
272    }
273
274    /**
275     * @param string $driver
276     *
277     * @return Collection
278     */
279    private function databaseDriverErrors(string $driver): Collection
280    {
281        switch ($driver) {
282            case 'mysql':
283                return Collection::make([
284                    $this->checkPhpExtension('pdo'),
285                    $this->checkPhpExtension('pdo_mysql'),
286                ]);
287
288            case 'sqlite':
289                return Collection::make([
290                    $this->checkPhpExtension('pdo'),
291                    $this->checkPhpExtension('sqlite3'),
292                    $this->checkPhpExtension('pdo_sqlite'),
293                    $this->checkSqliteVersion(),
294                ]);
295
296            case 'pgsql':
297                return Collection::make([
298                    $this->checkPhpExtension('pdo'),
299                    $this->checkPhpExtension('pdo_pgsql'),
300                ]);
301
302            case 'sqlsvr':
303                return Collection::make([
304                    $this->checkPhpExtension('pdo'),
305                    $this->checkPhpExtension('pdo_odbc'),
306                ]);
307
308            default:
309                return new Collection();
310        }
311    }
312
313    /**
314     * @param string $driver
315     *
316     * @return Collection
317     */
318    private function databaseDriverWarnings(string $driver): Collection
319    {
320        switch ($driver) {
321            case 'sqlite':
322                return new Collection([
323                    I18N::translate('SQLite is only suitable for small sites, testing and evaluation.'),
324                ]);
325
326            case 'pgsql':
327                return new Collection([
328                    I18N::translate('Support for PostgreSQL is experimental.'),
329                ]);
330
331            case 'sqlsvr':
332                return new Collection([
333                    I18N::translate('Support for SQL Server is experimental.'),
334                ]);
335
336            default:
337                return new Collection();
338        }
339    }
340
341    /**
342     * @return Collection
343     */
344    private function databaseEngineWarnings(): Collection
345    {
346        $warnings = new Collection();
347
348        try {
349            $connection = DB::connection();
350        } catch (Throwable $ex) {
351            // During setup, there won't be a connection.
352            return new Collection();
353        }
354
355        if ($connection->getDriverName() === 'mysql') {
356            $sql = "SELECT table_name FROM information_schema.tables JOIN information_schema.engines USING (engine) WHERE table_schema = ? AND LEFT(table_name, ?) = ? AND transactions <> 'YES'";
357
358            $bindings = [
359                $connection->getDatabaseName(),
360                mb_strlen($connection->getTablePrefix()),
361                $connection->getTablePrefix(),
362            ];
363
364            $rows = DB::connection()->select($sql, $bindings);
365
366            $rows = new Collection($rows);
367
368            $rows = $rows->map(static function (stdClass $row): string {
369                $table = $row->TABLE_NAME ?? $row->table_name;
370                return '<code>ALTER TABLE `' . $table . '` ENGINE=InnoDB;</code>';
371            });
372
373            if ($rows->isNotEmpty()) {
374                $warning =
375                    'The database uses non-transactional tables.' .
376                    ' ' .
377                    'You may get errors if more than one user updates data at the same time.' .
378                    ' ' .
379                    'To fix this, run the following SQL commands.' .
380                    '<br>' .
381                    $rows->implode('<br>');
382
383                $warnings->push($warning);
384            }
385        }
386
387        return $warnings;
388    }
389}
390