AssetControllerTest.php 9.38 KB
Newer Older
1 2 3 4 5 6
<?php

use yiiunit\TestCase;
use yii\console\controllers\AssetController;

/**
7
 * Unit test for [[\yii\console\controllers\AssetController]].
8 9 10 11 12 13 14 15
 * @see AssetController
 */
class AssetControllerTest extends TestCase
{
	/**
	 * @var string path for the test files.
	 */
	protected $testFilePath = '';
16 17 18 19
	/**
	 * @var string test assets path.
	 */
	protected $testAssetsBasePath = '';
20 21 22 23 24

	public function setUp()
	{
		$this->testFilePath = Yii::getAlias('@yiiunit/runtime') . DIRECTORY_SEPARATOR . get_class($this);
		$this->createDir($this->testFilePath);
25 26
		$this->testAssetsBasePath = $this->testFilePath . DIRECTORY_SEPARATOR . 'assets';
		$this->createDir($this->testAssetsBasePath);
27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
	}

	public function tearDown()
	{
		$this->removeDir($this->testFilePath);
	}

	/**
	 * Creates directory.
	 * @param $dirName directory full name.
	 */
	protected function createDir($dirName)
	{
		if (!file_exists($dirName)) {
			mkdir($dirName, 0777, true);
		}
	}

	/**
	 * Removes directory.
	 * @param $dirName directory full name
	 */
	protected function removeDir($dirName)
	{
		if (!empty($dirName) && file_exists($dirName)) {
			exec("rm -rf {$dirName}");
		}
	}

	/**
	 * Creates test asset controller instance.
	 * @return AssetController
	 */
	protected function createAssetController()
	{
		$module = $this->getMock('yii\\base\\Module', array('fake'), array('console'));
		$assetController = new AssetController('asset', $module);
64
		$assetController->interactive = false;
65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87
		$assetController->jsCompressor = 'cp {from} {to}';
		$assetController->cssCompressor = 'cp {from} {to}';
		return $assetController;
	}

	/**
	 * Emulates running of the asset controller action.
	 * @param string $actionId id of action to be run.
	 * @param array $args action arguments.
	 * @return string command output.
	 */
	protected function runAssetControllerAction($actionId, array $args=array())
	{
		$controller = $this->createAssetController();
		ob_start();
		ob_implicit_flush(false);
		$params = array(
			\yii\console\Request::ANONYMOUS_PARAMS => $args
		);
		$controller->run($actionId, $params);
		return ob_get_clean();
	}

88 89
	/**
	 * Creates test compress config.
90
	 * @param array[] $bundles asset bundles config.
91 92
	 * @return array config array.
	 */
93
	protected function createCompressConfig(array $bundles)
94
	{
95
		$baseUrl = '/test';
96
		$config = array(
97
			'bundles' => $this->createBundleConfig($bundles),
98 99
			'targets' => array(
				'all' => array(
100
					'basePath' => $this->testAssetsBasePath,
101
					'baseUrl' => $baseUrl,
102 103
					'js' => 'all.js',
					'css' => 'all.css',
104 105 106
				),
			),
			'assetManager' => array(
107
				'basePath' => $this->testAssetsBasePath,
108
				'baseUrl' => '',
109 110 111 112 113 114 115
			),
		);
		return $config;
	}

	/**
	 * Creates test bundle configuration.
116
	 * @param array[] $bundles asset bundles config.
117 118
	 * @return array bundle config.
	 */
119
	protected function createBundleConfig(array $bundles)
120
	{
121 122 123 124 125 126 127 128
		foreach ($bundles as $name => $config) {
			if (!array_key_exists('basePath', $config)) {
				$bundles[$name]['basePath'] = $this->testFilePath;
			}
			if (!array_key_exists('baseUrl', $config)) {
				$bundles[$name]['baseUrl'] = '';
			}
		}
129 130 131 132 133 134
		return $bundles;
	}

	/**
	 * Creates test compress config file.
	 * @param string $fileName output file name.
135 136
	 * @param array[] $bundles asset bundles config.
	 * @throws Exception on failure.
137
	 */
138
	protected function createCompressConfigFile($fileName, array $bundles)
139
	{
140 141 142 143
		$content = '<?php return '.var_export($this->createCompressConfig($bundles), true).';';
		if (file_put_contents($fileName, $content) <= 0) {
			throw new \Exception("Unable to create file '{$fileName}'!");
		}
144 145
	}

146 147 148 149
	/**
	 * Creates test asset file.
	 * @param string $fileRelativeName file name relative to [[testFilePath]]
	 * @param string $content file content
150
	 * @throws Exception on failure.
151
	 */
152
	protected function createAssetSourceFile($fileRelativeName, $content)
153 154 155
	{
		$fileFullName = $this->testFilePath.DIRECTORY_SEPARATOR.$fileRelativeName;
		$this->createDir(dirname($fileFullName));
156 157 158 159 160 161 162 163 164 165 166 167 168 169
		if (file_put_contents($fileFullName, $content)<=0) {
			throw new \Exception("Unable to create file '{$fileFullName}'!");
		}
	}

	/**
	 * Creates a list of asset source files.
	 * @param array $files assert source files in format: file/relative/name => fileContent
	 */
	protected function createAssertSourceFiles(array $files)
	{
		foreach ($files as $name => $content) {
			$this->createAssetSourceFile($name, $content);
		}
170 171
	}

172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188
	/**
	 * Invokes the asset controller method even if it is protected.
	 * @param string $methodName name of the method to be invoked.
	 * @param array $args method arguments.
	 * @return mixed method invoke result.
	 */
	protected function invokeAssetControllerMethod($methodName, array $args = array())
	{
		$controller = $this->createAssetController();
		$controllerClassReflection = new ReflectionClass(get_class($controller));
		$methodReflection = $controllerClassReflection->getMethod($methodName);
		$methodReflection->setAccessible(true);
		$result = $methodReflection->invokeArgs($controller, $args);
		$methodReflection->setAccessible(false);
		return $result;
	}

189 190 191 192 193 194 195 196
	// Tests :

	public function testActionTemplate()
	{
		$configFileName = $this->testFilePath . DIRECTORY_SEPARATOR . 'config.php';
		$this->runAssetControllerAction('template', array($configFileName));
		$this->assertTrue(file_exists($configFileName), 'Unable to create config file template!');
	}
197 198 199

	public function testActionCompress()
	{
200 201 202
		// Given :
		$cssFiles = array(
			'css/test_body.css' => 'body {
203 204
				padding-top: 20px;
				padding-bottom: 60px;
205 206 207 208 209
			}',
			'css/test_footer.css' => '.footer {
				margin: 20px;
				display: block;
			}',
210
		);
211 212 213 214
		$this->createAssertSourceFiles($cssFiles);

		$jsFiles = array(
			'js/test_alert.js' => "function test() {
215
				alert('Test message');
216 217 218 219
			}",
			'js/test_sum_ab.js' => "function sumAB(a, b) {
				return a + b;
			}",
220
		);
221
		$this->createAssertSourceFiles($jsFiles);
222

223 224 225 226
		$bundles = array(
			'app' => array(
				'css' => array_keys($cssFiles),
				'js' => array_keys($jsFiles),
227 228 229
				'depends' => array(
					'yii',
				),
230 231
			),
		);;
232
		$bundleFile = $this->testFilePath . DIRECTORY_SEPARATOR . 'bundle.php';
233 234 235

		$configFile = $this->testFilePath . DIRECTORY_SEPARATOR . 'config.php';
		$this->createCompressConfigFile($configFile, $bundles);
236

237
		// When :
238 239
		$this->runAssetControllerAction('compress', array($configFile, $bundleFile));

240
		// Then :
241 242
		$this->assertTrue(file_exists($bundleFile), 'Unable to create output bundle file!');

243
		$compressedCssFileName = $this->testAssetsBasePath . DIRECTORY_SEPARATOR . 'all.css';
244
		$this->assertTrue(file_exists($compressedCssFileName), 'Unable to compress CSS files!');
245
		$compressedJsFileName = $this->testAssetsBasePath . DIRECTORY_SEPARATOR . 'all.js';
246
		$this->assertTrue(file_exists($compressedJsFileName), 'Unable to compress JS files!');
247 248 249 250 251 252 253 254 255

		$compressedCssFileContent = file_get_contents($compressedCssFileName);
		foreach ($cssFiles as $name => $content) {
			$this->assertContains($content, $compressedCssFileContent, "Source of '{$name}' is missing in combined file!");
		}
		$compressedJsFileContent = file_get_contents($compressedJsFileName);
		foreach ($jsFiles as $name => $content) {
			$this->assertContains($content, $compressedJsFileContent, "Source of '{$name}' is missing in combined file!");
		}
256
	}
257 258 259 260 261 262 263 264 265

	/**
	 * Data provider for [[testAdjustCssUrl()]].
	 * @return array test data.
	 */
	public function adjustCssUrlDataProvider()
	{
		return array(
			array(
266
				'.published-same-dir-class {background-image: url(published_same_dir.png);}',
267 268
				'/test/base/path/assets/input',
				'/test/base/path/assets/output',
269
				'.published-same-dir-class {background-image: url(../input/published_same_dir.png);}',
270 271
			),
			array(
272
				'.published-relative-dir-class {background-image: url(../img/published_relative_dir.png);}',
273 274
				'/test/base/path/assets/input',
				'/test/base/path/assets/output',
275
				'.published-relative-dir-class {background-image: url(../img/published_relative_dir.png);}',
276 277
			),
			array(
278
				'.static-same-dir-class {background-image: url(\'static_same_dir.png\');}',
279 280
				'/test/base/path/css',
				'/test/base/path/assets/output',
281
				'.static-same-dir-class {background-image: url(\'../../css/static_same_dir.png\');}',
282 283 284 285 286 287
			),
			array(
				'.static-relative-dir-class {background-image: url("../img/static_relative_dir.png");}',
				'/test/base/path/css',
				'/test/base/path/assets/output',
				'.static-relative-dir-class {background-image: url("../../img/static_relative_dir.png");}',
288
			),
289 290 291 292 293 294 295 296 297 298 299 300
			array(
				'.absolute-url-class {background-image: url(http://domain.com/img/image.gif);}',
				'/test/base/path/assets/input',
				'/test/base/path/assets/output',
				'.absolute-url-class {background-image: url(http://domain.com/img/image.gif);}',
			),
			array(
				'.absolute-url-secure-class {background-image: url(https://secure.domain.com/img/image.gif);}',
				'/test/base/path/assets/input',
				'/test/base/path/assets/output',
				'.absolute-url-secure-class {background-image: url(https://secure.domain.com/img/image.gif);}',
			),
301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317
		);
	}

	/**
	 * @dataProvider adjustCssUrlDataProvider
	 *
	 * @param $cssContent
	 * @param $inputFilePath
	 * @param $outputFilePath
	 * @param $expectedCssContent
	 */
	public function testAdjustCssUrl($cssContent, $inputFilePath, $outputFilePath, $expectedCssContent)
	{
		$adjustedCssContent = $this->invokeAssetControllerMethod('adjustCssUrl', array($cssContent, $inputFilePath, $outputFilePath));

		$this->assertEquals($expectedCssContent, $adjustedCssContent, 'Unable to adjust CSS correctly!');
	}
318
}