From fd655c4185a6be904c29e163683218a884de98fa Mon Sep 17 00:00:00 2001 From: Thorsten Rinne Date: Sat, 1 Jun 2024 13:26:47 +0200 Subject: [PATCH] refactor: migrated setup and update to Twig --- nginx.conf | 4 + phpmyfaq/.htaccess | 4 + phpmyfaq/assets/src/configuration/update.js | 8 +- phpmyfaq/assets/src/setup.js | 25 +- .../admin/configuration/upgrade.twig | 258 +++++----- phpmyfaq/assets/templates/setup/base.twig | 49 ++ phpmyfaq/assets/templates/setup/index.twig | 335 +++++++++++++ phpmyfaq/assets/templates/setup/install.twig | 50 ++ phpmyfaq/assets/templates/setup/update.twig | 2 +- .../assets/templates/setup/update/step2.twig | 4 +- phpmyfaq/setup/index.php | 451 +----------------- phpmyfaq/setup/update.php | 72 --- phpmyfaq/src/Bootstrap.php | 2 +- phpmyfaq/src/admin-routes.php | 2 +- phpmyfaq/src/api-routes.php | 2 +- phpmyfaq/src/phpMyFAQ/Application.php | 24 +- .../Controller/AbstractController.php | 2 + .../{Setup => Api}/SetupController.php | 3 +- .../Controller/Frontend/SetupController.php | 146 ++++++ phpmyfaq/src/phpMyFAQ/Language.php | 36 +- phpmyfaq/src/phpMyFAQ/Setup/Installer.php | 51 +- phpmyfaq/src/phpMyFAQ/Sitemap.php | 4 - phpmyfaq/src/phpMyFAQ/System.php | 12 +- .../src/phpMyFAQ/Template/TwigWrapper.php | 5 +- phpmyfaq/src/public-routes.php | 29 ++ phpmyfaq/update/index.php | 38 ++ 26 files changed, 909 insertions(+), 709 deletions(-) create mode 100644 phpmyfaq/assets/templates/setup/base.twig create mode 100644 phpmyfaq/assets/templates/setup/install.twig delete mode 100644 phpmyfaq/setup/update.php rename phpmyfaq/src/phpMyFAQ/Controller/{Setup => Api}/SetupController.php (98%) create mode 100644 phpmyfaq/src/phpMyFAQ/Controller/Frontend/SetupController.php create mode 100644 phpmyfaq/src/public-routes.php create mode 100644 phpmyfaq/update/index.php diff --git a/nginx.conf b/nginx.conf index f7a142474e..b79356706c 100644 --- a/nginx.conf +++ b/nginx.conf @@ -80,6 +80,10 @@ server { # Private APIs rewrite api/(autocomplete|bookmark/([0-9]+)|user/data/update|user/password/update|user/request-removal|contact|voting|register|captcha|share|comment/create|faq/create|question/create) /api/index.php last; + # Setup and update pages + rewrite setup/ /setup/index.php last; + rewrite update/ /update/index.php last; + # Setup APIs rewrite api/setup/(check|backup|update-database) /api/index.php last; diff --git a/phpmyfaq/.htaccess b/phpmyfaq/.htaccess index 1d0d7cd5d9..5f3a396010 100644 --- a/phpmyfaq/.htaccess +++ b/phpmyfaq/.htaccess @@ -96,6 +96,10 @@ Header set Access-Control-Allow-Headers "Content-Type, Authorization" # User pages RewriteRule ^user/(ucp|bookmarks|request-removal|logout|register) index.php?action=$1 [L,QSA] + # Setup and update pages + RewriteRule ^setup/(.*) setup/index.php [L,QSA] + RewriteRule ^update/(.*) update/index.php [L,QSA] + # Administration API RewriteRule ^admin/api/(.*) admin/api/index.php [L,QSA] diff --git a/phpmyfaq/assets/src/configuration/update.js b/phpmyfaq/assets/src/configuration/update.js index 9d330af518..eb88e525c0 100644 --- a/phpmyfaq/assets/src/configuration/update.js +++ b/phpmyfaq/assets/src/configuration/update.js @@ -20,13 +20,13 @@ export const handleUpdateNextStepButton = () => { if (nextStepButton && nextStep) { nextStepButton.addEventListener('click', (event) => { event.preventDefault(); - window.location.replace(`./update.php?step=${nextStep.value}`); + window.location.replace(`./update?step=${nextStep.value}`); }); } }; export const handleUpdateInformation = async () => { - if (window.location.href.endsWith('update.php')) { + if (window.location.href.endsWith('/update')) { const installedVersion = document.getElementById('phpmyfaq-update-installed-version'); try { @@ -67,7 +67,7 @@ export const handleUpdateInformation = async () => { }; export const handleConfigBackup = async () => { - if (window.location.href.endsWith('update.php?step=2')) { + if (window.location.href.endsWith('/update?step=2')) { const installedVersion = document.getElementById('phpmyfaq-update-installed-version'); try { @@ -96,7 +96,7 @@ export const handleConfigBackup = async () => { }; export const handleDatabaseUpdate = async () => { - if (window.location.href.endsWith('update.php?step=3')) { + if (window.location.href.endsWith('/update?step=3')) { const installedVersion = document.getElementById('phpmyfaq-update-installed-version'); try { diff --git a/phpmyfaq/assets/src/setup.js b/phpmyfaq/assets/src/setup.js index 8e835e17d5..a8b7b9e9e7 100644 --- a/phpmyfaq/assets/src/setup.js +++ b/phpmyfaq/assets/src/setup.js @@ -55,23 +55,26 @@ document.addEventListener('DOMContentLoaded', () => { function showTab(n) { const currentStep = document.getElementsByClassName('step'); - if (currentStep) { + if (currentStep.length > 0) { currentStep[n].style.display = 'block'; } const prevButton = document.getElementById('prevBtn'); const nextButton = document.getElementById('nextBtn'); - if (n === 0) { - prevButton.style.display = 'none'; - } else { - prevButton.style.display = 'inline'; - } - if (n === currentStep.length - 1) { - nextButton.innerHTML = 'Submit'; - } else { - nextButton.innerHTML = 'Next'; + + if (prevButton && nextButton) { + if (n === 0) { + prevButton.style.display = 'none'; + } else { + prevButton.style.display = 'inline'; + } + if (n === currentStep.length - 1) { + nextButton.innerHTML = 'Submit'; + } else { + nextButton.innerHTML = 'Next'; + } + stepIndicator(n); } - stepIndicator(n); } const nextPrev = (n) => { diff --git a/phpmyfaq/assets/templates/admin/configuration/upgrade.twig b/phpmyfaq/assets/templates/admin/configuration/upgrade.twig index 7345bef430..cc7fc6fc41 100644 --- a/phpmyfaq/assets/templates/admin/configuration/upgrade.twig +++ b/phpmyfaq/assets/templates/admin/configuration/upgrade.twig @@ -1,145 +1,151 @@
-

- - {{ adminHeaderUpgrade }} -

+

+ + {{ adminHeaderUpgrade }} +

-
+
-
-
-
- {{ headerCheckHealth }} -
-

- {{ msgHealthCheck }} -

- -
- - -
-
+
+
+
+ {{ headerCheckHealth }} +
+

+ {{ msgHealthCheck }} +

+ +
+ +
+
+
-
-
-
- {{ headerCheckUpdates }} -
-

- {{ msgUpdateCheck }} -

- -
- - -
-
+
+
+
+ {{ headerCheckUpdates }} +
+

+ {{ msgUpdateCheck }} +

+ +
+ +
+
+
-
-
-
- {{ headerDownloadPackage }} -
-

- {{ msgDownloadPackage }} -

- - {% if isOnNightlies %} -

- {{ alertNightlyBuild }} -

-
- - -
- {% endif %} -
-
+
+
+
+ {{ headerDownloadPackage }} +
+

+ {{ msgDownloadPackage }} +

+ + {% if isOnNightlies %} +

+ {{ alertNightlyBuild }} +

+
+ + +
+ {% endif %} +
+
-
-
-
- {{ headerExtractPackage }} -
-

- {{ msgExtractPackage }} -

- -
-
- - -
-
-
+
+
+
+ {{ headerExtractPackage }} +
+

+ {{ msgExtractPackage }} +

+ +
+
+ + +
+
+
-
-
-
- 5. Install downloaded package -
-

- This extracts the downloaded package into your filesystem. -

- -
-
-
- -
-
-
    -
  1. Backup of current installation
  2. -
  3. Installation of new files
  4. -
-
-
-
-
- 0% -
-
-
-
- 0% -
-
-
-
+
+
+
+ 5. Install downloaded package +
+

+ This extracts the downloaded package into your filesystem. +

+ +
+
+
+ +
+
+
    +
  1. Backup of current installation
  2. +
  3. Installation of new files
  4. +
+
+
+
+
+ 0% +
+
+
+
+ 0%
+
+
+
+
+
+
+
+

+ You can still use the manual update process if you prefer. +

-
-
- {{ msgLastCheckDate }} - {{ dateLastChecked | date('Y-m-d H:i:s') }} -
-
- {{ msgLastVersionAvailable }} - n/a -
-
- {{ msgReleaseEnvironment }} {{ releaseEnvironment }} -
+
+ {{ msgLastCheckDate }} + {{ dateLastChecked | date('Y-m-d H:i:s') }} +
+
+ {{ msgLastVersionAvailable }} + n/a +
+
+ {{ msgReleaseEnvironment }} + {{ releaseEnvironment }}
+
diff --git a/phpmyfaq/assets/templates/setup/base.twig b/phpmyfaq/assets/templates/setup/base.twig new file mode 100644 index 0000000000..97d0c08daf --- /dev/null +++ b/phpmyfaq/assets/templates/setup/base.twig @@ -0,0 +1,49 @@ + + + + + phpMyFAQ {{ newVersion }} {{ setupType }} + + + + + + + + + + + +{% block content %}{% endblock %} + + + + + + diff --git a/phpmyfaq/assets/templates/setup/index.twig b/phpmyfaq/assets/templates/setup/index.twig index e69de29bb2..28b92becfa 100644 --- a/phpmyfaq/assets/templates/setup/index.twig +++ b/phpmyfaq/assets/templates/setup/index.twig @@ -0,0 +1,335 @@ +{% extends 'setup/base.twig' %} + +{% block content %} +
+
+ +
+
+ +
+

phpMyFAQ {{ newVersion }}

+
+

+ Did you already read our + documentation + carefully before starting the phpMyFAQ setup? +

+
+
+ +
+ Database Setup + LDAP Setup + Elasticsearch Setup + Admin user account +
+ + {% for hint in nonCriticalSettings %} + + {% endfor %} + + {% if checkBasicError %} + + {% else %} + +
+ +

Step 1/4: Database setup

+ +
+ +
+ + Please select your preferred database type. +
+
+ +
+
+ +
+ + + Please enter the host or path to the socket of your database server. + +
+
+
+ +
+ + Please enter the port your database server. +
+
+
+ +
+ + Please enter your database user. +
+
+
+ +
+
+ + + + +
+ Please enter your database password. +
+
+
+ +
+ + Please enter your existing database name. +
+
+
+ +
+
+ +
+ + + Please enter the full path to your SQLite datafile which should be outside your document root. + +
+
+
+ +
+ +
+ + + Please enter a table prefix here if you want to install more phpMyFAQ installations in one + database. + +
+
+
+ +
+

Step 2/4: LDAP setup

+ {% if isLdapEnabled %} +
+
+ + +
+
+
+ +
+ + Please enter the host of your LDAP server. +
+
+
+ +
+ + Please enter the port of your LDAP server. +
+
+
+ +
+ + Please enter your specified RDN username. +
+
+
+ +
+ + Please enter your LDAP password. +
+
+
+ +
+ + + Please enter your distinguished name, e.g. 'cn=John Doe,ou=Accounts,o=My Company,c=US'. + +
+
+

+ + You can add additional LDAP configurations later in the admin configuration panel. +

+ {% else %} +

+ + LDAP support is not enabled in your PHP installation. +

+ {% endif %} +
+ +
+

Step 3/4: Elasticsearch setup

+ {% if isElasticsearchEnabled %} +
+
+ + +
+
+
+ +
+
+ + + Add another Elasticsearch Host + +
+ + Please enter the host (domain or IP) with port number of your Elasticsearch server. + +
+
+
+ +
+ + Please enter your Elasticsearch index name. +
+
+ {% else %} +

+ + cURL and OpenSSL support are not enabled in your PHP installation. +

+ {% endif %} +
+ +
+

Step 4/4: Admin user setup

+ +
+ +
+ + Please select your default language. +
+
+ +
+ +
+ + + Complexity of rights and permissions. + +
+
+
+ +
+ + Please enter your real name. +
+
+
+ +
+ + Please enter your email address. +
+
+
+ +
+ + Please enter your login name. +
+
+
+ +
+
+ + + + +
+ Please enter your password with at least 8 characters. +
+
+
+ +
+
+ + + + +
+ Please retype your password. +
+
+
+
+

+ + After clicking the "Submit" button, all necessary tables will be created and filled with your data. + Depending on your system, this may take some time. Stay tuned. +

+
+
+
+ + + + + + {% endif %} +
+
+
+
+{% endblock %} diff --git a/phpmyfaq/assets/templates/setup/install.twig b/phpmyfaq/assets/templates/setup/install.twig new file mode 100644 index 0000000000..be6b69a653 --- /dev/null +++ b/phpmyfaq/assets/templates/setup/install.twig @@ -0,0 +1,50 @@ +{% extends 'setup/base.twig' %} + +{% block content %} +
+
+ +
+ +
+

phpMyFAQ {{ newVersion }}

+
+

+ Did you already read our + documentation + carefully before starting the phpMyFAQ setup? +

+
+
+ + {% if checkBasicError %} + + {% elseif installationError %} + + {% else %} +
+
+

Installation

+ + {{ feedbackIndicator }} + +

+ Wow, looks like the installation worked like a charm. This is pretty cool, isn't it? :-) +

+ +

+ You can visit your version of phpMyFAQ or login into your + admin section. +

+
+
+ {% endif %} + +
+
+
+{% endblock %} diff --git a/phpmyfaq/assets/templates/setup/update.twig b/phpmyfaq/assets/templates/setup/update.twig index 345a45bbc2..bd8407d36b 100644 --- a/phpmyfaq/assets/templates/setup/update.twig +++ b/phpmyfaq/assets/templates/setup/update.twig @@ -54,7 +54,7 @@ Database updates
- {{ include('setup/update/step' ~ currentStep ~ '.twig') }} + {{ include('./setup/update/step' ~ currentStep ~ '.twig') }}
diff --git a/phpmyfaq/assets/templates/setup/update/step2.twig b/phpmyfaq/assets/templates/setup/update/step2.twig index 34d0aac70b..8bd21c5e65 100644 --- a/phpmyfaq/assets/templates/setup/update/step2.twig +++ b/phpmyfaq/assets/templates/setup/update/step2.twig @@ -22,8 +22,8 @@
-
diff --git a/phpmyfaq/setup/index.php b/phpmyfaq/setup/index.php index 0ce5c69c38..6a80672582 100644 --- a/phpmyfaq/setup/index.php +++ b/phpmyfaq/setup/index.php @@ -1,36 +1,21 @@ - * @author Tom Rochester - * @author Johannes Schlüter - * @author Uwe Pries - * @author Matteo Scaramuccia - * @author Florian Anderiasch - * @copyright 2002-2024 phpMyFAQ Team - * @license https://www.mozilla.org/MPL/2.0/ Mozilla Public License Version 2.0 - * @link https://www.phpmyfaq.de - * @since 2002-08-20 - */ use Composer\Autoload\ClassLoader; -use phpMyFAQ\Component\Alert; -use phpMyFAQ\Core\Exception; -use phpMyFAQ\Language\LanguageCodes; -use phpMyFAQ\Setup\Installer; +use phpMyFAQ\Application; +use phpMyFAQ\Controller\Frontend\SetupController; use phpMyFAQ\Strings; -use phpMyFAQ\System; use phpMyFAQ\Translation; +use Symfony\Component\Routing\Route; +use Symfony\Component\Routing\RouteCollection; define('PMF_ROOT_DIR', dirname(__FILE__, 2)); + +// +// The directory where the translations reside +// +define('PMF_TRANSLATION_DIR', dirname(__DIR__) . '/translations'); + const PMF_SRC_DIR = PMF_ROOT_DIR . '/src'; const IS_VALID_PHPMYFAQ = null; @@ -61,71 +46,7 @@ $loader = new ClassLoader(); $loader->add('phpMyFAQ', PMF_SRC_DIR); $loader->register(); -?> - - - - - phpMyFAQ <?= System::getVersion() ?> Setup - - - - - - - - - - - -
-
- -
-
- -
-

phpMyFAQ

-
-

- Did you already read our - documentation - carefully before starting the phpMyFAQ setup? -

-
-
- -
- Database Setup - LDAP Setup - Elasticsearch Setup - Admin user account -
- echo 'Error: ' . $e->getMessage(); } -$system = new System(); -$installer = new Installer($system); - -try { - $installer->checkBasicStuff(); -} catch (Exception $e) { - echo Alert::danger('ad_entryins_fail', $e->getMessage()); - System::renderFooter(); -} - -$installer->checkFilesystemPermissions(); - -// not yet POSTed -if (!isset($_POST['sql_server']) && !isset($_POST['sql_user']) && !isset($_POST['sql_db'])) { - $installer->checkNoncriticalSettings() - ?> - -
- -

Step 1/4: Database setup

+$routes = new RouteCollection(); -
- -
- - Please select your preferred database type. -
-
+$routeDefinitions = [ + 'public.setup.index' => ['/', SetupController::class, 'index'], + 'public.setup.install' => ['/install', SetupController::class, 'install'], +]; -
-
- -
- - - Please enter the host or path to the socket of your database server. - -
-
-
- -
- - Please enter the port your database server. -
-
-
- -
- - Please enter your database user. -
-
-
- -
-
- - - - -
- Please enter your database password. -
-
-
- -
- - Please enter your existing database name. -
-
-
- -
-
- -
- - - Please enter the full path to your SQLite datafile which should be outside your document root. - -
-
-
- -
- -
- - - Please enter a table prefix here if you want to install more phpMyFAQ installations in one - database. - -
-
-
- - -
-

Step 2/4: LDAP setup

- -
-
- - -
-
-
- -
- - Please enter the host of your LDAP server. -
-
-
- -
- - Please enter the port of your LDAP server. -
-
-
- -
- - Please enter your specified RDN username. -
-
-
- -
- - Please enter your LDAP password. -
-
-
- -
- - - Please enter your distinguished name, e.g. 'cn=John Doe,ou=Accounts,o=My Company,c=US'. - -
-
-

- You can add additional LDAP configurations later in the admin configuration panel. -

- -
- - - -
-

Step 3/4: Elasticsearch setup

- -
-
- - -
-
-
- -
-
- - - Add another Elasticsearch Host - -
- - Please enter the host (domain or IP) with port number of your Elasticsearch server. - -
-
-
- -
- - Please enter your Elasticsearch index name. -
-
- -
- - - -
-

Step 4/4: Admin user setup

- -
- -
- - Please select your default language. -
-
- -
- -
- - - Complexity of rights and permissions. - -
-
-
- -
- - Please enter your real name. -
-
-
- -
- - Please enter your email address. -
-
-
- -
- - Please enter your login name. -
-
-
- -
-
- - - - -
- Please enter your password with at least 8 characters. -
-
-
- -
-
- - - - -
- Please retype your password. -
-
-
-
-

- - After clicking the "Submit" button, all necessary tables will be created and filled with your data. - Depending on your system, this may take some time. Stay tuned. -

-
-
-
- - - - - -
-
- - -
-
-

Installation

- startInstall(); - } catch (Exception $e) { - echo $e->getMessage(); - } - ?> -

- Wow, looks like the installation worked like a charm. This is pretty cool, isn't it? :-) -

- -

- You can visit your version of phpMyFAQ or login into your - admin section. -

-
-
- [$path, $controller, $action]) { + $routes->add($name, new Route($path, ['_controller' => [$controller, $action]])); } - System::renderFooter(); +$app = new Application(); +try { + $app->run($routes); +} catch (Exception $exception) { + echo $exception->getMessage(); +} diff --git a/phpmyfaq/setup/update.php b/phpmyfaq/setup/update.php deleted file mode 100644 index 20bc8d0be8..0000000000 --- a/phpmyfaq/setup/update.php +++ /dev/null @@ -1,72 +0,0 @@ - - * @author Thomas Melchinger - * @author Matteo Scaramuccia - * @copyright 2002-2024 phpMyFAQ Team - * @license https://www.mozilla.org/MPL/2.0/ Mozilla Public License Version 2.0 - * @link https://www.phpmyfaq.de - * @since 2002-01-10 - */ - -use phpMyFAQ\Configuration; -use phpMyFAQ\Configuration\DatabaseConfiguration; -use phpMyFAQ\Filter; -use phpMyFAQ\Setup\Update; -use phpMyFAQ\Strings; -use phpMyFAQ\System; -use phpMyFAQ\Template\TwigWrapper; -use Symfony\Component\HttpFoundation\RedirectResponse; - -const COPYRIGHT = '© 2001-2024 phpMyFAQ Team'; -const IS_VALID_PHPMYFAQ = null; - -define('PMF_ROOT_DIR', dirname(__FILE__, 2)); - -if (version_compare(PHP_VERSION, '8.2.0') < 0) { - die('Sorry, but you need PHP 8.2.0 or later!'); -} - -set_time_limit(0); - -require PMF_ROOT_DIR . '/src/Bootstrap.php'; - -Strings::init(); - -$system = new System(); -$faqConfig = Configuration::getConfigurationInstance(); - -$step = Filter::filterInput(INPUT_GET, 'step', FILTER_VALIDATE_INT, 1); -$version = Filter::filterInput(INPUT_POST, 'installed-version', FILTER_SANITIZE_SPECIAL_CHARS); - -$installedVersion = $faqConfig->getVersion(); - -$update = new Update($system, $faqConfig); -$update->setVersion($installedVersion); - -if (!$update->checkDatabaseFile()) { - $redirect = new RedirectResponse('./index.php'); - $redirect->send(); -} - -$twig = new TwigWrapper(PMF_ROOT_DIR . '/assets/templates'); -$template = $twig->loadTemplate('./setup/update.twig'); - -$templateVars = [ - 'newVersion' => System::getVersion(), - 'installedVersion' => $installedVersion, - 'currentYear' => date('Y'), - 'currentStep' => $step, - 'documentationUrl' => System::getDocumentationUrl(), - 'configTableNotAvailable' => $update->isConfigTableNotAvailable($faqConfig->getDb()), -]; - -echo $template->render($templateVars); diff --git a/phpmyfaq/src/Bootstrap.php b/phpmyfaq/src/Bootstrap.php index 9344f5c531..ab101959eb 100644 --- a/phpmyfaq/src/Bootstrap.php +++ b/phpmyfaq/src/Bootstrap.php @@ -94,7 +94,7 @@ // Check if config/database.php exist -> if not, redirect to installer // if (!file_exists(PMF_CONFIG_DIR . '/database.php') && !file_exists(PMF_LEGACY_CONFIG_DIR . '/database.php')) { - $response = new RedirectResponse('./setup/index.php'); + $response = new RedirectResponse('./setup/'); $response->send(); } else { if (file_exists(PMF_CONFIG_DIR . '/database.php')) { diff --git a/phpmyfaq/src/admin-routes.php b/phpmyfaq/src/admin-routes.php index 5fb4764874..5c87ad2127 100644 --- a/phpmyfaq/src/admin-routes.php +++ b/phpmyfaq/src/admin-routes.php @@ -30,6 +30,7 @@ use phpMyFAQ\Controller\Administration\ImageController; use phpMyFAQ\Controller\Administration\InstanceController; use phpMyFAQ\Controller\Administration\MarkdownController; +use phpMyFAQ\Controller\Administration\NewsController; use phpMyFAQ\Controller\Administration\QuestionController; use phpMyFAQ\Controller\Administration\SearchController; use phpMyFAQ\Controller\Administration\SessionController; @@ -40,7 +41,6 @@ use phpMyFAQ\Controller\Administration\UserController; use Symfony\Component\Routing\Route; use Symfony\Component\Routing\RouteCollection; -use phpMyFAQ\Controller\Administration\NewsController; $routes = new RouteCollection(); diff --git a/phpmyfaq/src/api-routes.php b/phpmyfaq/src/api-routes.php index 5bb42955a5..e58c4d0f2b 100644 --- a/phpmyfaq/src/api-routes.php +++ b/phpmyfaq/src/api-routes.php @@ -29,6 +29,7 @@ use phpMyFAQ\Controller\Api\QuestionController; use phpMyFAQ\Controller\Api\RegistrationController; use phpMyFAQ\Controller\Api\SearchController; +use phpMyFAQ\Controller\Api\SetupController; use phpMyFAQ\Controller\Api\TagController; use phpMyFAQ\Controller\Api\TitleController; use phpMyFAQ\Controller\Api\VersionController; @@ -43,7 +44,6 @@ use phpMyFAQ\Controller\Frontend\ShareController; use phpMyFAQ\Controller\Frontend\UserController; use phpMyFAQ\Controller\Frontend\VotingController; -use phpMyFAQ\Controller\Setup\SetupController; use phpMyFAQ\System; use Symfony\Component\Routing\Route; use Symfony\Component\Routing\RouteCollection; diff --git a/phpmyfaq/src/phpMyFAQ/Application.php b/phpmyfaq/src/phpMyFAQ/Application.php index d136e5934a..443a730fc3 100644 --- a/phpmyfaq/src/phpMyFAQ/Application.php +++ b/phpmyfaq/src/phpMyFAQ/Application.php @@ -32,7 +32,7 @@ readonly class Application { - public function __construct(private Configuration $configuration) + public function __construct(private ?Configuration $configuration = null) { } @@ -49,18 +49,22 @@ public function run(RouteCollection $routeCollection): void private function setLanguage(): string { - $language = new Language($this->configuration); - $currentLanguage = $language->setLanguageByAcceptLanguage(); + if ($this->configuration) { + $language = new Language($this->configuration); + $currentLanguage = $language->setLanguageByAcceptLanguage(); - if (Language::isASupportedLanguage($currentLanguage)) { - require sprintf('%s/language_%s.php', PMF_TRANSLATION_DIR, $currentLanguage); - } else { - require sprintf('%s/language_en.php', PMF_TRANSLATION_DIR); - } + if (Language::isASupportedLanguage($currentLanguage)) { + require sprintf('%s/language_%s.php', PMF_TRANSLATION_DIR, $currentLanguage); + } else { + require sprintf('%s/language_en.php', PMF_TRANSLATION_DIR); + } - $this->configuration->setLanguage($language); + $this->configuration->setLanguage($language); - return $currentLanguage; + return $currentLanguage; + } else { + return 'en'; + } } /** diff --git a/phpmyfaq/src/phpMyFAQ/Controller/AbstractController.php b/phpmyfaq/src/phpMyFAQ/Controller/AbstractController.php index cc4d6a9ad2..09b2146d6a 100644 --- a/phpmyfaq/src/phpMyFAQ/Controller/AbstractController.php +++ b/phpmyfaq/src/phpMyFAQ/Controller/AbstractController.php @@ -30,6 +30,7 @@ use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Response; use Symfony\Component\HttpKernel\Exception\UnauthorizedHttpException; +use Twig\Extension\DebugExtension; #[OA\Info( version: '3.0', @@ -67,6 +68,7 @@ public function render(string $pathToTwigFile, array $templateVars = [], Respons { $response ??= new Response(); $twigWrapper = new TwigWrapper(PMF_ROOT_DIR . '/assets/templates'); + $twigWrapper->addExtension(new DebugExtension()); $template = $twigWrapper->loadTemplate($pathToTwigFile); $response->setContent($template->render($templateVars)); diff --git a/phpmyfaq/src/phpMyFAQ/Controller/Setup/SetupController.php b/phpmyfaq/src/phpMyFAQ/Controller/Api/SetupController.php similarity index 98% rename from phpmyfaq/src/phpMyFAQ/Controller/Setup/SetupController.php rename to phpmyfaq/src/phpMyFAQ/Controller/Api/SetupController.php index 8a482d21d7..1cc6e6aa08 100644 --- a/phpmyfaq/src/phpMyFAQ/Controller/Setup/SetupController.php +++ b/phpmyfaq/src/phpMyFAQ/Controller/Api/SetupController.php @@ -15,9 +15,8 @@ * @since 2023-10-17 */ -namespace phpMyFAQ\Controller\Setup; +namespace phpMyFAQ\Controller\Api; -use phpMyFAQ\Configuration; use phpMyFAQ\Controller\AbstractController; use phpMyFAQ\Core\Exception; use phpMyFAQ\Database; diff --git a/phpmyfaq/src/phpMyFAQ/Controller/Frontend/SetupController.php b/phpmyfaq/src/phpMyFAQ/Controller/Frontend/SetupController.php new file mode 100644 index 0000000000..9bcff78f9d --- /dev/null +++ b/phpmyfaq/src/phpMyFAQ/Controller/Frontend/SetupController.php @@ -0,0 +1,146 @@ + + * @copyright 2024 phpMyFAQ Team + * @license https://www.mozilla.org/MPL/2.0/ Mozilla Public License Version 2.0 + * @link https://www.phpmyfaq.de + * @since 2024-06-01 + */ + +namespace phpMyFAQ\Controller\Frontend; + +use Elastic\Elasticsearch\Exception\AuthenticationException; +use phpMyFAQ\Configuration; +use phpMyFAQ\Core\Exception; +use phpMyFAQ\Filter; +use phpMyFAQ\Language\LanguageCodes; +use phpMyFAQ\Setup\Installer; +use phpMyFAQ\Setup\Update; +use phpMyFAQ\System; +use phpMyFAQ\Template\TemplateException; +use phpMyFAQ\Template\TwigWrapper; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\Response; +use Symfony\Component\Routing\Annotation\Route; +use Twig\Extension\DebugExtension; + +class SetupController +{ + /** + * @throws TemplateException + * @throws \Exception + */ + #[Route('/setup', name: 'public.setup.update')] + public function index(): Response + { + $system = new System(); + $installer = new Installer($system); + + $checkBasicError = ''; + + try { + $installer->checkBasicStuff(); + } catch (Exception $e) { + $checkBasicError = $e->getMessage(); + } + + return $this->render( + 'setup/index.twig', + [ + 'newVersion' => System::getVersion(), + 'setupType' => 'Setup', + 'currentYear' => date('Y'), + 'currentLanguage' => 'en', + 'documentationUrl' => System::getDocumentationUrl(), + 'checkBasicError' => $checkBasicError, + 'nonCriticalSettings' => $installer->checkNoncriticalSettings(), + 'supportedDatabases' => $system->getSupportedSafeDatabases(), + 'currentPath' => dirname(__DIR__, 4), + 'isLdapEnabled' => $installer->hasLdapSupport(), + 'isElasticsearchEnabled' => $installer->hasElasticsearchSupport(), + 'supportedTranslations' => LanguageCodes::getAllSupported(), + ] + ); + } + + /** + * @throws TemplateException + * @throws \Exception + */ + #[Route('/setup/install', name: 'public.setup.install')] + public function install(): Response + { + $system = new System(); + $installer = new Installer($system); + + $installationError = ''; + + try { + $installer->startInstall(); + } catch (Exception | AuthenticationException $e) { + $installationError = $e->getMessage(); + } + + return $this->render( + 'setup/install.twig', + [ + 'newVersion' => System::getVersion(), + 'setupType' => 'Setup', + 'currentYear' => date('Y'), + 'documentationUrl' => System::getDocumentationUrl(), + 'installationError' => $installationError, + ] + ); + } + + /** + * @throws TemplateException + */ + #[Route('/setup/update', name: 'public.setup.update')] + public function update(Request $request): Response + { + $currentStep = Filter::filterVar($request->get('step'), FILTER_VALIDATE_INT); + + $configuration = Configuration::getConfigurationInstance(); + + $update = new Update(new System(), $configuration); + + return $this->render( + 'setup/update.twig', + [ + 'currentStep' => $currentStep ?? 1, + 'installedVersion' => System::getVersion(), + 'newVersion' => System::getVersion(), + 'currentYear' => date('Y'), + 'documentationUrl' => System::getDocumentationUrl(), + 'configTableNotAvailable' => $update->isConfigTableNotAvailable($configuration->getDb()), + ] + ); + } + /** + * Returns a Twig rendered template as response. + * + * @param string[] $templateVars + * @param Response|null $response + * @throws TemplateException + */ + public function render(string $pathToTwigFile, array $templateVars = [], Response $response = null): Response + { + $response ??= new Response(); + $twigWrapper = new TwigWrapper(PMF_ROOT_DIR . '/assets/templates'); + $twigWrapper->addExtension(new DebugExtension()); + $template = $twigWrapper->loadTemplate($pathToTwigFile); + + $response->setContent($template->render($templateVars)); + + return $response; + } +} diff --git a/phpmyfaq/src/phpMyFAQ/Language.php b/phpmyfaq/src/phpMyFAQ/Language.php index 45421d32ac..4d4c4dd6e6 100644 --- a/phpmyfaq/src/phpMyFAQ/Language.php +++ b/phpmyfaq/src/phpMyFAQ/Language.php @@ -165,6 +165,24 @@ public function setLanguageByAcceptLanguage(): string return self::$language = $this->acceptLanguage; } + /** + * True if the language is supported by the current phpMyFAQ installation. + * + * @param string|null $langCode Language code + */ + public static function isASupportedLanguage(?string $langCode): bool + { + return $langCode !== null && LanguageCodes::getSupported($langCode) !== null; + } + + /** + * Returns the current language. + */ + public function getLanguage(): string + { + return self::$language; + } + /** * Gets the accepted language from the user agent. * @@ -193,22 +211,4 @@ private function getUserAgentLanguage(): void } } } - - /** - * True if the language is supported by the current phpMyFAQ installation. - * - * @param string|null $langCode Language code - */ - public static function isASupportedLanguage(?string $langCode): bool - { - return $langCode !== null && LanguageCodes::getSupported($langCode) !== null; - } - - /** - * Returns the current language. - */ - public function getLanguage(): string - { - return self::$language; - } } diff --git a/phpmyfaq/src/phpMyFAQ/Setup/Installer.php b/phpmyfaq/src/phpMyFAQ/Setup/Installer.php index fa39261a80..df57eac76d 100644 --- a/phpmyfaq/src/phpMyFAQ/Setup/Installer.php +++ b/phpmyfaq/src/phpMyFAQ/Setup/Installer.php @@ -621,7 +621,7 @@ public function checkBasicStuff(): void if (!$this->system->checkInstallation()) { throw new Exception( - 'phpMyFAQ is already installed! Please use the update.' + 'phpMyFAQ is already installed! Please use the update.' ); } } @@ -706,43 +706,47 @@ public function checkFilesystemPermissions(): void /** * Checks some non-critical settings and print some hints. * - * @todo We should return an array of messages + * @return string[] */ - public function checkNoncriticalSettings(): void + public function checkNoncriticalSettings(): array { + $hints = []; if (!$this->system->getHttpsStatus()) { - echo '

phpMyFAQ could not find HTTPS support. For security reasons we ' . - 'recommend activating HTTPS.

'; + $hints[] = '

phpMyFAQ could not find HTTPS support. For security reasons ' . + 'we recommend activating HTTPS.

'; } if (!extension_loaded('gd')) { - echo '

You don\'t have GD support enabled in your PHP installation. Please ' . - "enable GD support in your php.ini file otherwise you can't use Captchas for spam protection.

"; + $hints[] = '

You don\'t have GD support enabled in your PHP installation. ' . + "Please enable GD support in your php.ini file otherwise you can't use Captchas for spam protection." . + "

"; } if (!function_exists('imagettftext')) { - echo '

You don\'t have Freetype support enabled in the GD extension of ' . - 'your PHP installation. Please enable Freetype support in GD extension otherwise the Captchas ' . + $hints[] = '

You don\'t have Freetype support enabled in the GD extension ' . + ' ofyour PHP installation. Please enable Freetype support in GD extension otherwise the Captchas ' . 'for spam protection will be quite easy to break.

'; } if (!extension_loaded('curl') || !extension_loaded('openssl')) { - echo '

You don\'t have cURL and/or OpenSSL support enabled in your PHP ' . - "installation. Please enable cURL and/or OpenSSL support in your php.ini file otherwise you can't " . - 'use Elasticsearch.

'; + $hints[] = '

You don\'t have cURL and/or OpenSSL support enabled in your ' . + "PHP installation. Please enable cURL and/or OpenSSL support in your php.ini file otherwise you " . + " can't use Elasticsearch.

"; } if (!extension_loaded('fileinfo')) { - echo '

You don\'t have Fileinfo support enabled in your PHP installation. ' . - "Please enable Fileinfo support in your php.ini file otherwise you can't use our backup/restore " . - 'functionality.

'; + $hints[] = '

You don\'t have Fileinfo support enabled in your PHP ' . + "installation. Please enable Fileinfo support in your php.ini file otherwise you can't use our " . + 'backup/restore functionality.

'; } if (!extension_loaded('sodium')) { - echo '

You don\'t have Sodium support enabled in your PHP installation. ' . - "Please enable Sodium support in your php.ini file otherwise you can't use our backup/restore " . - 'functionality.

'; + $hints[] = '

You don\'t have Sodium support enabled in your PHP ' . + "installation. Please enable Sodium support in your php.ini file otherwise you can't use our " . + 'backup/restore functionality.

'; } + + return $hints; } /** @@ -753,6 +757,7 @@ public function checkNoncriticalSettings(): void */ public function startInstall(array|null $setup = null): void { + $feedbacks = []; $ldapSetup = []; $query = []; $uninstall = []; @@ -1233,4 +1238,14 @@ public function adjustRewriteBaseHtaccess(string $path): bool return file_put_contents($htaccessPath, implode('', $newLines)) !== false; } + + public function hasLdapSupport(): bool + { + return extension_loaded('ldap'); + } + + public function hasElasticsearchSupport(): bool + { + return extension_loaded('curl') && extension_loaded('openssl'); + } } diff --git a/phpmyfaq/src/phpMyFAQ/Sitemap.php b/phpmyfaq/src/phpMyFAQ/Sitemap.php index 70331d5625..862bdaefba 100644 --- a/phpmyfaq/src/phpMyFAQ/Sitemap.php +++ b/phpmyfaq/src/phpMyFAQ/Sitemap.php @@ -213,14 +213,10 @@ public function getRecordsFromLetter(string $letter = 'A'): string $permPart ); - var_dump($query); - $result = $this->configuration->getDb()->query($query); $oldId = 0; $parseDownExtra = new ParsedownExtra(); - var_dump($this->configuration->getDb()->fetchObject($result)); - while ($row = $this->configuration->getDb()->fetchObject($result)) { if ($oldId != $row->id) { $title = Strings::htmlspecialchars($row->thema, ENT_QUOTES); diff --git a/phpmyfaq/src/phpMyFAQ/System.php b/phpmyfaq/src/phpMyFAQ/System.php index 5bebde18a0..bb34f04d24 100644 --- a/phpmyfaq/src/phpMyFAQ/System.php +++ b/phpmyfaq/src/phpMyFAQ/System.php @@ -244,9 +244,9 @@ public function getAvailableTemplates(): array * * @return array */ - public function getSupportedSafeDatabases(bool $returnAsHtml = false): array + public function getSupportedSafeDatabases(): array { - $retVal = []; + $databases = []; foreach ($this->getSupportedDatabases() as $extension => $database) { if (!extension_loaded($extension)) { continue; @@ -254,14 +254,10 @@ public function getSupportedSafeDatabases(bool $returnAsHtml = false): array if (version_compare(PHP_VERSION, $database[0]) < 0) { continue; } - if ($returnAsHtml) { - $retVal[] = sprintf('', $extension, $database[1]); - } else { - $retVal[$extension] = $database; - } + $databases[$extension] = $database[1]; } - return $retVal; + return $databases; } /** diff --git a/phpmyfaq/src/phpMyFAQ/Template/TwigWrapper.php b/phpmyfaq/src/phpMyFAQ/Template/TwigWrapper.php index c1296c1048..ac5ea98501 100644 --- a/phpmyfaq/src/phpMyFAQ/Template/TwigWrapper.php +++ b/phpmyfaq/src/phpMyFAQ/Template/TwigWrapper.php @@ -17,6 +17,7 @@ namespace phpMyFAQ\Template; +use phpMyFAQ\Core\Exception; use phpMyFAQ\System; use Twig\Environment; use Twig\Error\LoaderError; @@ -44,14 +45,14 @@ public function __construct(string $templatePath) } /** - * @throws TemplateException + * @throws Exception */ public function loadTemplate(string $templateFile): TemplateWrapper { try { return $this->twigEnvironment->load($templateFile); } catch (LoaderError | RuntimeError | SyntaxError $exception) { - throw new TemplateException($exception->getMessage()); + throw new Exception($exception->getMessage()); } } diff --git a/phpmyfaq/src/public-routes.php b/phpmyfaq/src/public-routes.php new file mode 100644 index 0000000000..f1f2e3bdfb --- /dev/null +++ b/phpmyfaq/src/public-routes.php @@ -0,0 +1,29 @@ + + * @copyright 2024 phpMyFAQ Team + * @license https://www.mozilla.org/MPL/2.0/ Mozilla Public License Version 2.0 + * @link https://www.phpmyfaq.de + * @since 2024-05-31 + */ + +use Symfony\Component\Routing\Route; +use Symfony\Component\Routing\RouteCollection; + +$routes = new RouteCollection(); + +$routeDefinitions = []; + +foreach ($routeDefinitions as $name => [$path, $controller, $action]) { + $routes->add($name, new Route($path, ['_controller' => [$controller, $action]])); +} + +return $routes; diff --git a/phpmyfaq/update/index.php b/phpmyfaq/update/index.php new file mode 100644 index 0000000000..586c978799 --- /dev/null +++ b/phpmyfaq/update/index.php @@ -0,0 +1,38 @@ + + * @copyright 2024 phpMyFAQ Team + * @license https://www.mozilla.org/MPL/2.0/ Mozilla Public License Version 2.0 + * @link https://www.phpmyfaq.de + * @since 2024-05-31 + */ + +use phpMyFAQ\Application; +use phpMyFAQ\Configuration; +use phpMyFAQ\Controller\Frontend\SetupController; +use Symfony\Component\Routing\Route; +use Symfony\Component\Routing\RouteCollection; + +require '../src/Bootstrap.php'; + +$faqConfig = Configuration::getConfigurationInstance(); + +$routes = new RouteCollection(); +$routes->add('public.update.index', new Route('/', [SetupController::class, 'update'])); + +$app = new Application($faqConfig); +try { + $app->run($routes); +} catch (Exception $exception) { + echo $exception->getMessage(); +}