diff options
Diffstat (limited to 'framework/console/CConsoleCommand.php')
| -rw-r--r-- | framework/console/CConsoleCommand.php | 502 |
1 files changed, 502 insertions, 0 deletions
diff --git a/framework/console/CConsoleCommand.php b/framework/console/CConsoleCommand.php new file mode 100644 index 0000000..87fbcea --- /dev/null +++ b/framework/console/CConsoleCommand.php @@ -0,0 +1,502 @@ +<?php +/** + * CConsoleCommand class file. + * + * @author Qiang Xue <qiang.xue@gmail.com> + * @link http://www.yiiframework.com/ + * @copyright Copyright © 2008-2011 Yii Software LLC + * @license http://www.yiiframework.com/license/ + */ + +/** + * CConsoleCommand represents an executable console command. + * + * It works like {@link CController} by parsing command line options and dispatching + * the request to a specific action with appropriate option values. + * + * Users call a console command via the following command format: + * <pre> + * yiic CommandName ActionName --Option1=Value1 --Option2=Value2 ... + * </pre> + * + * Child classes mainly needs to implement various action methods whose name must be + * prefixed with "action". The parameters to an action method are considered as options + * for that specific action. The action specified as {@link defaultAction} will be invoked + * when a user does not specify the action name in his command. + * + * Options are bound to action parameters via parameter names. For example, the following + * action method will allow us to run a command with <code>yiic sitemap --type=News</code>: + * <pre> + * class SitemapCommand { + * public function actionIndex($type) { + * .... + * } + * } + * </pre> + * + * @property string $name The command name. + * @property CConsoleCommandRunner $commandRunner The command runner instance. + * @property string $help The command description. Defaults to 'Usage: php entry-script.php command-name'. + * @property array $optionHelp The command option help information. Each array element describes + * the help information for a single action. + * + * @author Qiang Xue <qiang.xue@gmail.com> + * @version $Id: CConsoleCommand.php 3548 2012-01-24 11:42:59Z mdomba $ + * @package system.console + * @since 1.0 + */ +abstract class CConsoleCommand extends CComponent +{ + /** + * @var string the name of the default action. Defaults to 'index'. + * @since 1.1.5 + */ + public $defaultAction='index'; + + private $_name; + private $_runner; + + /** + * Constructor. + * @param string $name name of the command + * @param CConsoleCommandRunner $runner the command runner + */ + public function __construct($name,$runner) + { + $this->_name=$name; + $this->_runner=$runner; + } + + /** + * Initializes the command object. + * This method is invoked after a command object is created and initialized with configurations. + * You may override this method to further customize the command before it executes. + * @since 1.1.6 + */ + public function init() + { + } + + /** + * Executes the command. + * The default implementation will parse the input parameters and + * dispatch the command request to an appropriate action with the corresponding + * option values + * @param array $args command line parameters for this command. + */ + public function run($args) + { + list($action, $options, $args)=$this->resolveRequest($args); + $methodName='action'.$action; + if(!preg_match('/^\w+$/',$action) || !method_exists($this,$methodName)) + $this->usageError("Unknown action: ".$action); + + $method=new ReflectionMethod($this,$methodName); + $params=array(); + // named and unnamed options + foreach($method->getParameters() as $i=>$param) + { + $name=$param->getName(); + if(isset($options[$name])) + { + if($param->isArray()) + $params[]=is_array($options[$name]) ? $options[$name] : array($options[$name]); + else if(!is_array($options[$name])) + $params[]=$options[$name]; + else + $this->usageError("Option --$name requires a scalar. Array is given."); + } + else if($name==='args') + $params[]=$args; + else if($param->isDefaultValueAvailable()) + $params[]=$param->getDefaultValue(); + else + $this->usageError("Missing required option --$name."); + unset($options[$name]); + } + + // try global options + if(!empty($options)) + { + $class=new ReflectionClass(get_class($this)); + foreach($options as $name=>$value) + { + if($class->hasProperty($name)) + { + $property=$class->getProperty($name); + if($property->isPublic() && !$property->isStatic()) + { + $this->$name=$value; + unset($options[$name]); + } + } + } + } + + if(!empty($options)) + $this->usageError("Unknown options: ".implode(', ',array_keys($options))); + + if($this->beforeAction($action,$params)) + { + $method->invokeArgs($this,$params); + $this->afterAction($action,$params); + } + } + + /** + * This method is invoked right before an action is to be executed. + * You may override this method to do last-minute preparation for the action. + * @param string $action the action name + * @param array $params the parameters to be passed to the action method. + * @return boolean whether the action should be executed. + */ + protected function beforeAction($action,$params) + { + return true; + } + + /** + * This method is invoked right after an action finishes execution. + * You may override this method to do some postprocessing for the action. + * @param string $action the action name + * @param array $params the parameters to be passed to the action method. + */ + protected function afterAction($action,$params) + { + } + + /** + * Parses the command line arguments and determines which action to perform. + * @param array $args command line arguments + * @return array the action name, named options (name=>value), and unnamed options + * @since 1.1.5 + */ + protected function resolveRequest($args) + { + $options=array(); // named parameters + $params=array(); // unnamed parameters + foreach($args as $arg) + { + if(preg_match('/^--(\w+)(=(.*))?$/',$arg,$matches)) // an option + { + $name=$matches[1]; + $value=isset($matches[3]) ? $matches[3] : true; + if(isset($options[$name])) + { + if(!is_array($options[$name])) + $options[$name]=array($options[$name]); + $options[$name][]=$value; + } + else + $options[$name]=$value; + } + else if(isset($action)) + $params[]=$arg; + else + $action=$arg; + } + if(!isset($action)) + $action=$this->defaultAction; + + return array($action,$options,$params); + } + + /** + * @return string the command name. + */ + public function getName() + { + return $this->_name; + } + + /** + * @return CConsoleCommandRunner the command runner instance + */ + public function getCommandRunner() + { + return $this->_runner; + } + + /** + * Provides the command description. + * This method may be overridden to return the actual command description. + * @return string the command description. Defaults to 'Usage: php entry-script.php command-name'. + */ + public function getHelp() + { + $help='Usage: '.$this->getCommandRunner()->getScriptName().' '.$this->getName(); + $options=$this->getOptionHelp(); + if(empty($options)) + return $help; + if(count($options)===1) + return $help.' '.$options[0]; + $help.=" <action>\nActions:\n"; + foreach($options as $option) + $help.=' '.$option."\n"; + return $help; + } + + /** + * Provides the command option help information. + * The default implementation will return all available actions together with their + * corresponding option information. + * @return array the command option help information. Each array element describes + * the help information for a single action. + * @since 1.1.5 + */ + public function getOptionHelp() + { + $options=array(); + $class=new ReflectionClass(get_class($this)); + foreach($class->getMethods(ReflectionMethod::IS_PUBLIC) as $method) + { + $name=$method->getName(); + if(!strncasecmp($name,'action',6) && strlen($name)>6) + { + $name=substr($name,6); + $name[0]=strtolower($name[0]); + $help=$name; + + foreach($method->getParameters() as $param) + { + $optional=$param->isDefaultValueAvailable(); + $defaultValue=$optional ? $param->getDefaultValue() : null; + $name=$param->getName(); + if($optional) + $help.=" [--$name=$defaultValue]"; + else + $help.=" --$name=value"; + } + $options[]=$help; + } + } + return $options; + } + + /** + * Displays a usage error. + * This method will then terminate the execution of the current application. + * @param string $message the error message + */ + public function usageError($message) + { + echo "Error: $message\n\n".$this->getHelp()."\n"; + exit(1); + } + + /** + * Copies a list of files from one place to another. + * @param array $fileList the list of files to be copied (name=>spec). + * The array keys are names displayed during the copy process, and array values are specifications + * for files to be copied. Each array value must be an array of the following structure: + * <ul> + * <li>source: required, the full path of the file/directory to be copied from</li> + * <li>target: required, the full path of the file/directory to be copied to</li> + * <li>callback: optional, the callback to be invoked when copying a file. The callback function + * should be declared as follows: + * <pre> + * function foo($source,$params) + * </pre> + * where $source parameter is the source file path, and the content returned + * by the function will be saved into the target file.</li> + * <li>params: optional, the parameters to be passed to the callback</li> + * </ul> + * @see buildFileList + */ + public function copyFiles($fileList) + { + $overwriteAll=false; + foreach($fileList as $name=>$file) + { + $source=strtr($file['source'],'/\\',DIRECTORY_SEPARATOR); + $target=strtr($file['target'],'/\\',DIRECTORY_SEPARATOR); + $callback=isset($file['callback']) ? $file['callback'] : null; + $params=isset($file['params']) ? $file['params'] : null; + + if(is_dir($source)) + { + $this->ensureDirectory($target); + continue; + } + + if($callback!==null) + $content=call_user_func($callback,$source,$params); + else + $content=file_get_contents($source); + if(is_file($target)) + { + if($content===file_get_contents($target)) + { + echo " unchanged $name\n"; + continue; + } + if($overwriteAll) + echo " overwrite $name\n"; + else + { + echo " exist $name\n"; + echo " ...overwrite? [Yes|No|All|Quit] "; + $answer=trim(fgets(STDIN)); + if(!strncasecmp($answer,'q',1)) + return; + else if(!strncasecmp($answer,'y',1)) + echo " overwrite $name\n"; + else if(!strncasecmp($answer,'a',1)) + { + echo " overwrite $name\n"; + $overwriteAll=true; + } + else + { + echo " skip $name\n"; + continue; + } + } + } + else + { + $this->ensureDirectory(dirname($target)); + echo " generate $name\n"; + } + file_put_contents($target,$content); + } + } + + /** + * Builds the file list of a directory. + * This method traverses through the specified directory and builds + * a list of files and subdirectories that the directory contains. + * The result of this function can be passed to {@link copyFiles}. + * @param string $sourceDir the source directory + * @param string $targetDir the target directory + * @param string $baseDir base directory + * @return array the file list (see {@link copyFiles}) + */ + public function buildFileList($sourceDir, $targetDir, $baseDir='') + { + $list=array(); + $handle=opendir($sourceDir); + while(($file=readdir($handle))!==false) + { + if($file==='.' || $file==='..' || $file==='.svn' ||$file==='.yii') + continue; + $sourcePath=$sourceDir.DIRECTORY_SEPARATOR.$file; + $targetPath=$targetDir.DIRECTORY_SEPARATOR.$file; + $name=$baseDir===''?$file : $baseDir.'/'.$file; + $list[$name]=array('source'=>$sourcePath, 'target'=>$targetPath); + if(is_dir($sourcePath)) + $list=array_merge($list,$this->buildFileList($sourcePath,$targetPath,$name)); + } + closedir($handle); + return $list; + } + + /** + * Creates all parent directories if they do not exist. + * @param string $directory the directory to be checked + */ + public function ensureDirectory($directory) + { + if(!is_dir($directory)) + { + $this->ensureDirectory(dirname($directory)); + echo " mkdir ".strtr($directory,'\\','/')."\n"; + mkdir($directory); + } + } + + /** + * Renders a view file. + * @param string $_viewFile_ view file path + * @param array $_data_ optional data to be extracted as local view variables + * @param boolean $_return_ whether to return the rendering result instead of displaying it + * @return mixed the rendering result if required. Null otherwise. + */ + public function renderFile($_viewFile_,$_data_=null,$_return_=false) + { + if(is_array($_data_)) + extract($_data_,EXTR_PREFIX_SAME,'data'); + else + $data=$_data_; + if($_return_) + { + ob_start(); + ob_implicit_flush(false); + require($_viewFile_); + return ob_get_clean(); + } + else + require($_viewFile_); + } + + /** + * Converts a word to its plural form. + * @param string $name the word to be pluralized + * @return string the pluralized word + */ + public function pluralize($name) + { + $rules=array( + '/move$/i' => 'moves', + '/foot$/i' => 'feet', + '/child$/i' => 'children', + '/human$/i' => 'humans', + '/man$/i' => 'men', + '/tooth$/i' => 'teeth', + '/person$/i' => 'people', + '/([m|l])ouse$/i' => '\1ice', + '/(x|ch|ss|sh|us|as|is|os)$/i' => '\1es', + '/([^aeiouy]|qu)y$/i' => '\1ies', + '/(?:([^f])fe|([lr])f)$/i' => '\1\2ves', + '/(shea|lea|loa|thie)f$/i' => '\1ves', + '/([ti])um$/i' => '\1a', + '/(tomat|potat|ech|her|vet)o$/i' => '\1oes', + '/(bu)s$/i' => '\1ses', + '/(ax|test)is$/i' => '\1es', + '/s$/' => 's', + ); + foreach($rules as $rule=>$replacement) + { + if(preg_match($rule,$name)) + return preg_replace($rule,$replacement,$name); + } + return $name.'s'; + } + + /** + * Reads input via the readline PHP extension if that's available, or fgets() if readline is not installed. + * + * @param string $message to echo out before waiting for user input + * @return mixed line read as a string, or false if input has been closed + * + * @since 1.1.9 + */ + public function prompt($message) + { + if(extension_loaded('readline')) + { + $input = readline($message.' '); + readline_add_history($input); + return $input; + } + else + { + echo $message.' '; + return trim(fgets(STDIN)); + } + } + + /** + * Asks user to confirm by typing y or n. + * + * @param string $message to echo out before waiting for user input + * @return bool if user confirmed + * + * @since 1.1.9 + */ + public function confirm($message) + { + echo $message.' [yes|no] '; + return !strncasecmp(trim(fgets(STDIN)),'y',1); + } +}
\ No newline at end of file |
