* @copyright Copyright (C) 2008-2018, Rémi Jean * @author Frédéric Tempez * @copyright Copyright (C) 2018-2023, Frédéric Tempez * @license CC Attribution-NonCommercial-NoDerivatives 4.0 International * @link http://zwiicms.fr/ */ class install extends common { public static $actions = [ 'index' => self::GROUP_VISITOR, "postinstall" => self::GROUP_VISITOR, 'steps' => self::GROUP_ADMIN, 'update' => self::GROUP_ADMIN ]; // Type de proxy public static $proxyType = [ 'tcp://' => 'TCP', 'http://' => 'HTTP' ]; // Thèmes proposés à l'installation public static $themes = []; public static $newVersion; // Fichiers des Interface public static $i18nFiles = []; /** * Pré-installation - choix de la langue */ public function index() { // Accès refusé if ($this->getData(['user']) !== []) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Accès autorisé // Soumission du formulaire if ($this->isPost()) { $lang = $this->getInput('installLanguage'); // Place le cookie pour la suite de l'installation setcookie('ZWII_UI', $lang, time() + 3600, helper::baseUrl(false, false), '', false, false); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'install/postinstall/' . $lang ]); } //Nettoyage anciennes installations helper::deleteCookie('ZWII_CONTENT'); // Liste des langues UI disponibles if (is_dir(self::I18N_DIR)) { foreach ($this->getData(['language']) as $lang => $value) { self::$i18nFiles[$lang] = self::$languages[$lang]; ; } } $this->addOutput([ 'display' => self::DISPLAY_LAYOUT_LIGHT, 'title' => helper::translate('Installation'), 'view' => 'index' ]); } /** * post Installation */ public function postInstall() { // Accès refusé if ($this->getData(['user']) !== []) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Accès autorisé else { // Soumission du formulaire if ($this->isPost()) { $success = true; // Validation de la langue transmise self::$i18nUI = $this->getUrl(2); self::$i18nUI = array_key_exists(self::$i18nUI, self::$languages) ? self::$i18nUI : 'fr_FR'; // par défaut le contenu est la langue d'installation self::$i18nContent = self::$i18nUI; setcookie('ZWII_CONTENT', self::$i18nContent, time() + 3600, helper::baseUrl(false, false), '', helper::isHttps(), true); // Double vérification pour le mot de passe if ($this->getInput('installPassword', helper::FILTER_STRING_SHORT, true) !== $this->getInput('installConfirmPassword', helper::FILTER_STRING_SHORT, true)) { self::$inputNotices['installConfirmPassword'] = 'Incorrect'; $success = false; } // Utilisateur $userFirstname = $this->getInput('installFirstname', helper::FILTER_STRING_SHORT, true); $userLastname = $this->getInput('installLastname', helper::FILTER_STRING_SHORT, true); $userMail = $this->getInput('installMail', helper::FILTER_MAIL, true); $userId = $this->getInput('installId', helper::FILTER_ID, true); // Création de l'utilisateur si les données sont complétées. // success retour de l'enregistrement des données $success = $this->setData([ 'user', $userId, [ 'firstname' => $userFirstname, 'forgot' => 0, 'group' => self::GROUP_ADMIN, 'lastname' => $userLastname, 'pseudo' => 'Admin', 'signature' => 1, 'mail' => $userMail, 'password' => $this->getInput('installPassword', helper::FILTER_PASSWORD, true), 'language' => self::$i18nUI ] ]); // Compte créé, envoi du mail et création des données du site if ($success) { // Formulaire complété envoi du mail // Envoie le mail // Sent contient true si réussite sinon code erreur d'envoi en clair $sent = $this->sendMail( $userMail, 'Installation de votre site', 'Bonjour' . ' ' . $userFirstname . ' ' . $userLastname . ',

' . 'Voici les détails de votre installation.

' . 'URL du site : ' . helper::baseUrl(false) . '
' . 'Identifiant du compte : ' . $this->getInput('installId') . '
', null, $this->getData(['config', 'smtp', 'from']), ); // Installation du site de test if ( $this->getInput('installDefaultData', helper::FILTER_BOOLEAN) === false && self::$i18nContent === 'fr_FR' ) { $this->initData('page', self::$i18nContent, true); $this->initData('module', self::$i18nContent, true); $this->setData(['module', 'blog', 'posts', 'mon-premier-article', 'userId', $userId]); $this->setData(['module', 'blog', 'posts', 'mon-deuxieme-article', 'userId', $userId]); $this->setData(['module', 'blog', 'posts', 'mon-troisieme-article', 'userId', $userId]); } // Jeu réduit pour les pages étrangères if (self::$i18nContent !== 'fr_FR') { $this->initData('page', self::$i18nContent, false); $this->initData('module', self::$i18nContent, false); // Supprime l'installation FR générée par défaut. $this->removeDir(self::DATA_DIR . 'fr_FR'); } // Sauvegarder la configuration du Proxy $this->setData(['config', 'proxyType', $this->getInput('installProxyType')]); $this->setData(['config', 'proxyUrl', $this->getInput('installProxyUrl')]); $this->setData(['config', 'proxyPort', $this->getInput('installProxyPort', helper::FILTER_INT)]); // Images exemples livrées dans tous les cas try { // Décompression dans le dossier de fichier temporaires if (file_exists(self::TEMP_DIR . 'files.tar.gz')) { unlink(self::TEMP_DIR . 'files.tar.gz'); } if (file_exists(self::TEMP_DIR . 'files.tar')) { unlink(self::TEMP_DIR . 'files.tar'); } copy('core/module/install/ressource/files.tar.gz', self::TEMP_DIR . 'files.tar.gz'); $pharData = new PharData(self::TEMP_DIR . 'files.tar.gz'); $pharData->decompress(); // Installation $pharData->extractTo(__DIR__ . '/../../../', null, true); } catch (Exception $e) { $success = $e->getMessage(); } // Nettoyage unlink(self::TEMP_DIR . 'files.tar.gz'); unlink(self::TEMP_DIR . 'files.tar'); // Créer le dossier des fontes if (!is_dir(self::DATA_DIR . 'font')) { mkdir(self::DATA_DIR . 'font'); } // Installation du thème sélectionné $dataThemes = json_decode(file_get_contents('core/module/install/ressource/themes/themes.json'), true); $dataThemes = $dataThemes['themes']; $themeFilename = $dataThemes[$this->getInput('installTheme', helper::FILTER_STRING_SHORT)]['filename']; if ($themeFilename !== '') { $theme = new theme; $theme->import('core/module/install/ressource/themes/' . $themeFilename); } // Copie des thèmes dans les fichiers if (!is_dir(self::FILE_DIR . 'source/theme')) { mkdir(self::FILE_DIR . 'source/theme'); } $this->copyDir('core/module/install/ressource/themes', self::FILE_DIR . 'source/theme'); unlink(self::FILE_DIR . 'source/theme/themes.json'); // Copie des langues de l'UI et génération de la base de données if (is_dir(self::I18N_DIR) === false) { mkdir(self::I18N_DIR); } // Créer la base de données des langues copy('core/module/install/ressource/i18n/language.json', self::DATA_DIR . 'language.json'); $this->copyDir('core/module/install/ressource/i18n', self::I18N_DIR); unlink(self::I18N_DIR . 'language.json'); // Fixe l'adresse from pour les envois d'email $this->setData(['config', 'smtp', 'from', 'no-reply@' . str_replace('www.', '', $_SERVER['HTTP_HOST'])]); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl(true) . $this->getData(['locale', 'homePageId']), 'notification' => $sent === true ? helper::translate('Installation terminée') : $sent, 'state' => ($sent === true && $success === true) ? true : null ]); } } // Affichage du formulaire // Récupération de la liste des thèmes $dataThemes = json_decode(file_get_contents('core/module/install/ressource/themes/themes.json'), true); $dataThemes = $dataThemes['themes']; self::$themes = helper::arrayColumn($dataThemes, 'name'); // Valeurs en sortie $this->addOutput([ 'display' => self::DISPLAY_LAYOUT_LIGHT, 'title' => helper::translate('Installation'), 'view' => 'postinstall' ]); } } /** * Étapes de mise à jour */ public function steps() { switch ($this->getInput('step', helper::FILTER_INT)) { // Préparation case 1: $success = true; $message = ''; // RAZ la mise à jour auto $this->setData(['core', 'updateAvailable', false]); // Backup du dossier Data helper::autoBackup(self::BACKUP_DIR, ['backup', 'tmp', 'file']); // Sauvegarde htaccess if ($this->getData(['config', 'autoUpdateHtaccess'])) { $success = copy('.htaccess', '.htaccess' . '.bak'); $message = 'Erreur de copie du fichier htaccess'; } // Nettoyage des fichiers d'installation précédents if (file_exists(self::TEMP_DIR . 'update.tar.gz') && $success) { $success = $success || unlink(self::TEMP_DIR . 'update.tar.gz'); $message = 'Impossible d\'effacer la mise à jour précédente'; } if (file_exists(self::TEMP_DIR . 'update.tar') && $success) { $success = $success || unlink(self::TEMP_DIR . 'update.tar'); $message = 'Impossible d\'effacer la mise à jour précédente'; } // Valeurs en sortie $this->addOutput([ 'display' => self::DISPLAY_JSON, 'content' => [ 'success' => $success, 'data' => $success ? null : json_encode($message) ] ]); break; // Téléchargement case 2: file_put_contents(self::TEMP_DIR . 'update.tar.gz', helper::getUrlContents(common::ZWII_UPDATE_URL . common::ZWII_UPDATE_CHANNEL . '/update.tar.gz')); $md5origin = helper::getUrlContents(common::ZWII_UPDATE_URL . common::ZWII_UPDATE_CHANNEL . '/update.md5'); $md5origin = explode(' ', $md5origin); $md5target = md5_file(self::TEMP_DIR . 'update.tar.gz'); // Valeurs en sortie $this->addOutput([ 'display' => self::DISPLAY_JSON, 'content' => [ 'success' => $md5origin[0] === $md5target, 'data' => $md5origin[0] === $md5target ? null : json_encode('Erreur de téléchargement ou de somme de contrôle') ] ]); break; // Installation case 3: $success = true; // Check la réécriture d'URL avant d'écraser les fichiers $rewrite = helper::checkRewrite(); // Décompression et installation try { // Décompression dans le dossier de fichier temporaires $pharData = new PharData(self::TEMP_DIR . 'update.tar.gz'); $pharData->decompress(); // Installation $pharData->extractTo(__DIR__ . '/../../../', null, true); } catch (Exception $e) { $success = $e->getMessage(); } // Nettoyage du dossier if (file_exists(self::TEMP_DIR . 'update.tar.gz')) { unlink(self::TEMP_DIR . 'update.tar.gz'); } if (file_exists(self::TEMP_DIR . 'update.tar')) { unlink(self::TEMP_DIR . 'update.tar'); } // Valeurs en sortie $this->addOutput([ 'display' => self::DISPLAY_JSON, 'content' => [ 'success' => $success, 'data' => $rewrite ] ]); break; // Configuration case 4: $success = true; $message = null; $rewrite = $this->getInput('data'); // Réécriture d'URL if ($rewrite === "true") { // Ajout des lignes dans le .htaccess $fileContent = file_get_contents('.htaccess'); $rewriteData = PHP_EOL . '# URL rewriting' . PHP_EOL . '' . PHP_EOL . "\tRewriteEngine on" . PHP_EOL . "\tRewriteBase " . helper::baseUrl(false, false) . PHP_EOL . "\tRewriteCond %{REQUEST_FILENAME} !-f" . PHP_EOL . "\tRewriteCond %{REQUEST_FILENAME} !-d" . PHP_EOL . "\tRewriteRule ^(.*)$ index.php?$1 [L]" . PHP_EOL . '' . PHP_EOL . '# URL rewriting' . PHP_EOL; $fileContent = str_replace('# URL rewriting', $rewriteData, $fileContent); $r = file_put_contents( '.htaccess', $fileContent ); $success = $r === false ? false : true; $message = $r === false ? 'Le fichier htaccess n\'a pas été restauré' : null; } // Recopie htaccess if ( $this->getData(['config', 'autoUpdateHtaccess']) && $success && file_exists('.htaccess.bak') ) { // L'écraser avec le backup $success = $success || copy('.htaccess.bak', '.htaccess'); // Effacer le backup unlink('.htaccess.bak'); } /** * Met à jour les dictionnaires des langues depuis les modèles installés */ // Langues installées $installedUI = $this->getData(['language']); // Langues disponibles avec la mise à jour $store = json_decode(file_get_contents('core/module/install/ressource/i18n/language.json'), true); $store = $store['language']; foreach ($installedUI as $key => $value) { if ($store[$key]['version'] > $value['version']) { echo copy('core/module/install/ressource/i18n/' . $key . '.json', self::I18N_DIR . $key . '.json'); $this->setData(['language', $key, $store[$key]]); } } // Valeurs en sortie $this->addOutput([ 'display' => self::DISPLAY_JSON, 'content' => [ 'success' => $success, 'data' => $message ] ]); } } /** * Mise à jour */ public function update() { // Nouvelle version self::$newVersion = helper::getUrlContents(common::ZWII_UPDATE_URL . common::ZWII_UPDATE_CHANNEL . '/version'); // Valeurs en sortie $this->addOutput([ 'display' => self::DISPLAY_LAYOUT_LIGHT, 'title' => helper::translate('Mise à jour'), 'view' => 'update' ]); } }