pictcode / lib / Cake / Console / Command / Task / ProjectTask.php @ 9d2f0219
履歴 | 表示 | アノテート | ダウンロード (14.805 KB)
1 |
<?php
|
---|---|
2 |
/**
|
3 |
* The Project Task handles creating the base application
|
4 |
*
|
5 |
* CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
|
6 |
* Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
|
7 |
*
|
8 |
* Licensed under The MIT License
|
9 |
* For full copyright and license information, please see the LICENSE.txt
|
10 |
* Redistributions of files must retain the above copyright notice.
|
11 |
*
|
12 |
* @copyright Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
|
13 |
* @link http://cakephp.org CakePHP(tm) Project
|
14 |
* @since CakePHP(tm) v 1.2
|
15 |
* @license http://www.opensource.org/licenses/mit-license.php MIT License
|
16 |
*/
|
17 |
|
18 |
App::uses('AppShell', 'Console/Command'); |
19 |
App::uses('File', 'Utility'); |
20 |
App::uses('Folder', 'Utility'); |
21 |
App::uses('CakeText', 'Utility'); |
22 |
App::uses('Security', 'Utility'); |
23 |
|
24 |
/**
|
25 |
* Task class for creating new project apps and plugins
|
26 |
*
|
27 |
* @package Cake.Console.Command.Task
|
28 |
*/
|
29 |
class ProjectTask extends AppShell { |
30 |
|
31 |
/**
|
32 |
* configs path (used in testing).
|
33 |
*
|
34 |
* @var string
|
35 |
*/
|
36 |
public $configPath = null; |
37 |
|
38 |
/**
|
39 |
* Checks that given project path does not already exist, and
|
40 |
* finds the app directory in it. Then it calls bake() with that information.
|
41 |
*
|
42 |
* @return mixed
|
43 |
*/
|
44 |
public function execute() { |
45 |
$project = null; |
46 |
if (isset($this->args[0])) { |
47 |
$project = $this->args[0]; |
48 |
} else {
|
49 |
$appContents = array_diff(scandir(APP), array('.', '..')); |
50 |
if (empty($appContents)) { |
51 |
$suggestedPath = rtrim(APP, DS); |
52 |
} else {
|
53 |
$suggestedPath = APP . 'myapp'; |
54 |
} |
55 |
} |
56 |
|
57 |
while (!$project) { |
58 |
$prompt = __d('cake_console', "What is the path to the project you want to bake?"); |
59 |
$project = $this->in($prompt, null, $suggestedPath); |
60 |
} |
61 |
|
62 |
if ($project && !Folder::isAbsolute($project) && isset($_SERVER['PWD'])) { |
63 |
$project = $_SERVER['PWD'] . DS . $project; |
64 |
} |
65 |
|
66 |
$response = false; |
67 |
while (!$response && is_dir($project) === true && file_exists($project . 'Config' . 'core.php')) { |
68 |
$prompt = __d('cake_console', '<warning>A project already exists in this location:</warning> %s Overwrite?', $project); |
69 |
$response = $this->in($prompt, array('y', 'n'), 'n'); |
70 |
if (strtolower($response) === 'n') { |
71 |
$response = $project = false; |
72 |
} |
73 |
} |
74 |
|
75 |
$success = true; |
76 |
if ($this->bake($project)) { |
77 |
$path = Folder::slashTerm($project); |
78 |
|
79 |
if ($this->securitySalt($path) === true) { |
80 |
$this->out(__d('cake_console', ' * Random hash key created for \'Security.salt\'')); |
81 |
} else {
|
82 |
$this->err(__d('cake_console', 'Unable to generate random hash for \'Security.salt\', you should change it in %s', APP . 'Config' . DS . 'core.php')); |
83 |
$success = false; |
84 |
} |
85 |
|
86 |
if ($this->securityCipherSeed($path) === true) { |
87 |
$this->out(__d('cake_console', ' * Random seed created for \'Security.cipherSeed\'')); |
88 |
} else {
|
89 |
$this->err(__d('cake_console', 'Unable to generate random seed for \'Security.cipherSeed\', you should change it in %s', APP . 'Config' . DS . 'core.php')); |
90 |
$success = false; |
91 |
} |
92 |
|
93 |
if ($this->cachePrefix($path)) { |
94 |
$this->out(__d('cake_console', ' * Cache prefix set')); |
95 |
} else {
|
96 |
$this->err(__d('cake_console', 'The cache prefix was <error>NOT</error> set')); |
97 |
$success = false; |
98 |
} |
99 |
|
100 |
if ($this->consolePath($path) === true) { |
101 |
$this->out(__d('cake_console', ' * app/Console/cake.php path set.')); |
102 |
} else {
|
103 |
$this->err(__d('cake_console', 'Unable to set console path for app/Console.')); |
104 |
$success = false; |
105 |
} |
106 |
|
107 |
$hardCode = false; |
108 |
if ($this->cakeOnIncludePath()) { |
109 |
$this->out(__d('cake_console', '<info>CakePHP is on your `include_path`. CAKE_CORE_INCLUDE_PATH will be set, but commented out.</info>')); |
110 |
} else {
|
111 |
$this->out(__d('cake_console', '<warning>CakePHP is not on your `include_path`, CAKE_CORE_INCLUDE_PATH will be hard coded.</warning>')); |
112 |
$this->out(__d('cake_console', 'You can fix this by adding CakePHP to your `include_path`.')); |
113 |
$hardCode = true; |
114 |
} |
115 |
$success = $this->corePath($path, $hardCode) === true; |
116 |
if ($success) { |
117 |
$this->out(__d('cake_console', ' * CAKE_CORE_INCLUDE_PATH set to %s in %s', CAKE_CORE_INCLUDE_PATH, 'webroot/index.php')); |
118 |
$this->out(__d('cake_console', ' * CAKE_CORE_INCLUDE_PATH set to %s in %s', CAKE_CORE_INCLUDE_PATH, 'webroot/test.php')); |
119 |
} else {
|
120 |
$this->err(__d('cake_console', 'Unable to set CAKE_CORE_INCLUDE_PATH, you should change it in %s', $path . 'webroot' . DS . 'index.php')); |
121 |
$success = false; |
122 |
} |
123 |
if ($success && $hardCode) { |
124 |
$this->out(__d('cake_console', ' * <warning>Remember to check these values after moving to production server</warning>')); |
125 |
} |
126 |
|
127 |
$Folder = new Folder($path); |
128 |
if (!$Folder->chmod($path . 'tmp', 0777)) { |
129 |
$this->err(__d('cake_console', 'Could not set permissions on %s', $path . DS . 'tmp')); |
130 |
$this->out('chmod -R 0777 ' . $path . DS . 'tmp'); |
131 |
$success = false; |
132 |
} |
133 |
if ($success) { |
134 |
$this->out(__d('cake_console', '<success>Project baked successfully!</success>')); |
135 |
} else {
|
136 |
$this->out(__d('cake_console', 'Project baked but with <warning>some issues.</warning>.')); |
137 |
} |
138 |
return $path; |
139 |
} |
140 |
} |
141 |
|
142 |
/**
|
143 |
* Checks PHP's include_path for CakePHP.
|
144 |
*
|
145 |
* @return bool Indicates whether or not CakePHP exists on include_path
|
146 |
*/
|
147 |
public function cakeOnIncludePath() { |
148 |
$paths = explode(PATH_SEPARATOR, ini_get('include_path')); |
149 |
foreach ($paths as $path) { |
150 |
if (file_exists($path . DS . 'Cake' . DS . 'bootstrap.php')) { |
151 |
return true; |
152 |
} |
153 |
} |
154 |
return false; |
155 |
} |
156 |
|
157 |
/**
|
158 |
* Looks for a skeleton template of a Cake application,
|
159 |
* and if not found asks the user for a path. When there is a path
|
160 |
* this method will make a deep copy of the skeleton to the project directory.
|
161 |
*
|
162 |
* @param string $path Project path
|
163 |
* @param string $skel Path to copy from
|
164 |
* @param string $skip array of directories to skip when copying
|
165 |
* @return mixed
|
166 |
*/
|
167 |
public function bake($path, $skel = null, $skip = array('empty')) { |
168 |
if (!$skel && !empty($this->params['skel'])) { |
169 |
$skel = $this->params['skel']; |
170 |
} |
171 |
while (!$skel) { |
172 |
$skel = $this->in( |
173 |
__d('cake_console', "What is the path to the directory layout you wish to copy?"), |
174 |
null,
|
175 |
CAKE . 'Console' . DS . 'Templates' . DS . 'skel' |
176 |
); |
177 |
if (!$skel) { |
178 |
$this->err(__d('cake_console', 'The directory path you supplied was empty. Please try again.')); |
179 |
} else {
|
180 |
while (is_dir($skel) === false) { |
181 |
$skel = $this->in( |
182 |
__d('cake_console', 'Directory path does not exist please choose another:'), |
183 |
null,
|
184 |
CAKE . 'Console' . DS . 'Templates' . DS . 'skel' |
185 |
); |
186 |
} |
187 |
} |
188 |
} |
189 |
|
190 |
$app = basename($path); |
191 |
|
192 |
$this->out(__d('cake_console', '<info>Skel Directory</info>: ') . $skel); |
193 |
$this->out(__d('cake_console', '<info>Will be copied to</info>: ') . $path); |
194 |
$this->hr();
|
195 |
|
196 |
$looksGood = $this->in(__d('cake_console', 'Look okay?'), array('y', 'n', 'q'), 'y'); |
197 |
|
198 |
switch (strtolower($looksGood)) { |
199 |
case 'y': |
200 |
$Folder = new Folder($skel); |
201 |
if (!empty($this->params['empty'])) { |
202 |
$skip = array(); |
203 |
} |
204 |
|
205 |
if ($Folder->copy(array('to' => $path, 'skip' => $skip))) { |
206 |
$this->hr();
|
207 |
$this->out(__d('cake_console', '<success>Created:</success> %s in %s', $app, $path)); |
208 |
$this->hr();
|
209 |
} else {
|
210 |
$this->err(__d('cake_console', "<error>Could not create</error> '%s' properly.", $app)); |
211 |
return false; |
212 |
} |
213 |
|
214 |
foreach ($Folder->messages() as $message) { |
215 |
$this->out(CakeText::wrap(' * ' . $message), 1, Shell::VERBOSE); |
216 |
} |
217 |
|
218 |
return true; |
219 |
case 'n': |
220 |
unset($this->args[0]); |
221 |
$this->execute();
|
222 |
return false; |
223 |
case 'q': |
224 |
$this->out(__d('cake_console', '<error>Bake Aborted.</error>')); |
225 |
return false; |
226 |
} |
227 |
} |
228 |
|
229 |
/**
|
230 |
* Generates the correct path to the CakePHP libs that are generating the project
|
231 |
* and points app/console/cake.php to the right place
|
232 |
*
|
233 |
* @param string $path Project path.
|
234 |
* @return bool success
|
235 |
*/
|
236 |
public function consolePath($path) { |
237 |
$File = new File($path . 'Console' . DS . 'cake.php'); |
238 |
$contents = $File->read(); |
239 |
if (preg_match('/(__CAKE_PATH__)/', $contents, $match)) { |
240 |
$root = strpos(CAKE_CORE_INCLUDE_PATH, '/') === 0 ? " DS . '" : "'"; |
241 |
$replacement = $root . str_replace(DS, "' . DS . '", trim(CAKE_CORE_INCLUDE_PATH, DS)) . "'"; |
242 |
$result = str_replace($match[0], $replacement, $contents); |
243 |
if ($File->write($result)) { |
244 |
return true; |
245 |
} |
246 |
return false; |
247 |
} |
248 |
return false; |
249 |
} |
250 |
|
251 |
/**
|
252 |
* Generates and writes 'Security.salt'
|
253 |
*
|
254 |
* @param string $path Project path
|
255 |
* @return bool Success
|
256 |
*/
|
257 |
public function securitySalt($path) { |
258 |
$File = new File($path . 'Config' . DS . 'core.php'); |
259 |
$contents = $File->read(); |
260 |
if (preg_match('/([\s]*Configure::write\(\'Security.salt\',[\s\'A-z0-9]*\);)/', $contents, $match)) { |
261 |
$string = Security::generateAuthKey(); |
262 |
$result = str_replace($match[0], "\t" . 'Configure::write(\'Security.salt\', \'' . $string . '\');', $contents); |
263 |
if ($File->write($result)) { |
264 |
return true; |
265 |
} |
266 |
return false; |
267 |
} |
268 |
return false; |
269 |
} |
270 |
|
271 |
/**
|
272 |
* Generates and writes 'Security.cipherSeed'
|
273 |
*
|
274 |
* @param string $path Project path
|
275 |
* @return bool Success
|
276 |
*/
|
277 |
public function securityCipherSeed($path) { |
278 |
$File = new File($path . 'Config' . DS . 'core.php'); |
279 |
$contents = $File->read(); |
280 |
if (preg_match('/([\s]*Configure::write\(\'Security.cipherSeed\',[\s\'A-z0-9]*\);)/', $contents, $match)) { |
281 |
App::uses('Security', 'Utility'); |
282 |
$string = substr(bin2hex(Security::generateAuthKey()), 0, 30); |
283 |
$result = str_replace($match[0], "\t" . 'Configure::write(\'Security.cipherSeed\', \'' . $string . '\');', $contents); |
284 |
if ($File->write($result)) { |
285 |
return true; |
286 |
} |
287 |
return false; |
288 |
} |
289 |
return false; |
290 |
} |
291 |
|
292 |
/**
|
293 |
* Writes cache prefix using app's name
|
294 |
*
|
295 |
* @param string $dir Path to project
|
296 |
* @return bool Success
|
297 |
*/
|
298 |
public function cachePrefix($dir) { |
299 |
$app = basename($dir); |
300 |
$File = new File($dir . 'Config' . DS . 'core.php'); |
301 |
$contents = $File->read(); |
302 |
if (preg_match('/(\$prefix = \'myapp_\';)/', $contents, $match)) { |
303 |
$result = str_replace($match[0], '$prefix = \'' . $app . '_\';', $contents); |
304 |
return $File->write($result); |
305 |
} |
306 |
return false; |
307 |
} |
308 |
|
309 |
/**
|
310 |
* Generates and writes CAKE_CORE_INCLUDE_PATH
|
311 |
*
|
312 |
* @param string $path Project path
|
313 |
* @param bool $hardCode Whether or not define calls should be hardcoded.
|
314 |
* @return bool Success
|
315 |
*/
|
316 |
public function corePath($path, $hardCode = true) { |
317 |
if (dirname($path) !== CAKE_CORE_INCLUDE_PATH) { |
318 |
$filename = $path . 'webroot' . DS . 'index.php'; |
319 |
if (!$this->_replaceCorePath($filename, $hardCode)) { |
320 |
return false; |
321 |
} |
322 |
$filename = $path . 'webroot' . DS . 'test.php'; |
323 |
if (!$this->_replaceCorePath($filename, $hardCode)) { |
324 |
return false; |
325 |
} |
326 |
return true; |
327 |
} |
328 |
} |
329 |
|
330 |
/**
|
331 |
* Replaces the __CAKE_PATH__ placeholder in the template files.
|
332 |
*
|
333 |
* @param string $filename The filename to operate on.
|
334 |
* @param bool $hardCode Whether or not the define should be uncommented.
|
335 |
* @return bool Success
|
336 |
*/
|
337 |
protected function _replaceCorePath($filename, $hardCode) { |
338 |
$contents = file_get_contents($filename); |
339 |
|
340 |
$root = strpos(CAKE_CORE_INCLUDE_PATH, '/') === 0 ? " DS . '" : "'"; |
341 |
$corePath = $root . str_replace(DS, "' . DS . '", trim(CAKE_CORE_INCLUDE_PATH, DS)) . "'"; |
342 |
|
343 |
$composer = ROOT . DS . APP_DIR . DS . 'Vendor' . DS . 'cakephp' . DS . 'cakephp' . DS . 'lib'; |
344 |
if (file_exists($composer)) { |
345 |
$corePath = " ROOT . DS . APP_DIR . DS . 'Vendor' . DS . 'cakephp' . DS . 'cakephp' . DS . 'lib'"; |
346 |
} |
347 |
|
348 |
$result = str_replace('__CAKE_PATH__', $corePath, $contents, $count); |
349 |
if ($hardCode) { |
350 |
$result = str_replace('//define(\'CAKE_CORE', 'define(\'CAKE_CORE', $result); |
351 |
} |
352 |
if (!file_put_contents($filename, $result)) { |
353 |
return false; |
354 |
} |
355 |
return (bool)$count; |
356 |
} |
357 |
|
358 |
/**
|
359 |
* Enables Configure::read('Routing.prefixes') in /app/Config/core.php
|
360 |
*
|
361 |
* @param string $name Name to use as admin routing
|
362 |
* @return bool Success
|
363 |
*/
|
364 |
public function cakeAdmin($name) { |
365 |
$path = (empty($this->configPath)) ? APP . 'Config' . DS : $this->configPath; |
366 |
$File = new File($path . 'core.php'); |
367 |
$contents = $File->read(); |
368 |
if (preg_match('%(\s*[/]*Configure::write\(\'Routing.prefixes\',[\s\'a-z,\)\(]*\);)%', $contents, $match)) { |
369 |
$result = str_replace($match[0], "\n" . 'Configure::write(\'Routing.prefixes\', array(\'' . $name . '\'));', $contents); |
370 |
if ($File->write($result)) { |
371 |
Configure::write('Routing.prefixes', array($name)); |
372 |
return true; |
373 |
} |
374 |
} |
375 |
return false; |
376 |
} |
377 |
|
378 |
/**
|
379 |
* Checks for Configure::read('Routing.prefixes') and forces user to input it if not enabled
|
380 |
*
|
381 |
* @return string Admin route to use
|
382 |
*/
|
383 |
public function getPrefix() { |
384 |
$admin = ''; |
385 |
$prefixes = Configure::read('Routing.prefixes'); |
386 |
if (!empty($prefixes)) { |
387 |
if (count($prefixes) === 1) { |
388 |
return $prefixes[0] . '_'; |
389 |
} |
390 |
if ($this->interactive) { |
391 |
$this->out();
|
392 |
$this->out(__d('cake_console', 'You have more than one routing prefix configured')); |
393 |
} |
394 |
$options = array(); |
395 |
foreach ($prefixes as $i => $prefix) { |
396 |
$options[] = $i + 1; |
397 |
if ($this->interactive) { |
398 |
$this->out($i + 1 . '. ' . $prefix); |
399 |
} |
400 |
} |
401 |
$selection = $this->in(__d('cake_console', 'Please choose a prefix to bake with.'), $options, 1); |
402 |
return $prefixes[$selection - 1] . '_'; |
403 |
} |
404 |
if ($this->interactive) { |
405 |
$this->hr();
|
406 |
$this->out(__d('cake_console', 'You need to enable %s in %s to use prefix routing.', |
407 |
'Configure::write(\'Routing.prefixes\', array(\'admin\'))',
|
408 |
'/app/Config/core.php'));
|
409 |
$this->out(__d('cake_console', 'What would you like the prefix route to be?')); |
410 |
$this->out(__d('cake_console', 'Example: %s', 'www.example.com/admin/controller')); |
411 |
while (!$admin) { |
412 |
$admin = $this->in(__d('cake_console', 'Enter a routing prefix:'), null, 'admin'); |
413 |
} |
414 |
if ($this->cakeAdmin($admin) !== true) { |
415 |
$this->out(__d('cake_console', '<error>Unable to write to</error> %s.', '/app/Config/core.php')); |
416 |
$this->out(__d('cake_console', 'You need to enable %s in %s to use prefix routing.', |
417 |
'Configure::write(\'Routing.prefixes\', array(\'admin\'))',
|
418 |
'/app/Config/core.php'));
|
419 |
return $this->_stop(); |
420 |
} |
421 |
return $admin . '_'; |
422 |
} |
423 |
return ''; |
424 |
} |
425 |
|
426 |
/**
|
427 |
* Gets the option parser instance and configures it.
|
428 |
*
|
429 |
* @return ConsoleOptionParser
|
430 |
*/
|
431 |
public function getOptionParser() { |
432 |
$parser = parent::getOptionParser(); |
433 |
|
434 |
$parser->description(
|
435 |
__d('cake_console', 'Generate a new CakePHP project skeleton.') |
436 |
)->addArgument('name', array( |
437 |
'help' => __d('cake_console', 'Application directory to make, if it starts with "/" the path is absolute.') |
438 |
))->addOption('empty', array( |
439 |
'boolean' => true, |
440 |
'help' => __d('cake_console', 'Create empty files in each of the directories. Good if you are using git') |
441 |
))->addOption('theme', array( |
442 |
'short' => 't', |
443 |
'help' => __d('cake_console', 'Theme to use when baking code.') |
444 |
))->addOption('skel', array( |
445 |
'default' => current(App::core('Console')) . 'Templates' . DS . 'skel', |
446 |
'help' => __d('cake_console', 'The directory layout to use for the new application skeleton.' . |
447 |
' Defaults to cake/Console/Templates/skel of CakePHP used to create the project.')
|
448 |
)); |
449 |
|
450 |
return $parser; |
451 |
} |
452 |
|
453 |
} |