Skip to content
Snippets Groups Projects
unpack.php 7.35 KiB
Newer Older
  • Learn to ignore specific revisions
  • Jared Hancock's avatar
    Jared Hancock committed
    <?php
    
    require_once dirname(__file__) . "/class.module.php";
    
    class Unpacker extends Module {
    
    Jared Hancock's avatar
    Jared Hancock committed
        var $prologue = "Unpacks osTicket into target install path";
    
        var $epilog =
            "Copies an unpacked osticket tarball or zipfile into a production
    
             location, optionally placing the include/ folder in a separate
    
    Jared Hancock's avatar
    Jared Hancock committed
             location if requested";
    
        var $options = array(
            'include' => array('-i','--include', 'metavar'=>'path', 'help'=>
                "The include/ folder, which contains the bulk of osTicket's source
                 code can be located outside of the install path. This is recommended
                 for better security. If you would like to install the include/
                 folder somewhere else, give the path here. Note that the full
                 path is assumed, so give path/to/include/ to unpack the source
                 code in that folder. The folder will be automatically created if
                 it doesn't already exist."
            ),
    
            'verbose' => array('-v','--verbose', 'default'=>false, 'nargs'=>0,
                'action'=>'store_true', 'help'=>
                "Move verbose logging to stdout"),
    
    Jared Hancock's avatar
    Jared Hancock committed
        );
    
        var $arguments = array(
            'install-path' =>
                "The destination for osTicket to reside. Use the --include
                 option to specify destination of the include/ folder, if the
                 administrator should chose to locate it separate from the
                 main installation path.",
        );
    
        function find_upload_folder() {
            # Hop up to the root folder
            $start = dirname(__file__);
            for (;;) {
                if (is_dir($start . '/upload')) break;
                $start .= '/..';
            }
            return realpath($start.'/upload');
        }
    
        function change_include_dir($include_path) {
            # Read the main.inc.php script
            $main_inc_php = $this->destination . '/main.inc.php';
            $lines = explode("\n", file_get_contents($main_inc_php));
    
            # Try and use ROOT_DIR
    
    Jared Hancock's avatar
    Jared Hancock committed
            if (strpos($include_path, $this->destination) === 0)
    
                $include_path = "ROOT_DIR . '" .
    
    Jared Hancock's avatar
    Jared Hancock committed
                    str_replace($this->destination, '', $include_path) . "'";
            else
                $include_path = "'$include_path'";
            # Find the line that defines INCLUDE_DIR
            foreach ($lines as &$line) {
                if (preg_match("/(\s*)define\s*\(\s*'INCLUDE_DIR'/", $line, $match)) {
                    # Replace the definition with the new locatin
                    $line = $match[1] . "define('INCLUDE_DIR', "
                        . $include_path
                        . "); // Set by installer";
                    break;
                }
            }
            if (!file_put_contents($main_inc_php, implode("\n", $lines)))
                die("Unable to configure location of INCLUDE_DIR in main.inc.php\n");
        }
    
        function exclude($pattern, $match) {
            if (!$pattern) {
                return false;
            } elseif (is_array($pattern)) {
                foreach ($pattern as $p)
                    if (fnmatch($p, $match))
                        return true;
            } else {
                return fnmatch($pattern, $match);
            }
            return false;
        }
    
    
        /**
         * Copy from source to desination, perhaps recursing up to n folders.
         * Exclusions are also permitted. If any files match an MD5 sum, they
         * will be excluded from the copy operation.
         *
         * Parameters:
         * folder - (string) source folder root
         * destination - (string) destination folder root
         * recurse - (int) recuse up to this many folders. Use 0 or false to
         *      disable recursion, and -1 to recurse infinite folders.
         * exclude - (string | array<string>) patterns that will be matched
         *      using the PHP `fnmatch` function. If any file or folder matches,
         *      it will be excluded from the copy procedure. Omit or use false
         *      to disable exclusions
         */
        function unpackage($folder, $destination, $recurse=0, $exclude=false) {
            $verbose = $this->getOption('verbose');
            if (substr($destination, -1) !== '/')
                $destination .= '/';
    
    Jared Hancock's avatar
    Jared Hancock committed
            foreach (glob($folder, GLOB_BRACE|GLOB_NOSORT) as $file) {
                if ($this->exclude($exclude, $file))
                    continue;
                if (is_file($file)) {
                    if (!is_dir($destination))
                        mkdir($destination, 0751, true);
    
                    $target = $destination . basename($file);
                    if (is_file($target) && md5_file($target) == md5_file($file))
                        continue;
                    if ($verbose)
                        $this->stdout->write($target."\n");
                    copy($file, $target);
    
    Jared Hancock's avatar
    Jared Hancock committed
                }
            }
            if ($recurse) {
    
    Jared Hancock's avatar
    Jared Hancock committed
                foreach (glob(dirname($folder).'/'.basename($folder),
                        GLOB_BRACE|GLOB_ONLYDIR|GLOB_NOSORT) as $dir) {
                    if (in_array(basename($dir), array('.','..')))
                        continue;
                    elseif ($this->exclude($exclude, $dir))
    
    Jared Hancock's avatar
    Jared Hancock committed
                        continue;
                    $this->unpackage(
                        dirname($folder).'/'.basename($dir).'/'.basename($folder),
    
                        $destination.basename($dir),
    
    Jared Hancock's avatar
    Jared Hancock committed
                        $recurse - 1, $exclude);
                }
            }
        }
    
        function get_include_dir() {
            $main_inc_php = $this->destination . '/main.inc.php';
            $lines = preg_grep("/define\s*\(\s*'INCLUDE_DIR'/",
                explode("\n", file_get_contents($main_inc_php)));
    
            // NOTE: that this won't work for crafty folks who have a define or some
            //       variable in the value of their include path
            if (!defined('ROOT_DIR')) define('ROOT_DIR', $this->destination . '/');
            foreach ($lines as $line)
                eval($line);
    
            return INCLUDE_DIR;
        }
    
    
        function run($args, $options) {
    
    Jared Hancock's avatar
    Jared Hancock committed
            $this->destination = $this->getArgument('install-path');
            if (!is_dir($this->destination))
                if (!mkdir($this->destination, 0751, true))
    
                    die("Destination path does not exist and cannot be created");
    
    Jared Hancock's avatar
    Jared Hancock committed
    
            # Determine if this is an upgrade, and if so, where the include/
            # folder is currently located
            $upgrade = file_exists("{$this->destination}/main.inc.php");
    
            # Locate the upload folder
            $upload = $this->find_upload_folder();
    
            # Unpack the upload folder to the destination, except the include folder
            if ($upgrade)
                # Get the current value of the INCLUDE_DIR before overwriting
                # main.inc.php
                $include = $this->get_include_dir();
    
    Jared Hancock's avatar
    Jared Hancock committed
            $this->unpackage("$upload/{,.}*", $this->destination, -1, "*include");
    
    Jared Hancock's avatar
    Jared Hancock committed
    
            if (!$upgrade) {
                if ($this->getOption('include')) {
                    $location = $this->getOption('include');
                    if (!is_dir("$location/"))
                        if (!mkdir("$location/", 0751, true))
                            die("Unable to create folder for include/ files\n");
    
    Jared Hancock's avatar
    Jared Hancock committed
                    $this->unpackage("$upload/include/{,.}*", $location, -1);
    
    Jared Hancock's avatar
    Jared Hancock committed
                    $this->change_include_dir($location);
                }
                else
    
    Jared Hancock's avatar
    Jared Hancock committed
                    $this->unpackage("$upload/include/{,.}*", "{$this->destination}/include", -1);
    
    Jared Hancock's avatar
    Jared Hancock committed
            }
            else {
    
    Jared Hancock's avatar
    Jared Hancock committed
                $this->unpackage("$upload/include/{,.}*", $include, -1);
    
    Jared Hancock's avatar
    Jared Hancock committed
                # Change the new main.inc.php to reflect the location of the
                # include/ directory
                $this->change_include_dir($include);
            }
        }
    }
    
    Module::register('unpack', 'Unpacker');