Installer.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. <?php
  2. /**
  3. * @link http://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license http://www.yiiframework.com/license/
  6. */
  7. namespace yii\composer;
  8. use Composer\Package\PackageInterface;
  9. use Composer\Installer\LibraryInstaller;
  10. use Composer\Repository\InstalledRepositoryInterface;
  11. use Composer\Script\CommandEvent;
  12. use Composer\Script\Event;
  13. use Composer\Util\Filesystem;
  14. use React\Promise\PromiseInterface;
  15. /**
  16. * @author Qiang Xue <qiang.xue@gmail.com>
  17. * @since 2.0
  18. */
  19. class Installer extends LibraryInstaller
  20. {
  21. const EXTRA_BOOTSTRAP = 'bootstrap';
  22. const EXTENSION_FILE = 'yiisoft/extensions.php';
  23. /**
  24. * @inheritdoc
  25. */
  26. public function supports($packageType)
  27. {
  28. return $packageType === 'yii2-extension';
  29. }
  30. /**
  31. * @inheritdoc
  32. */
  33. public function install(InstalledRepositoryInterface $repo, PackageInterface $package)
  34. {
  35. $afterInstall = function () use ($package) {
  36. // add the package to yiisoft/extensions.php
  37. $this->addPackage($package);
  38. // ensure the yii2-dev package also provides Yii.php in the same place as yii2 does
  39. if ($package->getName() == 'yiisoft/yii2-dev') {
  40. $this->linkBaseYiiFiles();
  41. }
  42. };
  43. // install the package the normal composer way
  44. $promise = parent::install($repo, $package);
  45. // Composer v2 might return a promise here
  46. if ($promise instanceof PromiseInterface) {
  47. return $promise->then($afterInstall);
  48. }
  49. // If not, execute the code right away as parent::install executed synchronously (composer v1, or v2 without async)
  50. $afterInstall();
  51. }
  52. /**
  53. * @inheritdoc
  54. */
  55. public function update(InstalledRepositoryInterface $repo, PackageInterface $initial, PackageInterface $target)
  56. {
  57. $afterUpdate = function () use ($initial, $target) {
  58. $this->removePackage($initial);
  59. $this->addPackage($target);
  60. // ensure the yii2-dev package also provides Yii.php in the same place as yii2 does
  61. if ($initial->getName() == 'yiisoft/yii2-dev') {
  62. $this->linkBaseYiiFiles();
  63. }
  64. };
  65. // update the package the normal composer way
  66. $promise = parent::update($repo, $initial, $target);
  67. // Composer v2 might return a promise here
  68. if ($promise instanceof PromiseInterface) {
  69. return $promise->then($afterUpdate);
  70. }
  71. // If not, execute the code right away as parent::update executed synchronously (composer v1, or v2 without async)
  72. $afterUpdate();
  73. }
  74. /**
  75. * @inheritdoc
  76. */
  77. public function uninstall(InstalledRepositoryInterface $repo, PackageInterface $package)
  78. {
  79. $afterUninstall = function () use ($package) {
  80. // remove the package from yiisoft/extensions.php
  81. $this->removePackage($package);
  82. // remove links for Yii.php
  83. if ($package->getName() == 'yiisoft/yii2-dev') {
  84. $this->removeBaseYiiFiles();
  85. }
  86. };
  87. // uninstall the package the normal composer way
  88. $promise = parent::uninstall($repo, $package);
  89. // Composer v2 might return a promise here
  90. if ($promise instanceof PromiseInterface) {
  91. return $promise->then($afterUninstall);
  92. }
  93. // If not, execute the code right away as parent::uninstall executed synchronously (composer v1, or v2 without async)
  94. $afterUninstall();
  95. }
  96. protected function addPackage(PackageInterface $package)
  97. {
  98. $extension = [
  99. 'name' => $package->getName(),
  100. 'version' => $package->getVersion(),
  101. ];
  102. $alias = $this->generateDefaultAlias($package);
  103. if (!empty($alias)) {
  104. $extension['alias'] = $alias;
  105. }
  106. $extra = $package->getExtra();
  107. if (isset($extra[self::EXTRA_BOOTSTRAP])) {
  108. $extension['bootstrap'] = $extra[self::EXTRA_BOOTSTRAP];
  109. }
  110. $extensions = $this->loadExtensions();
  111. $extensions[$package->getName()] = $extension;
  112. $this->saveExtensions($extensions);
  113. }
  114. protected function generateDefaultAlias(PackageInterface $package)
  115. {
  116. $fs = new Filesystem;
  117. $vendorDir = $fs->normalizePath($this->vendorDir);
  118. $autoload = $package->getAutoload();
  119. $aliases = [];
  120. if (!empty($autoload['psr-0'])) {
  121. foreach ($autoload['psr-0'] as $name => $path) {
  122. $name = str_replace('\\', '/', trim($name, '\\'));
  123. if (!$fs->isAbsolutePath($path)) {
  124. $path = $this->vendorDir . '/' . $package->getPrettyName() . '/' . $path;
  125. }
  126. $path = $fs->normalizePath($path);
  127. if (strpos($path . '/', $vendorDir . '/') === 0) {
  128. $aliases["@$name"] = '<vendor-dir>' . substr($path, strlen($vendorDir)) . '/' . $name;
  129. } else {
  130. $aliases["@$name"] = $path . '/' . $name;
  131. }
  132. }
  133. }
  134. if (!empty($autoload['psr-4'])) {
  135. foreach ($autoload['psr-4'] as $name => $path) {
  136. if (is_array($path)) {
  137. // ignore psr-4 autoload specifications with multiple search paths
  138. // we can not convert them into aliases as they are ambiguous
  139. continue;
  140. }
  141. $name = str_replace('\\', '/', trim($name, '\\'));
  142. if (!$fs->isAbsolutePath($path)) {
  143. $path = $this->vendorDir . '/' . $package->getPrettyName() . '/' . $path;
  144. }
  145. $path = $fs->normalizePath($path);
  146. if (strpos($path . '/', $vendorDir . '/') === 0) {
  147. $aliases["@$name"] = '<vendor-dir>' . substr($path, strlen($vendorDir));
  148. } else {
  149. $aliases["@$name"] = $path;
  150. }
  151. }
  152. }
  153. return $aliases;
  154. }
  155. protected function removePackage(PackageInterface $package)
  156. {
  157. $packages = $this->loadExtensions();
  158. unset($packages[$package->getName()]);
  159. $this->saveExtensions($packages);
  160. }
  161. protected function loadExtensions()
  162. {
  163. $file = $this->vendorDir . '/' . static::EXTENSION_FILE;
  164. if (!is_file($file)) {
  165. return [];
  166. }
  167. // invalidate opcache of extensions.php if exists
  168. if (function_exists('opcache_invalidate')) {
  169. @opcache_invalidate($file, true);
  170. }
  171. $extensions = require($file);
  172. $vendorDir = str_replace('\\', '/', $this->vendorDir);
  173. $n = strlen($vendorDir);
  174. foreach ($extensions as &$extension) {
  175. if (isset($extension['alias'])) {
  176. foreach ($extension['alias'] as $alias => $path) {
  177. $path = str_replace('\\', '/', $path);
  178. if (strpos($path . '/', $vendorDir . '/') === 0) {
  179. $extension['alias'][$alias] = '<vendor-dir>' . substr($path, $n);
  180. }
  181. }
  182. }
  183. }
  184. return $extensions;
  185. }
  186. protected function saveExtensions(array $extensions)
  187. {
  188. $file = $this->vendorDir . '/' . static::EXTENSION_FILE;
  189. if (!file_exists(dirname($file))) {
  190. mkdir(dirname($file), 0777, true);
  191. }
  192. $array = str_replace("'<vendor-dir>", '$vendorDir . \'', var_export($extensions, true));
  193. file_put_contents($file, "<?php\n\n\$vendorDir = dirname(__DIR__);\n\nreturn $array;\n");
  194. // invalidate opcache of extensions.php if exists
  195. if (function_exists('opcache_invalidate')) {
  196. @opcache_invalidate($file, true);
  197. }
  198. }
  199. protected function linkBaseYiiFiles()
  200. {
  201. $yiiDir = $this->vendorDir . '/yiisoft/yii2';
  202. if (!file_exists($yiiDir)) {
  203. mkdir($yiiDir, 0777, true);
  204. }
  205. foreach (['Yii.php', 'BaseYii.php', 'classes.php'] as $file) {
  206. file_put_contents($yiiDir . '/' . $file, <<<EOF
  207. <?php
  208. /**
  209. * This is a link provided by the yiisoft/yii2-dev package via yii2-composer plugin.
  210. *
  211. * @link http://www.yiiframework.com/
  212. * @copyright Copyright (c) 2008 Yii Software LLC
  213. * @license http://www.yiiframework.com/license/
  214. */
  215. return require(__DIR__ . '/../yii2-dev/framework/$file');
  216. EOF
  217. );
  218. }
  219. }
  220. protected function removeBaseYiiFiles()
  221. {
  222. $yiiDir = $this->vendorDir . '/yiisoft/yii2';
  223. foreach (['Yii.php', 'BaseYii.php', 'classes.php'] as $file) {
  224. if (file_exists($yiiDir . '/' . $file)) {
  225. unlink($yiiDir . '/' . $file);
  226. }
  227. }
  228. if (file_exists($yiiDir)) {
  229. rmdir($yiiDir);
  230. }
  231. }
  232. /**
  233. * Special method to run tasks defined in `[extra][yii\composer\Installer::postCreateProject]` key in `composer.json`
  234. *
  235. * @param Event $event
  236. */
  237. public static function postCreateProject($event)
  238. {
  239. static::runCommands($event, __METHOD__);
  240. }
  241. /**
  242. * Special method to run tasks defined in `[extra][yii\composer\Installer::postInstall]` key in `composer.json`
  243. *
  244. * @param Event $event
  245. * @since 2.0.5
  246. */
  247. public static function postInstall($event)
  248. {
  249. static::runCommands($event, __METHOD__);
  250. }
  251. /**
  252. * Special method to run tasks defined in `[extra][$extraKey]` key in `composer.json`
  253. *
  254. * @param Event $event
  255. * @param string $extraKey
  256. * @since 2.0.5
  257. */
  258. protected static function runCommands($event, $extraKey)
  259. {
  260. $params = $event->getComposer()->getPackage()->getExtra();
  261. if (isset($params[$extraKey]) && is_array($params[$extraKey])) {
  262. foreach ($params[$extraKey] as $method => $args) {
  263. call_user_func_array([__CLASS__, $method], (array) $args);
  264. }
  265. }
  266. }
  267. /**
  268. * Sets the correct permission for the files and directories listed in the extra section.
  269. * @param array $paths the paths (keys) and the corresponding permission octal strings (values)
  270. */
  271. public static function setPermission(array $paths)
  272. {
  273. foreach ($paths as $path => $permission) {
  274. echo "chmod('$path', $permission)...";
  275. if (is_dir($path) || is_file($path)) {
  276. try {
  277. if (chmod($path, octdec($permission))) {
  278. echo "done.\n";
  279. };
  280. } catch (\Exception $e) {
  281. echo $e->getMessage() . "\n";
  282. }
  283. } else {
  284. echo "file not found.\n";
  285. }
  286. }
  287. }
  288. /**
  289. * Generates a cookie validation key for every app config listed in "config" in extra section.
  290. * You can provide one or multiple parameters as the configuration files which need to have validation key inserted.
  291. */
  292. public static function generateCookieValidationKey()
  293. {
  294. $configs = func_get_args();
  295. $key = self::generateRandomString();
  296. foreach ($configs as $config) {
  297. if (is_file($config)) {
  298. $content = preg_replace('/(("|\')cookieValidationKey("|\')\s*=>\s*)(""|\'\')/', "\\1'$key'", file_get_contents($config), -1, $count);
  299. if ($count > 0) {
  300. file_put_contents($config, $content);
  301. }
  302. }
  303. }
  304. }
  305. protected static function generateRandomString()
  306. {
  307. if (!extension_loaded('openssl')) {
  308. throw new \Exception('The OpenSSL PHP extension is required by Yii2.');
  309. }
  310. $length = 32;
  311. $bytes = openssl_random_pseudo_bytes($length);
  312. return strtr(substr(base64_encode($bytes), 0, $length), '+/=', '_-.');
  313. }
  314. /**
  315. * Copy files to specified locations.
  316. * @param array $paths The source files paths (keys) and the corresponding target locations
  317. * for copied files (values). Location can be specified as an array - first element is target
  318. * location, second defines whether file can be overwritten (by default method don't overwrite
  319. * existing files).
  320. * @since 2.0.5
  321. */
  322. public static function copyFiles(array $paths)
  323. {
  324. foreach ($paths as $source => $target) {
  325. // handle file target as array [path, overwrite]
  326. $target = (array) $target;
  327. echo "Copying file $source to $target[0] - ";
  328. if (!is_file($source)) {
  329. echo "source file not found.\n";
  330. continue;
  331. }
  332. if (is_file($target[0]) && empty($target[1])) {
  333. echo "target file exists - skip.\n";
  334. continue;
  335. } elseif (is_file($target[0]) && !empty($target[1])) {
  336. echo "target file exists - overwrite - ";
  337. }
  338. try {
  339. if (!is_dir(dirname($target[0]))) {
  340. mkdir(dirname($target[0]), 0777, true);
  341. }
  342. if (copy($source, $target[0])) {
  343. echo "done.\n";
  344. }
  345. } catch (\Exception $e) {
  346. echo $e->getMessage() . "\n";
  347. }
  348. }
  349. }
  350. }