mirror of
				https://github.com/getgrav/grav-plugin-admin.git
				synced 2025-10-31 18:35:57 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			431 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			431 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace Grav\Plugin\Admin;
 | |
| 
 | |
| use Grav\Common\Cache;
 | |
| use Grav\Common\Grav;
 | |
| use Grav\Common\GPM\GPM as GravGPM;
 | |
| use Grav\Common\GPM\Licenses;
 | |
| use Grav\Common\GPM\Installer;
 | |
| use Grav\Common\GPM\Response;
 | |
| use Grav\Common\GPM\Upgrader;
 | |
| use Grav\Common\Filesystem\Folder;
 | |
| use Grav\Common\GPM\Common\Package;
 | |
| 
 | |
| /**
 | |
|  * Class Gpm
 | |
|  *
 | |
|  * @package Grav\Plugin\Admin
 | |
|  */
 | |
| class Gpm
 | |
| {
 | |
|     // Probably should move this to Grav DI container?
 | |
|     /** @var GravGPM */
 | |
|     protected static $GPM;
 | |
| 
 | |
|     public static function GPM()
 | |
|     {
 | |
|         if (!static::$GPM) {
 | |
|             static::$GPM = new GravGPM();
 | |
|         }
 | |
| 
 | |
|         return static::$GPM;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Default options for the install
 | |
|      *
 | |
|      * @var array
 | |
|      */
 | |
|     protected static $options = [
 | |
|         'destination'     => GRAV_ROOT,
 | |
|         'overwrite'       => true,
 | |
|         'ignore_symlinks' => true,
 | |
|         'skip_invalid'    => true,
 | |
|         'install_deps'    => true,
 | |
|         'theme'           => false
 | |
|     ];
 | |
| 
 | |
|     /**
 | |
|      * @param Package[]|string[]|string $packages
 | |
|      * @param array                     $options
 | |
|      *
 | |
|      * @return string|bool
 | |
|      */
 | |
|     public static function install($packages, array $options)
 | |
|     {
 | |
|         $options = array_merge(self::$options, $options);
 | |
| 
 | |
|         if (!Installer::isGravInstance($options['destination']) || !Installer::isValidDestination($options['destination'],
 | |
|                 [Installer::EXISTS, Installer::IS_LINK])
 | |
|         ) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         $packages = is_array($packages) ? $packages : [$packages];
 | |
|         $count    = count($packages);
 | |
| 
 | |
|         $packages = array_filter(array_map(function ($p) {
 | |
|             return !is_string($p) ? $p instanceof Package ? $p : false : self::GPM()->findPackage($p);
 | |
|         }, $packages));
 | |
| 
 | |
|         if (!$options['skip_invalid'] && $count !== count($packages)) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         $messages = '';
 | |
| 
 | |
|         foreach ($packages as $package) {
 | |
|             if (isset($package->dependencies) && $options['install_deps']) {
 | |
|                 $result = static::install($package->dependencies, $options);
 | |
| 
 | |
|                 if (!$result) {
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Check destination
 | |
|             Installer::isValidDestination($options['destination'] . DS . $package->install_path);
 | |
| 
 | |
|             if (!$options['overwrite'] && Installer::lastErrorCode() === Installer::EXISTS) {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (!$options['ignore_symlinks'] && Installer::lastErrorCode() === Installer::IS_LINK) {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             $license = Licenses::get($package->slug);
 | |
|             $local   = static::download($package, $license);
 | |
| 
 | |
|             Installer::install($local, $options['destination'],
 | |
|                 ['install_path' => $package->install_path, 'theme' => $options['theme']]);
 | |
|             Folder::delete(dirname($local));
 | |
| 
 | |
|             $errorCode = Installer::lastErrorCode();
 | |
|             if ($errorCode) {
 | |
|                 $msg = Installer::lastErrorMsg();
 | |
|                 throw new \RuntimeException($msg);
 | |
|             }
 | |
| 
 | |
|             if (count($packages) === 1) {
 | |
|                 $message = Installer::getMessage();
 | |
|                 if ($message) {
 | |
|                     return $message;
 | |
|                 }
 | |
| 
 | |
|                 $messages .= $message;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $messages ?: true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Package[]|string[]|string $packages
 | |
|      * @param array                     $options
 | |
|      *
 | |
|      * @return string|bool
 | |
|      */
 | |
|     public static function update($packages, array $options)
 | |
|     {
 | |
|         $options['overwrite'] = true;
 | |
| 
 | |
|         return static::install($packages, $options);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Package[]|string[]|string $packages
 | |
|      * @param array                     $options
 | |
|      *
 | |
|      * @return string|bool
 | |
|      */
 | |
|     public static function uninstall($packages, array $options)
 | |
|     {
 | |
|         $options = array_merge(self::$options, $options);
 | |
| 
 | |
|         $packages = (array)$packages;
 | |
|         $count    = count($packages);
 | |
| 
 | |
|         $packages = array_filter(array_map(function ($p) {
 | |
| 
 | |
|             if (is_string($p)) {
 | |
|                 $p      = strtolower($p);
 | |
|                 $plugin = static::GPM()->getInstalledPlugin($p);
 | |
|                 $p      = $plugin ?: static::GPM()->getInstalledTheme($p);
 | |
|             }
 | |
| 
 | |
|             return $p instanceof Package ? $p : false;
 | |
| 
 | |
|         }, $packages));
 | |
| 
 | |
|         if (!$options['skip_invalid'] && $count !== count($packages)) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         foreach ($packages as $package) {
 | |
| 
 | |
|             $location = Grav::instance()['locator']->findResource($package->package_type . '://' . $package->slug);
 | |
| 
 | |
|             // Check destination
 | |
|             Installer::isValidDestination($location);
 | |
| 
 | |
|             if (!$options['ignore_symlinks'] && Installer::lastErrorCode() === Installer::IS_LINK) {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             Installer::uninstall($location);
 | |
| 
 | |
|             $errorCode = Installer::lastErrorCode();
 | |
|             if ($errorCode && $errorCode !== Installer::IS_LINK && $errorCode !== Installer::EXISTS) {
 | |
|                 $msg = Installer::lastErrorMsg();
 | |
|                 throw new \RuntimeException($msg);
 | |
|             }
 | |
| 
 | |
|             if (count($packages) === 1) {
 | |
|                 $message = Installer::getMessage();
 | |
|                 if ($message) {
 | |
|                     return $message;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Direct install a file
 | |
|      *
 | |
|      * @param string $package_file
 | |
|      *
 | |
|      * @return string|bool
 | |
|      */
 | |
|     public static function directInstall($package_file)
 | |
|     {
 | |
|         if (!$package_file) {
 | |
|             return Admin::translate('PLUGIN_ADMIN.NO_PACKAGE_NAME');
 | |
|         }
 | |
| 
 | |
|         $tmp_dir = Grav::instance()['locator']->findResource('tmp://', true, true);
 | |
|         $tmp_zip = $tmp_dir . '/Grav-' . uniqid('', false);
 | |
| 
 | |
|         if (Response::isRemote($package_file)) {
 | |
|             $zip = GravGPM::downloadPackage($package_file, $tmp_zip);
 | |
|         } else {
 | |
|             $zip = GravGPM::copyPackage($package_file, $tmp_zip);
 | |
|         }
 | |
| 
 | |
|         if (file_exists($zip)) {
 | |
|             $tmp_source = $tmp_dir . '/Grav-' . uniqid('', false);
 | |
|             $extracted  = Installer::unZip($zip, $tmp_source);
 | |
| 
 | |
|             if (!$extracted) {
 | |
|                 Folder::delete($tmp_source);
 | |
|                 Folder::delete($tmp_zip);
 | |
|                 return Admin::translate('PLUGIN_ADMIN.PACKAGE_EXTRACTION_FAILED');
 | |
|             }
 | |
| 
 | |
|             $type = GravGPM::getPackageType($extracted);
 | |
| 
 | |
|             if (!$type) {
 | |
|                 Folder::delete($tmp_source);
 | |
|                 Folder::delete($tmp_zip);
 | |
|                 return Admin::translate('PLUGIN_ADMIN.NOT_VALID_GRAV_PACKAGE');
 | |
|             }
 | |
| 
 | |
|             if ($type === 'grav') {
 | |
|                 Installer::isValidDestination(GRAV_ROOT . '/system');
 | |
|                 if (Installer::IS_LINK === Installer::lastErrorCode()) {
 | |
|                     Folder::delete($tmp_source);
 | |
|                     Folder::delete($tmp_zip);
 | |
|                     return Admin::translate('PLUGIN_ADMIN.CANNOT_OVERWRITE_SYMLINKS');
 | |
|                 }
 | |
| 
 | |
|                 static::upgradeGrav($zip, $extracted);
 | |
|             } else {
 | |
|                 $name = GravGPM::getPackageName($extracted);
 | |
| 
 | |
|                 if (!$name) {
 | |
|                     Folder::delete($tmp_source);
 | |
|                     Folder::delete($tmp_zip);
 | |
|                     return Admin::translate('PLUGIN_ADMIN.NAME_COULD_NOT_BE_DETERMINED');
 | |
|                 }
 | |
| 
 | |
|                 $install_path = GravGPM::getInstallPath($type, $name);
 | |
|                 $is_update    = file_exists($install_path);
 | |
| 
 | |
|                 Installer::isValidDestination(GRAV_ROOT . DS . $install_path);
 | |
|                 if (Installer::lastErrorCode() === Installer::IS_LINK) {
 | |
|                     Folder::delete($tmp_source);
 | |
|                     Folder::delete($tmp_zip);
 | |
|                     return Admin::translate('PLUGIN_ADMIN.CANNOT_OVERWRITE_SYMLINKS');
 | |
|                 }
 | |
| 
 | |
|                 Installer::install($zip, GRAV_ROOT,
 | |
|                     ['install_path' => $install_path, 'theme' => $type === 'theme', 'is_update' => $is_update],
 | |
|                     $extracted);
 | |
|             }
 | |
| 
 | |
|             Folder::delete($tmp_source);
 | |
| 
 | |
|             if (Installer::lastErrorCode()) {
 | |
|                 return Installer::lastErrorMsg();
 | |
|             }
 | |
| 
 | |
|         } else {
 | |
|             return Admin::translate('PLUGIN_ADMIN.ZIP_PACKAGE_NOT_FOUND');
 | |
|         }
 | |
| 
 | |
|         Folder::delete($tmp_zip);
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Package $package
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     private static function download(Package $package, $license = null)
 | |
|     {
 | |
|         $query = '';
 | |
| 
 | |
|         if ($package->premium) {
 | |
|             $query = \json_encode(array_merge($package->premium, [
 | |
|                 'slug'        => $package->slug,
 | |
|                 'filename'    => $package->premium['filename'],
 | |
|                 'license_key' => $license
 | |
|             ]));
 | |
| 
 | |
|             $query = '?d=' . base64_encode($query);
 | |
|         }
 | |
| 
 | |
|         try {
 | |
|             $contents = Response::get($package->zipball_url . $query, []);
 | |
|         } catch (\Exception $e) {
 | |
|             throw new \RuntimeException($e->getMessage());
 | |
|         }
 | |
| 
 | |
|         $tmp_dir = Admin::getTempDir() . '/Grav-' . uniqid('', false);
 | |
|         Folder::mkdir($tmp_dir);
 | |
| 
 | |
|         $bad_chars = array_merge(array_map('chr', range(0, 31)), ['<', '>', ':', '"', '/', '\\', '|', '?', '*']);
 | |
| 
 | |
|         $filename = $package->slug . str_replace($bad_chars, '', basename($package->zipball_url));
 | |
|         $filename = preg_replace('/[\\\\\/:"*?&<>|]+/m', '-', $filename);
 | |
| 
 | |
|         file_put_contents($tmp_dir . DS . $filename . '.zip', $contents);
 | |
| 
 | |
|         return $tmp_dir . DS . $filename . '.zip';
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param array  $package
 | |
|      * @param string $tmp
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     private static function _downloadSelfupgrade(array $package, $tmp)
 | |
|     {
 | |
|         $output = Response::get($package['download'], []);
 | |
|         Folder::mkdir($tmp);
 | |
|         file_put_contents($tmp . DS . $package['name'], $output);
 | |
| 
 | |
|         return $tmp . DS . $package['name'];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return bool
 | |
|      */
 | |
|     public static function selfupgrade()
 | |
|     {
 | |
|         $upgrader = new Upgrader();
 | |
| 
 | |
|         if (!Installer::isGravInstance(GRAV_ROOT)) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         if (is_link(GRAV_ROOT . DS . 'index.php')) {
 | |
|             Installer::setError(Installer::IS_LINK);
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         if (method_exists($upgrader, 'meetsRequirements') &&
 | |
|             method_exists($upgrader, 'minPHPVersion') &&
 | |
|             !$upgrader->meetsRequirements()) {
 | |
|             $error   = [];
 | |
|             $error[] = '<p>Grav has increased the minimum PHP requirement.<br />';
 | |
|             $error[] = 'You are currently running PHP <strong>' . phpversion() . '</strong>';
 | |
|             $error[] = ', but PHP <strong>' . $upgrader->minPHPVersion() . '</strong> is required.</p>';
 | |
|             $error[] = '<p><a href="http://getgrav.org/blog/changing-php-requirements-to-5.5" class="button button-small secondary">Additional information</a></p>';
 | |
| 
 | |
|             Installer::setError(implode("\n", $error));
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         $update = $upgrader->getAssets()['grav-update'];
 | |
|         $tmp    = Admin::getTempDir() . '/Grav-' . uniqid('', false);
 | |
|         if ($tmp) {
 | |
|             $file   = self::_downloadSelfupgrade($update, $tmp);
 | |
|             $folder = Installer::unZip($file, $tmp . '/zip');
 | |
|             $keepFolder = false;
 | |
|         } else {
 | |
|             // If you make $tmp empty, you can install your local copy of Grav (for testing purposes only).
 | |
|             $file = 'grav.zip';
 | |
|             $folder = '~/phpstorm/grav-clones/grav';
 | |
|             //$folder = '/home/matias/phpstorm/rockettheme/grav-devtools/grav-clones/grav';
 | |
|             $keepFolder = true;
 | |
|         }
 | |
| 
 | |
|         static::upgradeGrav($file, $folder, $keepFolder);
 | |
| 
 | |
|         $errorCode = Installer::lastErrorCode();
 | |
| 
 | |
|         if ($tmp) {
 | |
|             Folder::delete($tmp);
 | |
|         }
 | |
| 
 | |
|         return !(is_string($errorCode) || ($errorCode & (Installer::ZIP_OPEN_ERROR | Installer::ZIP_EXTRACT_ERROR)));
 | |
|     }
 | |
| 
 | |
|     private static function upgradeGrav($zip, $folder, $keepFolder = false)
 | |
|     {
 | |
|         static $ignores = [
 | |
|             'backup',
 | |
|             'cache',
 | |
|             'images',
 | |
|             'logs',
 | |
|             'tmp',
 | |
|             'user',
 | |
|             '.htaccess',
 | |
|             'robots.txt'
 | |
|         ];
 | |
| 
 | |
|         if (!is_dir($folder)) {
 | |
|             Installer::setError('Invalid source folder');
 | |
|         }
 | |
| 
 | |
|         try {
 | |
|             $script = $folder . '/system/install.php';
 | |
|             /** Install $installer */
 | |
|             if ((file_exists($script) && $install = include $script) && is_callable($install)) {
 | |
|                 $install($zip);
 | |
|             } else {
 | |
|                 Installer::install(
 | |
|                     $zip,
 | |
|                     GRAV_ROOT,
 | |
|                     ['sophisticated' => true, 'overwrite' => true, 'ignore_symlinks' => true, 'ignores' => $ignores],
 | |
|                     $folder,
 | |
|                     $keepFolder
 | |
|                 );
 | |
| 
 | |
|                 Cache::clearCache();
 | |
|             }
 | |
|         } catch (\Exception $e) {
 | |
|             Installer::setError($e->getMessage());
 | |
|         }
 | |
|     }
 | |
| }
 |