<?php session_start(); // === Constants === // const LANGUAGE = 'en'; const TRANSLATIONS = [ 'en' => array ( 'method' => 'Unpacking method', 'download' => 'Download file', 'unzip_it' => 'Unzip it', 'delete_it' => 'Delete it', 'zip_file' => 'zip file', 'zip_files' => 'zip files', 'msg_found_files' => 'Found %s in this directory.', 'msg_files_not_found' => 'There is no zip file in this directory.', 'msg_not_zip_file' => 'This %s is not a zip file.', 'msg_error_while_unzip' => 'Error while unzipping file %s.', 'msg_unzip_success' => 'File %s has been unziped.', 'msg_cannot_delete' => 'This file %s cannot be deleted.', 'msg_error_while_delete' => 'Error while deleting file %s.', 'msg_delete_success' => 'File %s has been deleted.', 'msg_missing_token' => 'Missing token.', 'msg_invalid_token' => 'Invalid token.', 'msg_warning_files_overwrite' => 'All unzipped files will be overwritten if they already exist.', 'msg_warning_file_delete' => 'File will be deleted permanently.', 'msg_warning_script_delete' => 'This script file will be deleted permanently.', 'msg_remind_to_delete' => 'Remember to delete this script when you are done.', 'msg_are_you_sure' => 'Are you sure?', 'msg_confirm_your_action' => 'Confirm your action.', 'msg_action_proceed' => 'Yes, proceed it', 'msg_action_close' => 'No, close it', )]; // === Helpers === // /** * TranslateHelper * * Helps manage translations. * * @author Robert Wierzchowski <revert@revert.pl> * @version 1.2.0 */ class TranslateHelper { private static $language; private static $defaultlanguage = 'en'; private static $availableLanguages = ['en', 'pl', 'de', 'es', 'ru']; private static $translationsFileName = 'src/lang/translations.php'; private static $translations; private function __construct() {} private static function findTranslation($key) { return (! empty(self::$translations[self::$language][$key])) ? self::$translations[self::$language][$key] : str_replace('_', ' ', ucfirst($key)); } private static function setLanguageFromGet($name) { if (! empty($_GET[$name])) { self::$language = (in_array($_GET[$name], self::$availableLanguages)) ? $_GET[$name] : self::$defaultlanguage; } } private static function setLanguageFromUri() { $args = explode('/', $_SERVER['REQUEST_URI']); $cleanArgs = array_filter($args, function ($value) { return $value !== ''; }); $lastArg = array_pop($cleanArgs); if (in_array($lastArg, self::$availableLanguages)) { self::$language = $lastArg; } } private static function setLanguageFromBrowser() { self::$availableLanguages = array_flip(self::$availableLanguages); $languagesWeight = []; preg_match_all('~([\w-]+)(?:[^,\d]+([\d.]+))?~', strtolower($_SERVER['HTTP_ACCEPT_LANGUAGE']), $matches, PREG_SET_ORDER); foreach ($matches as $match) { list($a, $b) = explode('-', $match[1]) + ['', '']; $value = isset($match[2]) ? (float) $match[2] : 1.0; if (isset(self::$availableLanguages[$match[1]])) { $languagesWeight[$match[1]] = $value; continue; } if (isset(self::$availableLanguages[$a])) { $languagesWeight[$a] = $value - 0.1; } } if ($languagesWeight) { arsort($languagesWeight); self::$language = key($languagesWeight); } } private static function setLanguage() { self::setLanguageFromGet('lang'); if (empty(self::$language)) { self::setLanguageFromUri(); } if (empty(self::$language)) { self::setLanguageFromBrowser(); } if (empty(self::$language)) { self::$language = self::$defaultlanguage; } if (! empty(LANGUAGE)) { self::$language = LANGUAGE; } } private static function readTranslations($fileName) { if (! file_exists($fileName)) { return false; } $extension = pathinfo($fileName, PATHINFO_EXTENSION); if ($extension != 'php') { return false; } return (include $fileName); } private static function setTranslations() { self::$translations = (! empty(TRANSLATIONS)) ? TRANSLATIONS : self::readTranslations(self::$translationsFileName); } public static function getTranslation($key) { self::setLanguage(); self::setTranslations(); return self::findTranslation($key); } public static function getLanguage() { self::setLanguage(); return self::$language; } } /** * Get translation from translate helper function */ function _t($key, ...$args) { $translation = TranslateHelper::getTranslation($key); $result = vsprintf($translation, $args); return $result; } /** * Get language from translate helper function */ function _lang() { return TranslateHelper::getLanguage(); } /** * Alias of htmlspecialchars helper function */ function _h($text) { return htmlspecialchars($text, ENT_COMPAT); } // === Classes === // /** * UnZipper * * Unzip zip files. One file server side simple unzipper with UI. * * @author Robert Wierzchowski <revert@revert.pl> * @version 1.2.0 */ class UnZipper { private $title = 'UnZipper'; private $dir = './'; private $zips = []; private $alertMessage; private $alertStatus; private $token; private $output; private $methods = [ 'zipArchive' => 'ZipArchive', 'execUnzip' => 'exec unzip', 'systemUnzip' => 'system unzip', ]; public function __construct() { $this->checkIfUnZip(); $this->checkIfDeleteFile(); $this->setToken(); $this->findZips(); $this->countZips(); } private function checkIfUnZip() { if (! empty($_POST['zipfile']) && $this->verifyToken($_POST['token'])) { $this->unZip($_POST['zipfile'], $_POST['method']); } } private function checkIfDeleteFile() { if (! empty($_POST['delfile']) && $this->verifyToken($_POST['token'])) { $this->deleteFile($_POST['delfile']); } } private function findZips() { $fileNames = scandir($this->dir); foreach ($fileNames as $fileName) { if ($this->checkZipFile($fileName)) { $this->zips[] = $fileName; } } } private function countZips() { if ($this->alertMessage) { return false; } $count = count($this->zips); if ($count) { $zipFile = ($count == 1) ? _t('zip_file') : _t('zip_files'); $this->alertMessage = _t('msg_found_files', '<strong>' . $count . ' ' . $zipFile . '</strong>'); $this->alertStatus = 'info'; } else { $this->alertMessage = _t('msg_files_not_found'); $this->alertStatus = 'warning'; } } private function checkZipFile($fileName) { if (! file_exists($fileName)) { return false; } $extension = pathinfo($fileName, PATHINFO_EXTENSION); if ($extension != 'zip') { return false; } return true; } private function unZip($zip, $method = null) { if (! $this->checkZipFile($zip) || ! in_array($zip, scandir($this->dir))) { $this->alertMessage = _t('msg_not_zip_file', '<strong>' . $zip . '</strong>'); $this->alertStatus = 'danger'; return false; } switch ($method) { case 'execUnzip': $unzipResult = $this->execUnzip($zip); break; case 'systemUnzip': $unzipResult = $this->systemUnzip($zip); break; default: $unzipResult = $this->unZipArchive($zip); } if (! $unzipResult) { $this->alertMessage = _t('msg_error_while_unzip', '<strong>' . $zip . '</strong>'); $this->alertStatus = 'danger'; return false; } $this->alertMessage = _t('msg_unzip_success', '<strong>' . $zip . '</strong>'); $this->alertStatus = 'success'; return true; } private function unZipArchive($fileName) { $dirPath = pathinfo(realpath($fileName), PATHINFO_DIRNAME); $zip = new ZipArchive; if ($zip->open($fileName) !== true) { return false; } $zip->extractTo($dirPath); $zip->close(); return true; } private function execUnzip($fileName) { if (! exec('unzip -o ' . $fileName, $output)) { return false; } $this->output = implode('<br>', $output); return true; } private function systemUnzip($fileName) { ob_start(); if (! system("unzip -o {$fileName}")) { return false; } $this->output = nl2br(ob_get_contents()); ob_end_clean(); return true; } private function deleteFile($fileName) { if (! $this->checkZipFile($fileName) && $fileName != basename(__FILE__)) { $this->alertMessage = _t('msg_cannot_delete', '<strong>' . $fileName . '</strong>'); $this->alertStatus = 'danger'; return false; } if (! unlink($fileName)) { $this->alertMessage = _t('msg_error_while_delete', '<strong>' . $fileName . '</strong>'); $this->alertStatus = 'danger'; return false; } $this->alertMessage = _t('msg_delete_success', '<strong>' . $fileName . '</strong>'); $this->alertStatus = 'success'; return true; } private function verifyToken($inputToken) { if (empty($inputToken)) { $this->alertMessage = _t('msg_missing_token'); $this->alertStatus = 'danger'; return false; } if (! hash_equals($_SESSION['token'], $inputToken)) { $this->alertMessage = _t('msg_invalid_token'); $this->alertStatus = 'danger'; return false; } return true; } private function setToken() { $_SESSION['token'] = md5(uniqid(rand(), true)); // PHP 7 only: bin2hex(random_bytes(32)); $this->token = $_SESSION['token']; } public function getToken() { return $this->token; } public function getZips() { return $this->zips; } public function getMessage() { return $this->alertMessage; } public function getStatus() { return $this->alertStatus; } public function getOutput() { return $this->output; } public function getScriptPath() { return $_SERVER['REQUEST_URI'] ; } public function getTitle() { return $this->title; } public function getMethods() { return $this->methods; } } // === Instances === // $unZipper = new UnZipper(); // === Template === // ?> <!DOCTYPE html> <html lang="<?=_lang()?>"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <title><?=$unZipper->getTitle()?></title> <link rel="stylesheet" type="text/css" media="screen" href="https://bootswatch.com/4/lumen/bootstrap.min.css"> <link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.0.8/css/all.css" integrity="sha384-3AB7yXWz4OeoZcPbieVW64vVXEwADiYyAEhwilzWsLw+9FgqpyjjStpPnpBO8o8S" crossorigin="anonymous"> <link rel="icon" href=""> <style> a:hover { text-decoration: none; } label { margin: 0; } .notification { position: absolute; top: 1rem; right: 1rem; } .output { max-height: 10rem; overflow: auto; } </style> </head> <body> <header> <div class="container"> <h1 class="page-title text-center my-3"> <a href="<?=$unZipper->getScriptPath()?>" title="<?=_h($unZipper->getTitle())?>"> <i class="fas fa-cube mr-1"></i> <?=$unZipper->getTitle()?> </a> </h1> </div> </header> <section id="body"> <div class="container"> <div class="notification-box"> <?php if ($unZipper->getMessage()): ?> <div class="notification alert alert-dismissible fade show alert-<?=_h($unZipper->getStatus())?>"> <button type="button" class="close" data-dismiss="alert">×</button> <?=$unZipper->getMessage()?> </div> <?php endif ?> </div> <form class="form-unzip" method="POST" action=""> <input type="hidden" name="token" value="<?=$unZipper->getToken()?>" /> <input type="hidden" name="zipfile" value="" /> <input type="hidden" name="delfile" value="" /> <?php if ($unZipper->getZips()): ?> <div class="form-control mb-3 d-flex justify-content-around align-items-center"> <strong><?=_t('method')?>:</strong> <?php $i = 1; foreach ($unZipper->getMethods() as $methodKey => $methodName): ?> <div class="form-check"> <label class="form-check-label"> <input class="form-check-input" type="radio" name="method" value="<?=$methodKey?>" <?=(empty($_POST['method']) && $i == 1 || ! empty($_POST['method']) && $_POST['method'] == $methodKey) ? 'checked' : ''?> /> <?=$methodName?> </label> </div> <?php $i++; endforeach; ?> </div> <ul class="list-group"> <?php foreach ($unZipper->getZips() as $key => $zip): ?> <li class="list-group-item clearfix"> <h2 class="text-nowrap float-left mb-0"> <a href="<?=$zip?>" title="<?=_t('download')?> <?=_h($zip)?>"> <i class="fas fa-file-archive mr-1"></i> <?=$zip?> </a> </h2> <input type="hidden" name="zipfiles[<?=$key?>]" value="<?=_h($zip)?>" /> <button type="submit" class="btn-unzip btn-modal btn btn-warning float-right mb-0" title="<?=_t('unzip_it')?>" data-modal-body="<?=_t('msg_warning_files_overwrite')?>"> <i class="fas fa-cubes mr-1"></i> <?=_t('unzip_it')?> </button> <input type="hidden" name="delfiles[<?=$key?>]" value="<?=_h($zip)?>" /> <button type="submit" class="btn-delete btn-modal btn btn-outline-danger float-right mb-0 mr-3" title="<?=_t('delete_it')?>" data-modal-body="<?=_t('msg_warning_file_delete')?>"> <i class="fas fa-trash-alt mr-1"></i> <?=_t('delete_it')?> </button> </li> <?php endforeach ?> </ul> <?php endif ?> <div class="output-box"> <?php if ($unZipper->getOutput()): ?> <div class="output alert alert-dismissible fade show alert-warning mt-3"> <button type="button" class="close" data-dismiss="alert">×</button> <?=$unZipper->getOutput()?> </div> <?php endif ?> </div> <div class="reminder-box mt-3 text-center"> <input type="hidden" name="delfiles[]" value="<?=basename(__FILE__)?>" /> <button type="button" class="btn-delete btn-modal btn btn-outline-warning" title="<?=_t('delete_it')?>" data-modal-body="<?=_t('msg_warning_script_delete')?>"> <i class="fas fa-exclamation-circle mr-1"></i> <?=_t('msg_remind_to_delete')?> </button> </div> </form> </div> </div> <div class="modal" tabindex="-1" role="dialog"> <div class="modal-dialog" role="document"> <div class="modal-content"> <div class="modal-header"> <h5 class="modal-title"><?=_t('msg_are_you_sure')?></h5> <button type="button" class="close" data-dismiss="modal" aria-label="Close"> <span aria-hidden="true">×</span> </button> </div> <div class="modal-body"> <?=_t('msg_confirm_your_action')?> </div> <div class="modal-footer d-flex justify-content-center"> <button type="button" class="btn btn-primary form-confirm"><?=_t('msg_action_proceed')?></button> <button type="button" class="btn btn-secondary" data-dismiss="modal"><?=_t('msg_action_close')?></button> </div> </div> </div> </div> <footer class="text-center my-3"> <div class="container"> Made by <a href="http://revert.pl" title="Full-Stack Developer" target="_blank"> Revert </a> </div> </footer> <script src="https://code.jquery.com/jquery-3.2.1.slim.min.js" integrity="sha384-KJ3o2DKtIkvYIK3UENzmM7KCkRr/rE9/Qpg6aAZGJwFDMVNA/GpGFF93hXpG5KkN" crossorigin="anonymous"></script> <script src="https://cdnjs.cloudflare.com/ajax/libs/popper.js/1.12.9/umd/popper.min.js" integrity="sha384-ApNbgh9B+Y1QKtv3Rn7W3mgPxhU9K/ScQsAP7hUibX39j7fakFPskvXusvfa0b4Q" crossorigin="anonymous"></script> <script src="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/js/bootstrap.min.js" integrity="sha384-JZR6Spejh4U02d8jOt6vLEHfe/JQGiRRSQQxSfFWpi1MquVdAyjUar5+76PVCmYl" crossorigin="anonymous"></script> <script> // Set zipfile name $('.btn-unzip').on('click', function (e) { e.preventDefault(); let zipfile = $(this).prev().val(); $('input[name=zipfile]').val(zipfile); $('.form-unzip').submit(); }); // Set delfile name $('.btn-delete').on('click', function (e) { e.preventDefault(); let delfile = $(this).prev().val(); $('input[name=delfile]').val(delfile); $('.form-unzip').submit(); }); // Set modal data $('.btn-modal').on('click', function (e) { let title = $(this).data('modal-title'); if (title) { $('.modal-title').html(title); } let body = $(this).data('modal-body'); if (body) { $('.modal-body').html(body); } }); // Form submit confirm with modal dialog $('.form-unzip').submit(function(e){ if ($(this).hasClass('confirmed')) { return true; } else { e.preventDefault(); $('.modal').modal('show'); } }); $('.form-confirm').on('click', function (e) { $('.form-unzip').addClass('confirmed'); $('.form-unzip').submit(); }); $('.modal').on('hidden.bs.modal', function (e) { $('input[name=zipfile]').val(''); $('input[name=delfile]').val(''); }); // Notification auto close let delay = 5000; // 5 s setTimeout(function(){ $('.notification').alert('close'); }, delay); </script> </body> </html>