<?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">&times;</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">&times;</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">&times;</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>