356 lines
12 KiB
PHP
356 lines
12 KiB
PHP
<?php
|
|
/*
|
|
* $Id: Cli.php 2761 2007-10-07 23:42:29Z zYne $
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* This software consists of voluntary contributions made by many individuals
|
|
* and is licensed under the LGPL. For more information, see
|
|
* <http://www.phpdoctrine.org>.
|
|
*/
|
|
|
|
/**
|
|
* Command line interface class
|
|
* Interface for easily executing Doctrine_Task classes from a
|
|
* command line interface
|
|
*
|
|
* @package Doctrine
|
|
* @subpackage Cli
|
|
* @license http://www.opensource.org/licenses/lgpl-license.php LGPL
|
|
* @link www.phpdoctrine.org
|
|
* @since 1.0
|
|
* @version $Revision: 2761 $
|
|
* @author Jonathan H. Wage <jwage@mac.com>
|
|
*/
|
|
class Doctrine_Cli
|
|
{
|
|
protected $_tasks = array(),
|
|
$_taskInstance = null,
|
|
$_formatter = null,
|
|
$_scriptName = null,
|
|
$_message = null,
|
|
$_config = array();
|
|
|
|
/**
|
|
* __construct
|
|
*
|
|
* @param string $config
|
|
* @return void
|
|
*/
|
|
public function __construct($config = array())
|
|
{
|
|
$this->_config = $config;
|
|
$this->_formatter = new Doctrine_Cli_AnsiColorFormatter();
|
|
|
|
$this->loadTasks();
|
|
}
|
|
|
|
/**
|
|
* Notify the formatter of a message
|
|
*
|
|
* @param string $notification The notification message
|
|
* @param string $style Style to format the notification with(INFO, ERROR)
|
|
* @return void
|
|
*/
|
|
public function notify($notification = null, $style = 'HEADER')
|
|
{
|
|
echo $this->_formatter->format($this->_taskInstance->getTaskName(), 'INFO') . ' - ' . $this->_formatter->format($notification, $style) . "\n";
|
|
}
|
|
|
|
/**
|
|
* Notify the formatter of an exception
|
|
*
|
|
* @param Exception $exception
|
|
* @return void
|
|
*/
|
|
public function notifyException($exception)
|
|
{
|
|
echo $this->_formatter->format($exception->getMessage(), 'ERROR') . "\n";
|
|
}
|
|
|
|
/**
|
|
* Public function to run the loaded task with the passed arguments
|
|
*
|
|
* @param array $args
|
|
* @return void
|
|
* @throws new Doctrine_Cli_Exception
|
|
*/
|
|
public function run($args)
|
|
{
|
|
try {
|
|
$this->_run($args);
|
|
} catch (Exception $exception) {
|
|
$this->notifyException($exception);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the name of the task class based on the first argument
|
|
* which is always the task name. Do some inflection to determine the class name
|
|
*
|
|
* @param array $args Array of arguments from the cli
|
|
* @return string $taskClass Task class name
|
|
*/
|
|
protected function _getTaskClassFromArgs($args)
|
|
{
|
|
$taskName = str_replace('-', '_', $args[1]);
|
|
$taskClass = 'Doctrine_Task_' . Doctrine_Inflector::classify($taskName);
|
|
|
|
return $taskClass;
|
|
}
|
|
|
|
/**
|
|
* Run the actual task execution with the passed arguments
|
|
*
|
|
* @param array $args Array of arguments for this task being executed
|
|
* @return void
|
|
* @throws Doctrine_Cli_Exception $e
|
|
*/
|
|
protected function _run($args)
|
|
{
|
|
$this->_scriptName = $args[0];
|
|
|
|
$arg1 = isset($args[1]) ? $args[1]:null;
|
|
|
|
if ( ! $arg1 || $arg1 == 'help') {
|
|
echo $this->printTasks(null, $arg1 == 'help' ? true:false);
|
|
return;
|
|
}
|
|
|
|
if (isset($args[1]) && isset($args[2]) && $args[2] === 'help') {
|
|
echo $this->printTasks($args[1], true);
|
|
return;
|
|
}
|
|
|
|
$taskClass = $this->_getTaskClassFromArgs($args);
|
|
|
|
if ( ! class_exists($taskClass)) {
|
|
throw new Doctrine_Cli_Exception('Cli task could not be found: ' . $taskClass);
|
|
}
|
|
|
|
unset($args[0]);
|
|
unset($args[1]);
|
|
|
|
$this->_taskInstance = new $taskClass($this);
|
|
|
|
$args = $this->prepareArgs($args);
|
|
|
|
$this->_taskInstance->setArguments($args);
|
|
|
|
try {
|
|
if ($this->_taskInstance->validate()) {
|
|
$this->_taskInstance->execute();
|
|
} else {
|
|
echo $this->_formatter->format('Requires arguments missing!!', 'ERROR') . "\n\n";
|
|
echo $this->printTasks($arg1, true);
|
|
}
|
|
} catch (Exception $e) {
|
|
throw new Doctrine_Cli_Exception($e->getMessage());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Prepare the raw arguments for execution. Combines with the required and optional argument
|
|
* list in order to determine a complete array of arguments for the task
|
|
*
|
|
* @param array $args Array of raw arguments
|
|
* @return array $prepared Array of prepared arguments
|
|
*/
|
|
protected function prepareArgs($args)
|
|
{
|
|
$taskInstance = $this->_taskInstance;
|
|
|
|
$args = array_values($args);
|
|
|
|
// First lets load populate an array with all the possible arguments. required and optional
|
|
$prepared = array();
|
|
|
|
$requiredArguments = $taskInstance->getRequiredArguments();
|
|
foreach ($requiredArguments as $key => $arg) {
|
|
$prepared[$arg] = null;
|
|
}
|
|
|
|
$optionalArguments = $taskInstance->getOptionalArguments();
|
|
foreach ($optionalArguments as $key => $arg) {
|
|
$prepared[$arg] = null;
|
|
}
|
|
|
|
// If we have a config array then lets try and fill some of the arguments with the config values
|
|
if (is_array($this->_config) && !empty($this->_config)) {
|
|
foreach ($this->_config as $key => $value) {
|
|
if (array_key_exists($key, $prepared)) {
|
|
$prepared[$key] = $value;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now lets fill in the entered arguments to the prepared array
|
|
$copy = $args;
|
|
foreach ($prepared as $key => $value) {
|
|
if ( ! $value && !empty($copy)) {
|
|
$prepared[$key] = $copy[0];
|
|
unset($copy[0]);
|
|
$copy = array_values($copy);
|
|
}
|
|
}
|
|
|
|
return $prepared;
|
|
}
|
|
|
|
/**
|
|
* Prints an index of all the available tasks in the CLI instance
|
|
*
|
|
* @return void
|
|
*/
|
|
public function printTasks($task = null, $full = false)
|
|
{
|
|
$task = Doctrine_Inflector::classify(str_replace('-', '_', $task));
|
|
|
|
$tasks = $this->getLoadedTasks();
|
|
|
|
echo $this->_formatter->format("Doctrine Command Line Interface", 'HEADER') . "\n\n";
|
|
|
|
foreach ($tasks as $taskName)
|
|
{
|
|
if ($task != null && strtolower($task) != strtolower($taskName)) {
|
|
continue;
|
|
}
|
|
|
|
$className = 'Doctrine_Task_' . $taskName;
|
|
$taskInstance = new $className();
|
|
$taskInstance->taskName = str_replace('_', '-', Doctrine_Inflector::tableize($taskName));
|
|
|
|
$syntax = $this->_scriptName . ' ' . $taskInstance->getTaskName();
|
|
|
|
echo $this->_formatter->format($syntax, 'INFO');
|
|
|
|
if ($full) {
|
|
echo " - " . $taskInstance->getDescription() . "\n";
|
|
|
|
$args = null;
|
|
|
|
$requiredArguments = $taskInstance->getRequiredArgumentsDescriptions();
|
|
|
|
if ( ! empty($requiredArguments)) {
|
|
foreach ($requiredArguments as $name => $description) {
|
|
$args .= $this->_formatter->format($name, "ERROR");
|
|
|
|
if (isset($this->_config[$name])) {
|
|
$args .= " - " . $this->_formatter->format($this->_config[$name], 'COMMENT');
|
|
} else {
|
|
$args .= " - " . $description;
|
|
}
|
|
|
|
$args .= "\n";
|
|
}
|
|
}
|
|
|
|
$optionalArguments = $taskInstance->getOptionalArgumentsDescriptions();
|
|
|
|
if ( ! empty($optionalArguments)) {
|
|
foreach ($optionalArguments as $name => $description) {
|
|
$args .= $name . ' - ' . $description."\n";
|
|
}
|
|
}
|
|
|
|
if ($args) {
|
|
echo "\n" . $this->_formatter->format('Arguments:', 'HEADER') . "\n" . $args;
|
|
}
|
|
}
|
|
|
|
echo "\n";
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Load tasks from the passed directory. If no directory is given it looks in the default
|
|
* Doctrine/Task folder for the core tasks.
|
|
*
|
|
* @param mixed $directory Can be a string path or array of paths
|
|
* @return array $loadedTasks Array of tasks loaded
|
|
*/
|
|
public function loadTasks($directory = null)
|
|
{
|
|
if ($directory === null) {
|
|
$directory = Doctrine::getPath() . DIRECTORY_SEPARATOR . 'Doctrine' . DIRECTORY_SEPARATOR . 'Task';
|
|
}
|
|
|
|
$parent = new ReflectionClass('Doctrine_Task');
|
|
|
|
$tasks = array();
|
|
|
|
if (is_dir($directory)) {
|
|
foreach ((array) $directory as $dir) {
|
|
$it = new RecursiveIteratorIterator(new RecursiveDirectoryIterator($dir),
|
|
RecursiveIteratorIterator::LEAVES_ONLY);
|
|
|
|
foreach ($it as $file) {
|
|
$e = explode('.', $file->getFileName());
|
|
if (end($e) === 'php' && strpos($file->getFileName(), '.inc') === false) {
|
|
|
|
$className = 'Doctrine_Task_' . $e[0];
|
|
|
|
if ( ! class_exists($className)) {
|
|
require_once($file->getPathName());
|
|
|
|
$class = new ReflectionClass($className);
|
|
|
|
if ($class->isSubClassOf($parent)) {
|
|
$tasks[$e[0]] = $e[0];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$classes = get_declared_classes();
|
|
foreach ($classes as $className) {
|
|
$class = new Reflectionclass($className);
|
|
if ($class->isSubClassOf($parent)) {
|
|
$task = str_replace('Doctrine_Task_', '', $className);
|
|
$tasks[$task] = $task;
|
|
}
|
|
}
|
|
|
|
$this->_tasks = array_merge($this->_tasks, $tasks);
|
|
|
|
return $this->_tasks;
|
|
}
|
|
|
|
/**
|
|
* Get array of all the Doctrine_Task child classes that are loaded
|
|
*
|
|
* @return array $tasks
|
|
*/
|
|
public function getLoadedTasks()
|
|
{
|
|
$parent = new ReflectionClass('Doctrine_Task');
|
|
|
|
$classes = get_declared_classes();
|
|
|
|
$tasks = array();
|
|
|
|
foreach ($classes as $className) {
|
|
$class = new ReflectionClass($className);
|
|
|
|
if ($class->isSubClassOf($parent)) {
|
|
$task = str_replace('Doctrine_Task_', '', $className);
|
|
|
|
$tasks[$task] = $task;
|
|
}
|
|
}
|
|
|
|
return array_merge($this->_tasks, $tasks);
|
|
}
|
|
} |