ZwiiCMS, le gestionnaire de site Web sans base de données à installer.
Conçu en 2008 par Rémi Jean, le développement a été repris par Frédéric Tempez en 2018.
zwii cms nosql json flat file
https://www.zwiicms.fr
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
796 lines
28 KiB
796 lines
28 KiB
<?php |
|
|
|
/** |
|
* This file is part of Zwii. |
|
* |
|
* For full copyright and license information, please see the LICENSE |
|
* file that was distributed with this source code. |
|
* |
|
* @author Rémi Jean <remi.jean@outlook.com> |
|
* @copyright Copyright (C) 2008-2018, Rémi Jean |
|
* @author Frédéric Tempez <frederic.tempez@outlook.com> |
|
* @copyright Copyright (C) 2018-2021, Frédéric Tempez |
|
* @license GNU General Public License, version 3 |
|
* @link http://zwiicms.fr/ |
|
*/ |
|
|
|
class config extends common { |
|
|
|
public static $actions = [ |
|
'backup' => self::GROUP_ADMIN, |
|
'copyBackups'=> self::GROUP_ADMIN, |
|
'configMetaImage' => self::GROUP_ADMIN, |
|
'generateFiles' => self::GROUP_ADMIN, |
|
'index' => self::GROUP_ADMIN, |
|
'advanced' => self::GROUP_ADMIN, |
|
'restore' => self::GROUP_ADMIN, |
|
'updateBaseUrl' => self::GROUP_ADMIN, |
|
'script' => self::GROUP_ADMIN, |
|
'logReset' => self::GROUP_ADMIN, |
|
'logDownload'=> self::GROUP_ADMIN, |
|
'blacklistReset' => self::GROUP_ADMIN, |
|
'blacklistDownload' => self::GROUP_ADMIN, |
|
|
|
]; |
|
|
|
public static $timezones = [ |
|
'Pacific/Midway' => '(GMT-11:00) Midway Island', |
|
'US/Samoa' => '(GMT-11:00) Samoa', |
|
'US/Hawaii' => '(GMT-10:00) Hawaii', |
|
'US/Alaska' => '(GMT-09:00) Alaska', |
|
'US/Pacific' => '(GMT-08:00) Pacific Time (US & Canada)', |
|
'America/Tijuana' => '(GMT-08:00) Tijuana', |
|
'US/Arizona' => '(GMT-07:00) Arizona', |
|
'US/Mountain' => '(GMT-07:00) Mountain Time (US & Canada)', |
|
'America/Chihuahua' => '(GMT-07:00) Chihuahua', |
|
'America/Mazatlan' => '(GMT-07:00) Mazatlan', |
|
'America/Mexico_City' => '(GMT-06:00) Mexico City', |
|
'America/Monterrey' => '(GMT-06:00) Monterrey', |
|
'Canada/Saskatchewan' => '(GMT-06:00) Saskatchewan', |
|
'US/Central' => '(GMT-06:00) Central Time (US & Canada)', |
|
'US/Eastern' => '(GMT-05:00) Eastern Time (US & Canada)', |
|
'US/East-Indiana' => '(GMT-05:00) Indiana (East)', |
|
'America/Bogota' => '(GMT-05:00) Bogota', |
|
'America/Lima' => '(GMT-05:00) Lima', |
|
'America/Caracas' => '(GMT-04:30) Caracas', |
|
'Canada/Atlantic' => '(GMT-04:00) Atlantic Time (Canada)', |
|
'America/La_Paz' => '(GMT-04:00) La Paz', |
|
'America/Santiago' => '(GMT-04:00) Santiago', |
|
'Canada/Newfoundland' => '(GMT-03:30) Newfoundland', |
|
'America/Buenos_Aires' => '(GMT-03:00) Buenos Aires', |
|
'Greenland' => '(GMT-03:00) Greenland', |
|
'Atlantic/Stanley' => '(GMT-02:00) Stanley', |
|
'Atlantic/Azores' => '(GMT-01:00) Azores', |
|
'Atlantic/Cape_Verde' => '(GMT-01:00) Cape Verde Is.', |
|
'Africa/Casablanca' => '(GMT) Casablanca', |
|
'Europe/Dublin' => '(GMT) Dublin', |
|
'Europe/Lisbon' => '(GMT) Lisbon', |
|
'Europe/London' => '(GMT) London', |
|
'Africa/Monrovia' => '(GMT) Monrovia', |
|
'Europe/Amsterdam' => '(GMT+01:00) Amsterdam', |
|
'Europe/Belgrade' => '(GMT+01:00) Belgrade', |
|
'Europe/Berlin' => '(GMT+01:00) Berlin', |
|
'Europe/Bratislava' => '(GMT+01:00) Bratislava', |
|
'Europe/Brussels' => '(GMT+01:00) Brussels', |
|
'Europe/Budapest' => '(GMT+01:00) Budapest', |
|
'Europe/Copenhagen' => '(GMT+01:00) Copenhagen', |
|
'Europe/Ljubljana' => '(GMT+01:00) Ljubljana', |
|
'Europe/Madrid' => '(GMT+01:00) Madrid', |
|
'Europe/Paris' => '(GMT+01:00) Paris', |
|
'Europe/Prague' => '(GMT+01:00) Prague', |
|
'Europe/Rome' => '(GMT+01:00) Rome', |
|
'Europe/Sarajevo' => '(GMT+01:00) Sarajevo', |
|
'Europe/Skopje' => '(GMT+01:00) Skopje', |
|
'Europe/Stockholm' => '(GMT+01:00) Stockholm', |
|
'Europe/Vienna' => '(GMT+01:00) Vienna', |
|
'Europe/Warsaw' => '(GMT+01:00) Warsaw', |
|
'Europe/Zagreb' => '(GMT+01:00) Zagreb', |
|
'Europe/Athens' => '(GMT+02:00) Athens', |
|
'Europe/Bucharest' => '(GMT+02:00) Bucharest', |
|
'Africa/Cairo' => '(GMT+02:00) Cairo', |
|
'Africa/Harare' => '(GMT+02:00) Harare', |
|
'Europe/Helsinki' => '(GMT+02:00) Helsinki', |
|
'Europe/Istanbul' => '(GMT+02:00) Istanbul', |
|
'Asia/Jerusalem' => '(GMT+02:00) Jerusalem', |
|
'Europe/Kiev' => '(GMT+02:00) Kyiv', |
|
'Europe/Minsk' => '(GMT+02:00) Minsk', |
|
'Europe/Riga' => '(GMT+02:00) Riga', |
|
'Europe/Sofia' => '(GMT+02:00) Sofia', |
|
'Europe/Tallinn' => '(GMT+02:00) Tallinn', |
|
'Europe/Vilnius' => '(GMT+02:00) Vilnius', |
|
'Asia/Baghdad' => '(GMT+03:00) Baghdad', |
|
'Asia/Kuwait' => '(GMT+03:00) Kuwait', |
|
'Europe/Moscow' => '(GMT+03:00) Moscow', |
|
'Africa/Nairobi' => '(GMT+03:00) Nairobi', |
|
'Asia/Riyadh' => '(GMT+03:00) Riyadh', |
|
'Europe/Volgograd' => '(GMT+03:00) Volgograd', |
|
'Asia/Tehran' => '(GMT+03:30) Tehran', |
|
'Asia/Baku' => '(GMT+04:00) Baku', |
|
'Asia/Muscat' => '(GMT+04:00) Muscat', |
|
'Asia/Tbilisi' => '(GMT+04:00) Tbilisi', |
|
'Asia/Yerevan' => '(GMT+04:00) Yerevan', |
|
'Asia/Kabul' => '(GMT+04:30) Kabul', |
|
'Asia/Yekaterinburg' => '(GMT+05:00) Ekaterinburg', |
|
'Asia/Karachi' => '(GMT+05:00) Karachi', |
|
'Asia/Tashkent' => '(GMT+05:00) Tashkent', |
|
'Asia/Kolkata' => '(GMT+05:30) Kolkata', |
|
'Asia/Kathmandu' => '(GMT+05:45) Kathmandu', |
|
'Asia/Almaty' => '(GMT+06:00) Almaty', |
|
'Asia/Dhaka' => '(GMT+06:00) Dhaka', |
|
'Asia/Novosibirsk' => '(GMT+06:00) Novosibirsk', |
|
'Asia/Bangkok' => '(GMT+07:00) Bangkok', |
|
'Asia/Jakarta' => '(GMT+07:00) Jakarta', |
|
'Asia/Krasnoyarsk' => '(GMT+07:00) Krasnoyarsk', |
|
'Asia/Chongqing' => '(GMT+08:00) Chongqing', |
|
'Asia/Hong_Kong' => '(GMT+08:00) Hong Kong', |
|
'Asia/Irkutsk' => '(GMT+08:00) Irkutsk', |
|
'Asia/Kuala_Lumpur' => '(GMT+08:00) Kuala Lumpur', |
|
'Australia/Perth' => '(GMT+08:00) Perth', |
|
'Asia/Singapore' => '(GMT+08:00) Singapore', |
|
'Asia/Taipei' => '(GMT+08:00) Taipei', |
|
'Asia/Ulaanbaatar' => '(GMT+08:00) Ulaan Bataar', |
|
'Asia/Urumqi' => '(GMT+08:00) Urumqi', |
|
'Asia/Seoul' => '(GMT+09:00) Seoul', |
|
'Asia/Tokyo' => '(GMT+09:00) Tokyo', |
|
'Asia/Yakutsk' => '(GMT+09:00) Yakutsk', |
|
'Australia/Adelaide' => '(GMT+09:30) Adelaide', |
|
'Australia/Darwin' => '(GMT+09:30) Darwin', |
|
'Australia/Brisbane' => '(GMT+10:00) Brisbane', |
|
'Australia/Canberra' => '(GMT+10:00) Canberra', |
|
'Pacific/Guam' => '(GMT+10:00) Guam', |
|
'Australia/Hobart' => '(GMT+10:00) Hobart', |
|
'Australia/Melbourne' => '(GMT+10:00) Melbourne', |
|
'Pacific/Port_Moresby' => '(GMT+10:00) Port Moresby', |
|
'Australia/Sydney' => '(GMT+10:00) Sydney', |
|
'Asia/Vladivostok' => '(GMT+10:00) Vladivostok', |
|
'Asia/Magadan' => '(GMT+11:00) Magadan', |
|
'Pacific/Auckland' => '(GMT+12:00) Auckland', |
|
'Pacific/Fiji' => '(GMT+12:00) Fiji', |
|
'Asia/Kamchatka' => '(GMT+12:00) Kamchatka' |
|
]; |
|
// Type de proxy |
|
public static $proxyType = [ |
|
'tcp://' => 'TCP', |
|
'http://' => 'HTTP' |
|
]; |
|
// Authentification SMTP |
|
public static $SMTPauth = [ |
|
true => 'Oui', |
|
false => 'Non' |
|
]; |
|
// Encryptation SMTP |
|
public static $SMTPEnc = [ |
|
'' => 'Aucune', |
|
'tls' => 'START TLS', |
|
'ssl' => 'SSL/TLS' |
|
]; |
|
// Sécurité de la connexion - tentative max avant blocage |
|
public static $connectAttempt = [ |
|
999 => 'Sécurité désactivée', |
|
3 => '3 tentatives', |
|
5 => '5 tentatives', |
|
10 => '10 tentatives' |
|
]; |
|
// Sécurité de la connexion - durée du blocage |
|
public static $connectTimeout = [ |
|
0 => 'Sécurité désactivée', |
|
300 => '5 minutes', |
|
600 => '10 minutes', |
|
900 => '15 minutes' |
|
]; |
|
// Langue traduite courante |
|
public static $i18nSite = 'fr'; |
|
|
|
/** |
|
* Génére les fichiers pour les crawlers |
|
* Sitemap compressé et non compressé |
|
* Robots.txt |
|
*/ |
|
public function generateFiles() { |
|
|
|
// Mettre à jour le site map |
|
$successSitemap=$this->createSitemap(); |
|
|
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'notification' => $successSitemap ? 'Mises à jour des fichiers sitemap et robots.txt' : 'Echec d\'écriture, le site map n\'a pas été mis à jour', |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'state' => $successSitemap |
|
]); |
|
} |
|
|
|
|
|
/** |
|
* Sauvegarde des données |
|
*/ |
|
public function backup() { |
|
// Soumission du formulaire |
|
if($this->isPost()) { |
|
// Creation du ZIP |
|
$filter = $this->getInput('configBackupOption',helper::FILTER_BOOLEAN) === true ? ['backup','tmp'] : ['backup','tmp','file']; |
|
$fileName = helper::autoBackup(self::TEMP_DIR,$filter); |
|
// Créer le répertoire manquant |
|
if (!is_dir(self::FILE_DIR.'source/backup')) { |
|
mkdir(self::FILE_DIR.'source/backup', 0755); |
|
} |
|
// Copie dans les fichiers |
|
$success = copy (self::TEMP_DIR . $fileName , self::FILE_DIR.'source/backup/' . $fileName); |
|
// Détruire le temporaire |
|
unlink(self::TEMP_DIR . $fileName); |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'display' => self::DISPLAY_JSON, |
|
'content' => json_encode($success) |
|
]); |
|
} else { |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'title' => 'Sauvegarder', |
|
'view' => 'backup' |
|
]); |
|
} |
|
} |
|
|
|
/** |
|
* Réalise une copie d'écran du site |
|
* https://www.codexworld.com/capture-screenshot-website-url-php-google-api/ |
|
*/ |
|
public function configMetaImage() { |
|
// fonction désactivée pour un site local |
|
if ( strpos(helper::baseUrl(false),'localhost') > 0 OR strpos(helper::baseUrl(false),'127.0.0.1') > 0) { |
|
$site = 'https://zwiicms.fr/'; } else { |
|
$site = helper::baseUrl(false); } |
|
|
|
$success= false; |
|
$googlePagespeedData = helper::urlGetContents('https://www.googleapis.com/pagespeedonline/v5/runPagespeed?url='. $site .'&screenshot=true'); |
|
if ($googlePagespeedData !== false) { |
|
$googlePagespeedData = json_decode($googlePagespeedData, true); |
|
$data = str_replace('_','/',$googlePagespeedData['lighthouseResult']['audits']['final-screenshot']['details']['data']); |
|
$data = str_replace('-','+',$data); |
|
$img = base64_decode(preg_replace('#^data:image/\w+;base64,#i', '', $data)); |
|
// Effacer l'image et la miniature png |
|
if (file_exists(self::FILE_DIR.'thumb/screenshot.jpg')) { |
|
unlink (self::FILE_DIR.'thumb/screenshot.jpg'); |
|
} |
|
if (file_exists(self::FILE_DIR.'source/screenshot.jpg')) { |
|
unlink (self::FILE_DIR.'source/screenshot.jpg'); |
|
} |
|
$success = file_put_contents( self::FILE_DIR.'source/screenshot.jpg',$img) ; |
|
|
|
} |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'notification' => $success === false ? 'Service inaccessible ou erreur d\'écriture de l\'image' : 'Image générée avec succès', |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'state' => $success === false ? false : true |
|
]); |
|
} |
|
|
|
/** |
|
* Procédure d'importation |
|
*/ |
|
public function restore() { |
|
// Soumission du formulaire |
|
if($this->isPost()) { |
|
//if ($this->getInput('configRestoreImportFile')) |
|
$fileZip = $this->getInput('configRestoreImportFile'); |
|
$file_parts = pathinfo($fileZip); |
|
$folder = date('Y-m-d-h-i-s', time()); |
|
$zip = new ZipArchive(); |
|
if ($file_parts['extension'] !== 'zip') { |
|
// Valeurs en sortie erreur |
|
$this->addOutput([ |
|
'notification' => 'Le fichier n\'est pas une archive valide', |
|
'redirect' => helper::baseUrl() . 'config/restore', |
|
'state' => false |
|
]); |
|
} |
|
$successOpen = $zip->open(self::FILE_DIR . 'source/' . $fileZip); |
|
if ($successOpen === FALSE) { |
|
// Valeurs en sortie erreur |
|
$this->addOutput([ |
|
'notification' => 'Impossible de lire l\'archive', |
|
'redirect' => helper::baseUrl() . 'config/restore', |
|
'state' => false |
|
]); |
|
} |
|
// Lire le contenu de l'archive dans le tableau files |
|
for( $i = 0; $i < $zip->numFiles; $i++ ){ |
|
$stat = $zip->statIndex( $i ); |
|
$files [] = ( basename( $stat['name'] )); |
|
} |
|
|
|
// Lire la dataversion |
|
$tmpDir = uniqid(4); |
|
$success = $zip->extractTo( self::TEMP_DIR . $tmpDir ); |
|
$data = file_get_contents( self::TEMP_DIR . $tmpDir . '/data/core.json'); |
|
$obj = json_decode($data); |
|
$dataVersion = strval ($obj->core->dataVersion); |
|
switch (strlen($dataVersion)) { |
|
case 4: |
|
if (substr($dataVersion,0,1) === '9' ) { |
|
$version = 9; |
|
} else { |
|
$version = 0; |
|
} |
|
break; |
|
case 5: |
|
$version = substr($dataVersion,0,2); |
|
break; |
|
default: |
|
$version = 0; |
|
break; |
|
} |
|
$this->removeDir(self::TEMP_DIR . $tmpDir ); |
|
|
|
if ($version >= 10 ) { |
|
// Option active, les users sont stockées |
|
if ($this->getInput('configRestoreImportUser', helper::FILTER_BOOLEAN) === true ) { |
|
$users = $this->getData(['user']); |
|
} |
|
} elseif ($version === 0) { // Version invalide |
|
// Valeurs en sortie erreur |
|
$this->addOutput([ |
|
'notification' => 'Cette archive n\'est pas une sauvegarde valide', |
|
'redirect' => helper::baseUrl() . 'config/restore', |
|
'state' => false |
|
]); |
|
} |
|
// Préserver les comptes des utilisateurs d'une version 9 si option cochée |
|
// Positionnement d'une variable de session lue au constructeurs |
|
if ($version === 9) { |
|
$_SESSION['KEEP_USERS'] = $this->getInput('configRestoreImportUser', helper::FILTER_BOOLEAN); |
|
} |
|
// Extraire le zip ou 'site/' |
|
$this->removeDir(self::DATA_DIR); |
|
$success = $zip->extractTo( 'site/' ); |
|
// Fermer l'archive |
|
$zip->close(); |
|
|
|
|
|
// Restaurer les users originaux d'une v10 si option cochée |
|
if (!empty($users) && |
|
$version >= 10 && |
|
$this->getInput('configRestoreImportUser', helper::FILTER_BOOLEAN) === true) { |
|
$this->setData(['user',$users]); |
|
} |
|
// Message de notification |
|
$notification = $success === true ? 'Restauration réalisée avec succès' : 'Erreur inconnue'; |
|
$redirect = $this->getInput('configRestoreImportUser', helper::FILTER_BOOLEAN) === true ? helper::baseUrl() . 'config/restore' : helper::baseUrl() . 'user/login/'; |
|
// Valeurs en sortie erreur |
|
$this->addOutput([ |
|
'notification' => $notification, |
|
'redirect' =>$redirect, |
|
'state' => $success |
|
]); |
|
} |
|
|
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'title' => 'Restaurer', |
|
'view' => 'restore' |
|
]); |
|
} |
|
|
|
|
|
/** |
|
* Configuration |
|
*/ |
|
public function index() { |
|
// Soumission du formulaire |
|
if($this->isPost()) { |
|
|
|
// Répercuter la suppression de la page dans la configuration du footer |
|
if ( $this->getData(['theme','footer','displaySearch']) === true |
|
AND $this->getInput('configSearchPageId') === 'none' |
|
){ |
|
$this->setData(['theme', 'footer', 'displaySearch', false]); |
|
} |
|
if ( $this->getData(['theme','footer','displayLegal']) === true |
|
AND $this->getInput('configLegalPageId') === 'none' |
|
){ |
|
$this->setData(['theme', 'footer', 'displayLegal', false]); |
|
} |
|
|
|
// Sauvegarder |
|
$this->setData([ |
|
'locale', |
|
[ |
|
'homePageId' => $this->getInput('configHomePageId', helper::FILTER_ID, true), |
|
'page404' => $this->getInput('configPage404'), |
|
'page403' => $this->getInput('configPage403'), |
|
'page302' => $this->getInput('configPage302'), |
|
'legalPageId' => $this->getInput('configLegalPageId'), |
|
'searchPageId' => $this->getInput('configSearchPageId'), |
|
'searchPageLabel' => empty($this->getInput('configSearchPageLabel', helper::FILTER_STRING_SHORT)) ? 'Rechercher' : $this->getInput('configSearchPageLabel', helper::FILTER_STRING_SHORT), |
|
'legalPageLabel' => empty($this->getInput('configLegalPageLabel', helper::FILTER_STRING_SHORT)) ? 'Mentions légales' : $this->getInput('configLegalPageLabel', helper::FILTER_STRING_SHORT), |
|
'sitemapPageLabel' => empty($this->getInput('configSitemapPageLabel', helper::FILTER_STRING_SHORT)) ? 'Plan du site' : $this->getInput('configSitemapPageLabel', helper::FILTER_STRING_SHORT), |
|
'metaDescription' => $this->getInput('configMetaDescription', helper::FILTER_STRING_LONG, true), |
|
'title' => $this->getInput('configTitle', helper::FILTER_STRING_SHORT, true) |
|
] |
|
]); |
|
|
|
$this->setData(['config', 'i18n', 'enable', $this->getInput('configI18n',helper::FILTER_BOOLEAN) ]); |
|
|
|
// Générer robots.txt et sitemap |
|
$this->generateFiles(); |
|
|
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . $this->getUrl(), |
|
'notification' => 'Modifications enregistrées', |
|
'state' => true |
|
]); |
|
} |
|
|
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'title' => 'Configuration', |
|
'view' => 'index' |
|
]); |
|
} |
|
|
|
/** |
|
* Configuration avancée |
|
*/ |
|
public function advanced() { |
|
// Soumission du formulaire |
|
if($this->isPost()) { |
|
// Basculement en mise à jour auto |
|
// Remise à 0 du compteur |
|
if ($this->getData(['config','autoUpdate']) === false && |
|
$this->getInput('configAdvancedAutoUpdate', helper::FILTER_BOOLEAN) === true) { |
|
$this->setData(['core','lastAutoUpdate',0]); |
|
} |
|
// Eviter déconnexion automatique après son activation |
|
if ( $this->getData(['config','autoDisconnect']) === false |
|
AND $this->getInput('configAdvancedAutoDisconnect',helper::FILTER_BOOLEAN) === true ) { |
|
$this->setData(['user',$this->getuser('id'),'accessCsrf',$_SESSION['csrf']]); |
|
} |
|
// Sauvegarder |
|
$this->setData([ |
|
'config', |
|
[ |
|
'analyticsId' => $this->getInput('configAdvancedAnalyticsId'), |
|
'autoBackup' => $this->getInput('configAdvancedAutoBackup', helper::FILTER_BOOLEAN), |
|
'maintenance' => $this->getInput('configAdvancedMaintenance', helper::FILTER_BOOLEAN), |
|
'cookieConsent' => $this->getInput('configAdvancedCookieConsent', helper::FILTER_BOOLEAN), |
|
'favicon' => $this->getInput('configAdvancedFavicon'), |
|
'faviconDark' => $this->getInput('configAdvancedFaviconDark'), |
|
'social' => [ |
|
'facebookId' => $this->getInput('configAdvancedSocialFacebookId'), |
|
'linkedinId' => $this->getInput('configAdvancedSocialLinkedinId'), |
|
'instagramId' => $this->getInput('configAdvancedSocialInstagramId'), |
|
'pinterestId' => $this->getInput('configAdvancedSocialPinterestId'), |
|
'twitterId' => $this->getInput('configAdvancedSocialTwitterId'), |
|
'youtubeId' => $this->getInput('configAdvancedSocialYoutubeId'), |
|
'youtubeUserId' => $this->getInput('configAdvancedSocialYoutubeUserId'), |
|
'githubId' => $this->getInput('configAdvancedSocialGithubId') |
|
], |
|
'timezone' => $this->getInput('configAdvancedTimezone', helper::FILTER_STRING_SHORT, true), |
|
'autoUpdate' => $this->getInput('configAdvancedAutoUpdate', helper::FILTER_BOOLEAN), |
|
'autoUpdateHtaccess' => $this->getInput('configAdvancedAutoUpdateHtaccess', helper::FILTER_BOOLEAN), |
|
'proxyType' => $this->getInput('configAdvancedProxyType'), |
|
'proxyUrl' => $this->getInput('configAdvancedProxyUrl'), |
|
'proxyPort' => $this->getInput('configAdvancedProxyPort',helper::FILTER_INT), |
|
'captchaStrong' => $this->getInput('configAdvancedCaptchaStrong',helper::FILTER_BOOLEAN), |
|
'autoDisconnect' => $this->getInput('configAdvancedAutoDisconnect',helper::FILTER_BOOLEAN), |
|
'smtp' => [ |
|
'enable' => $this->getInput('configAdvancedSmtpEnable',helper::FILTER_BOOLEAN), |
|
'host' => $this->getInput('configAdvancedSmtpHost',helper::FILTER_STRING_SHORT), |
|
'port' => $this->getInput('configAdvancedSmtpPort',helper::FILTER_INT), |
|
'auth' => $this->getInput('configAdvancedSmtpAuth',helper::FILTER_BOOLEAN), |
|
'secure' => $this->getInput('configAdvancedSmtpSecure'), |
|
'username' => $this->getInput('configAdvancedSmtpUsername',helper::FILTER_STRING_SHORT), |
|
'password' =>helper::encrypt($this->getData(['config','smtp','username']),$this->getInput('configAdvancedSmtpPassword')), |
|
'sender' => $this->getInput('configAdvancedSmtpSender',helper::FILTER_MAIL) |
|
], |
|
'seo' => [ |
|
'robots' => $this->getInput('configAdvancedSeoRobots',helper::FILTER_BOOLEAN) |
|
], |
|
'connect' => [ |
|
'attempt' => $this->getInput('configAdvancedConnectAttempt',helper::FILTER_INT), |
|
'timeout' => $this->getInput('configAdvancedConnectTimeout',helper::FILTER_INT), |
|
'log' => $this->getInput('configAdvancedConnectLog',helper::FILTER_BOOLEAN), |
|
'captcha' => $this->getInput('configAdvancedConnectCaptcha',helper::FILTER_BOOLEAN), |
|
], |
|
'i18n' => [ |
|
'enable' => $this->getData(['config', 'i18n', 'enable']) |
|
] |
|
] |
|
]); |
|
// Efface les fichiers de backup lorsque l'option est désactivée |
|
if ($this->getInput('configAdvancedFileBackup', helper::FILTER_BOOLEAN) === false) { |
|
$path = realpath('site/data'); |
|
foreach (new RecursiveIteratorIterator(new RecursiveDirectoryIterator($path)) as $filename) |
|
{ |
|
if (strpos($filename,'backup.json')) { |
|
unlink($filename); |
|
} |
|
} |
|
if (file_exists('site/data/.backup')) unlink('site/data/.backup'); |
|
} else { |
|
touch('site/data/.backup'); |
|
} |
|
// Notice |
|
if(self::$inputNotices === []) { |
|
// Active la réécriture d'URL |
|
$rewrite = $this->getInput('rewrite', helper::FILTER_BOOLEAN); |
|
if( |
|
$rewrite |
|
AND helper::checkRewrite() === false |
|
) { |
|
// Ajout des lignes dans le .htaccess |
|
file_put_contents( |
|
'.htaccess', |
|
PHP_EOL . |
|
'<ifModule mod_rewrite.c>' . 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 . |
|
'</ifModule>', |
|
FILE_APPEND |
|
); |
|
// Change le statut de la réécriture d'URL (pour le helper::baseUrl() de la redirection) |
|
helper::$rewriteStatus = true; |
|
} |
|
// Désactive la réécriture d'URL |
|
elseif( |
|
$rewrite === false |
|
AND helper::checkRewrite() |
|
) { |
|
// Suppression des lignes dans le .htaccess |
|
$htaccess = explode('# URL rewriting', file_get_contents('.htaccess')); |
|
file_put_contents('.htaccess', $htaccess[0] . '# URL rewriting'); |
|
// Change le statut de la réécriture d'URL (pour le helper::baseUrl() de la redirection) |
|
helper::$rewriteStatus = false; |
|
} |
|
// Met à jour la baseUrl |
|
$this->setData(['core', 'baseUrl', helper::baseUrl(true,false) ]); |
|
} |
|
// Générer robots.txt et sitemap |
|
$this->generateFiles(); |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . $this->getUrl(), |
|
'notification' => 'Modifications enregistrées ' , |
|
'state' => true |
|
]); |
|
} |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'title' => 'Configuration avancée', |
|
'view' => 'advanced' |
|
]); |
|
} |
|
|
|
public function script() { |
|
// Soumission du formulaire |
|
if($this->isPost()) { |
|
// Ecrire les fichiers de script |
|
if ($this->geturl(2) === 'head') { |
|
file_put_contents(self::DATA_DIR . 'head.inc.html',$this->getInput('configScriptHead',null)); |
|
} |
|
if ($this->geturl(2) === 'body') { |
|
file_put_contents(self::DATA_DIR . 'body.inc.html',$this->getInput('configScriptBody',null)); |
|
} |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'notification' => 'Modifications enregistrées', |
|
'redirect' => helper::baseUrl() . 'config/script/'. $this->geturl(2), |
|
'state' => true |
|
]); |
|
} |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'title' => 'Éditeur de script dans ' . ucfirst($this->geturl(2)) , |
|
'vendor' => [ |
|
'codemirror' |
|
], |
|
'view' => 'script' |
|
]); |
|
} |
|
|
|
/** |
|
* Met à jour les données de site avec l'adresse transmise |
|
*/ |
|
public function updateBaseUrl () { |
|
// Supprimer l'information de redirection |
|
$old = str_replace('?','',$this->getData(['core', 'baseUrl'])); |
|
$new = helper::baseUrl(false,false); |
|
$c3 = 0; |
|
$success = false ; |
|
// Boucler sur les pages |
|
foreach($this->getHierarchy(null,null,null) as $parentId => $childIds) { |
|
$content = $this->getData(['page',$parentId,'content']); |
|
$replace = str_replace( 'href="' . $old , 'href="'. $new , stripslashes($content),$c1) ; |
|
$replace = str_replace( 'src="' . $old , 'src="'. $new , stripslashes($replace),$c2) ; |
|
|
|
if ($c1 > 0 || $c2 > 0) { |
|
$success = true; |
|
$this->setData(['page',$parentId,'content', $replace ]); |
|
$c3 += $c1 + $c2; |
|
} |
|
foreach($childIds as $childId) { |
|
$content = $this->getData(['page',$childId,'content']); |
|
$replace = str_replace( 'href="' . $old , 'href="'. $new , stripslashes($content),$c1) ; |
|
$replace = str_replace( 'src="' . $old , 'src="'. $new , stripslashes($replace),$c2) ; |
|
if ($c1 > 0 || $c2 > 0) { |
|
$success = true; |
|
$this->setData(['page',$childId,'content', $replace ]); |
|
$c3 += $c1 + $c2; |
|
} |
|
} |
|
} |
|
// Traiter les modules dont la redirection |
|
$content = $this->getdata(['module']); |
|
$replace = $this->recursive_array_replace('href="' . $old , 'href="'. $new, $content, $c1); |
|
$replace = $this->recursive_array_replace('src="' . $old , 'src="'. $new, $replace, $c2); |
|
if ($content !== $replace) { |
|
$this->setdata(['module',$replace]); |
|
$c3 += $c1 + $c2; |
|
$success = true; |
|
} |
|
// Mettre à jour la base URl |
|
$this->setData(['core','baseUrl',helper::baseUrl(true,false)]); |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'notification' => $success ? $c3. ' conversion' . ($c3 > 1 ? 's' : '') . ' effectuée' . ($c3 > 1 ? 's' : '') : 'Aucune conversion', |
|
'redirect' => helper::baseUrl() . 'config/restore', |
|
'state' => $success ? true : false |
|
]); |
|
} |
|
|
|
/** |
|
* Vider le fichier de log |
|
*/ |
|
|
|
public function logReset() { |
|
if ( file_exists(self::DATA_DIR . 'journal.log') ) { |
|
unlink(self::DATA_DIR . 'journal.log'); |
|
// Créer les en-têtes des journaux |
|
$d = 'Date;Heure;IP;Id;Action' . PHP_EOL; |
|
file_put_contents(self::DATA_DIR . 'journal.log',$d); |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'notification' => 'Journal réinitialisé avec succès', |
|
'state' => true |
|
]); |
|
} else { |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'notification' => 'Aucun journal à effacer', |
|
'state' => false |
|
]); |
|
} |
|
|
|
} |
|
|
|
|
|
|
|
/** |
|
* Télécharger le fichier de log |
|
*/ |
|
public function logDownload() { |
|
$fileName = self::DATA_DIR . 'journal.log'; |
|
if (file_exists($fileName)) { |
|
ob_start(); |
|
header('Content-Type: application/octet-stream'); |
|
header('Content-Disposition: attachment; filename="' . $fileName . '"'); |
|
header('Content-Length: ' . filesize($fileName)); |
|
ob_clean(); |
|
ob_end_flush(); |
|
readfile( $fileName); |
|
exit(); |
|
} else { |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'notification' => 'Aucun fichier journal à télécharger', |
|
'state' => false |
|
]); |
|
} |
|
} |
|
|
|
/** |
|
* Tableau des IP blacklistés |
|
*/ |
|
public function blacklistDownload () { |
|
ob_start(); |
|
$fileName = self::TEMP_DIR . 'blacklist.log'; |
|
$d = 'Date dernière tentative;Heure dernière tentative;Id;Adresse IP;Nombre d\'échecs' . PHP_EOL; |
|
file_put_contents($fileName,$d); |
|
if ( file_exists($fileName) ) { |
|
$d = $this->getData(['blacklist']); |
|
$data = ''; |
|
foreach ($d as $key => $item) { |
|
$data .= mb_detect_encoding(strftime('%d/%m/%y',$item['lastFail']), 'UTF-8', true) |
|
? strftime('%d/%m/%y',$item['lastFail']) . ';' . utf8_encode(strftime('%R',$item['lastFail'])) . ';' |
|
: utf8_encode(strftime('%d/%m/%y',$item['lastFail'])) . ';' . utf8_encode(strftime('%R',$item['lastFail'])) . ';' ; |
|
$data .= $key . ';' . $item['ip'] . ';' . $item['connectFail'] . PHP_EOL; |
|
} |
|
file_put_contents($fileName,$data,FILE_APPEND); |
|
header('Content-Type: application/octet-stream'); |
|
header('Content-Disposition: attachment; filename="' . $fileName . '"'); |
|
header('Content-Length: ' . filesize($fileName)); |
|
ob_clean(); |
|
ob_end_flush(); |
|
readfile( $fileName); |
|
unlink(self::TEMP_DIR . 'blacklist.log'); |
|
exit(); |
|
} else { |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'notification' => 'Aucune liste noire à télécharger', |
|
'state' => false |
|
]); |
|
} |
|
} |
|
|
|
/** |
|
* Réinitialiser les ip blacklistées |
|
*/ |
|
|
|
public function blacklistReset() { |
|
if ( file_exists(self::DATA_DIR . 'blacklist.json') ) { |
|
$this->setData(['blacklist',[]]); |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'notification' => 'Liste noire réinitialisée avec succès', |
|
'state' => true |
|
]); |
|
} else { |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'notification' => 'Pas de liste à effacer', |
|
'state' => false |
|
]); |
|
} |
|
} |
|
|
|
/** |
|
* Récupération des backups auto dans le gestionnaire de fichiers |
|
*/ |
|
public function copyBackups() { |
|
// Créer le répertoire manquant |
|
if (!is_dir(self::FILE_DIR.'source/backup')) { |
|
mkdir(self::FILE_DIR.'source/backup', 0755); |
|
} |
|
$this->copyDir(self::BACKUP_DIR, self::FILE_DIR . 'source/backup' ); |
|
// Valeurs en sortie |
|
$this->addOutput([ |
|
'redirect' => helper::baseUrl() . 'config/advanced', |
|
'notification' => 'Copie terminée', |
|
'state' => true |
|
]); |
|
} |
|
|
|
|
|
/** |
|
* Fonction de parcours des données de module |
|
* @param string $find donnée à rechercher |
|
* @param string $replace donnée à remplacer |
|
* @param array tableau à analyser |
|
* @param int count nombres d'occurrences |
|
* @return array avec les valeurs remplacées. |
|
*/ |
|
private function recursive_array_replace ($find, $replace, $array, &$count) { |
|
if (!is_array($array)) { |
|
return str_replace($find, $replace, $array, $count); |
|
} |
|
|
|
$newArray = []; |
|
foreach ($array as $key => $value) { |
|
$newArray[$key] = $this->recursive_array_replace($find, $replace, $value,$c); |
|
$count += $c; |
|
} |
|
return $newArray; |
|
} |
|
}
|
|
|