AssetConverter.php 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\web;
  8. use Yii;
  9. use yii\base\Component;
  10. use yii\base\Exception;
  11. /**
  12. * AssetConverter supports conversion of several popular formats into JS or CSS files.
  13. *
  14. * It is used by [[AssetManager]] to convert files after they have been published.
  15. *
  16. * @author Qiang Xue <qiang.xue@gmail.com>
  17. * @since 2.0
  18. */
  19. class AssetConverter extends Component implements AssetConverterInterface
  20. {
  21. /**
  22. * @var array the commands that are used to perform the asset conversion.
  23. * The keys are the asset file extension names, and the values are the corresponding
  24. * target script types (either "css" or "js") and the commands used for the conversion.
  25. *
  26. * The command placeholders: `{from}` - source file, `{to}` - converted file.
  27. *
  28. * You may also use a [path alias](guide:concept-aliases) to specify the location of the command:
  29. *
  30. * ```php
  31. * [
  32. * 'styl' => ['css', '@app/node_modules/bin/stylus < {from} > {to}'],
  33. * ]
  34. * ```
  35. *
  36. * Note: `Yii::getAlias()` can replace alias at the begin of the command only.
  37. *
  38. * @see https://sass-lang.com/documentation/cli/ SASS/SCSS
  39. */
  40. public $commands = [
  41. 'less' => ['css', 'lessc {from} {to} --no-color --source-map'],
  42. 'scss' => ['css', 'sass --style=compressed {from} {to}'],
  43. 'sass' => ['css', 'sass --style=compressed {from} {to}'],
  44. 'styl' => ['css', 'stylus < {from} > {to}'],
  45. 'coffee' => ['js', 'coffee -p {from} > {to}'],
  46. 'ts' => ['js', 'tsc --out {to} {from}'],
  47. ];
  48. /**
  49. * @var bool whether the source asset file should be converted even if its result already exists.
  50. * You may want to set this to be `true` during the development stage to make sure the converted
  51. * assets are always up-to-date. Do not set this to true on production servers as it will
  52. * significantly degrade the performance.
  53. */
  54. public $forceConvert = false;
  55. /**
  56. * Converts a given asset file into a CSS or JS file.
  57. * @param string $asset the asset file path, relative to $basePath
  58. * @param string $basePath the directory the $asset is relative to.
  59. * @return string the converted asset file path, relative to $basePath.
  60. */
  61. public function convert($asset, $basePath)
  62. {
  63. $pos = strrpos($asset, '.');
  64. if ($pos !== false) {
  65. $ext = substr($asset, $pos + 1);
  66. if (isset($this->commands[$ext])) {
  67. list($ext, $command) = $this->commands[$ext];
  68. $result = substr($asset, 0, $pos + 1) . $ext;
  69. if ($this->forceConvert || @filemtime("$basePath/$result") < @filemtime("$basePath/$asset")) {
  70. $this->runCommand($command, $basePath, $asset, $result);
  71. }
  72. return $result;
  73. }
  74. }
  75. return $asset;
  76. }
  77. /**
  78. * Runs a command to convert asset files.
  79. * @param string $command the command to run. If prefixed with an `@` it will be treated as a [path alias](guide:concept-aliases).
  80. * @param string $basePath asset base path and command working directory
  81. * @param string $asset the name of the asset file
  82. * @param string $result the name of the file to be generated by the converter command
  83. * @return bool true on success, false on failure. Failures will be logged.
  84. * @throws \yii\base\Exception when the command fails and YII_DEBUG is true.
  85. * In production mode the error will be logged.
  86. */
  87. protected function runCommand($command, $basePath, $asset, $result)
  88. {
  89. $command = Yii::getAlias($command);
  90. $command = strtr($command, [
  91. '{from}' => escapeshellarg("$basePath/$asset"),
  92. '{to}' => escapeshellarg("$basePath/$result"),
  93. ]);
  94. $descriptor = [
  95. 1 => ['pipe', 'w'],
  96. 2 => ['pipe', 'w'],
  97. ];
  98. $pipes = [];
  99. $proc = proc_open($command, $descriptor, $pipes, $basePath);
  100. $stdout = stream_get_contents($pipes[1]);
  101. $stderr = stream_get_contents($pipes[2]);
  102. foreach ($pipes as $pipe) {
  103. fclose($pipe);
  104. }
  105. $status = proc_close($proc);
  106. if ($status === 0) {
  107. Yii::debug("Converted $asset into $result:\nSTDOUT:\n$stdout\nSTDERR:\n$stderr", __METHOD__);
  108. } elseif (YII_DEBUG) {
  109. throw new Exception("AssetConverter command '$command' failed with exit code $status:\nSTDOUT:\n$stdout\nSTDERR:\n$stderr");
  110. } else {
  111. Yii::error("AssetConverter command '$command' failed with exit code $status:\nSTDOUT:\n$stdout\nSTDERR:\n$stderr", __METHOD__);
  112. }
  113. return $status === 0;
  114. }
  115. }