You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

506 lines
13 KiB

<?php
use \EE_CLI\Utils;
use \EE_CLI\Dispatcher;
use \EE_CLI\FileCache;
use \EE_CLI\Process;
use \EE_CLI\WpHttpCacheManager;
/**
* Various utilities for EE-CLI commands.
*/
class EE_CLI {
private static $configurator;
private static $logger;
private static $hooks = array(), $hooks_passed = array();
/**
* Set the logger instance.
*
* @param object $logger
*/
public static function set_logger( $logger ) {
self::$logger = $logger;
}
/**
* Get the Configurator instance
*
* @return \EE_CLI\Configurator
*/
public static function get_configurator() {
static $configurator;
if ( !$configurator ) {
$configurator = new EE_CLI\Configurator( EE_CLI_ROOT . '/php/config-spec.php' );
}
return $configurator;
}
public static function get_root_command() {
static $root;
if ( !$root ) {
$root = new Dispatcher\RootCommand;
}
return $root;
}
public static function get_runner() {
static $runner;
if ( !$runner ) {
$runner = new EE_CLI\Runner;
}
return $runner;
}
/**
* @return FileCache
*/
public static function get_cache() {
static $cache;
if ( !$cache ) {
$home = getenv( 'HOME' );
if ( !$home ) {
// sometime in windows $HOME is not defined
$home = getenv( 'HOMEDRIVE' ) . getenv( 'HOMEPATH' );
}
$dir = getenv( 'EE_CLI_CACHE_DIR' ) ? : "$home/.EE-CLI/cache";
// 6 months, 300mb
$cache = new FileCache( $dir, 15552000, 314572800 );
// clean older files on shutdown with 1/50 probability
if ( 0 === mt_rand( 0, 50 ) ) {
register_shutdown_function( function () use ( $cache ) {
$cache->clean();
} );
}
}
return $cache;
}
/**
* Set the context in which EE-CLI should be run
*/
public static function set_url( $url ) {
EE_CLI::debug( 'Set URL: ' . $url );
$url_parts = Utils\parse_url( $url );
self::set_url_params( $url_parts );
}
private static function set_url_params( $url_parts ) {
$f = function( $key ) use ( $url_parts ) {
return \EE_CLI\Utils\get_flag_value( $url_parts, $key, '' );
};
if ( isset( $url_parts['host'] ) ) {
if ( isset( $url_parts['scheme'] ) && 'https' === strtolower( $url_parts['scheme'] ) ) {
$_SERVER['HTTPS'] = 'on';
}
$_SERVER['HTTP_HOST'] = $url_parts['host'];
if ( isset( $url_parts['port'] ) ) {
$_SERVER['HTTP_HOST'] .= ':' . $url_parts['port'];
}
$_SERVER['SERVER_NAME'] = $url_parts['host'];
}
$_SERVER['REQUEST_URI'] = $f('path') . ( isset( $url_parts['query'] ) ? '?' . $url_parts['query'] : '' );
$_SERVER['SERVER_PORT'] = \EE_CLI\Utils\get_flag_value( $url_parts, 'port', '80' );
$_SERVER['QUERY_STRING'] = $f('query');
}
/**
* @return WpHttpCacheManager
*/
public static function get_http_cache_manager() {
static $http_cacher;
if ( !$http_cacher ) {
$http_cacher = new WpHttpCacheManager( self::get_cache() );
}
return $http_cacher;
}
public static function colorize( $string ) {
return \cli\Colors::colorize( $string, self::get_runner()->in_color() );
}
/**
* Schedule a callback to be executed at a certain point (before WP is loaded).
*/
public static function add_hook( $when, $callback ) {
if ( in_array( $when, self::$hooks_passed ) )
call_user_func( $callback );
self::$hooks[ $when ][] = $callback;
}
/**
* Execute registered callbacks.
*/
public static function do_hook( $when ) {
self::$hooks_passed[] = $when;
if ( !isset( self::$hooks[ $when ] ) )
return;
foreach ( self::$hooks[ $when ] as $callback ) {
call_user_func( $callback );
}
}
/**
* Add a command to the EE-CLI list of commands
*
* @param string $name The name of the command that will be used in the CLI
* @param string $class The command implementation
* @param array $args An associative array with additional parameters:
* 'before_invoke' => callback to execute before invoking the command
*/
public static function add_command( $name, $class, $args = array() ) {
if ( is_string( $class ) && ! class_exists( (string) $class ) ) {
EE_CLI::error( sprintf( "Class '%s' does not exist.", $class ) );
}
if ( isset( $args['before_invoke'] ) ) {
self::add_hook( "before_invoke:$name", $args['before_invoke'] );
}
$path = preg_split( '/\s+/', $name );
$leaf_name = array_pop( $path );
$full_path = $path;
$command = self::get_root_command();
while ( !empty( $path ) ) {
$subcommand_name = $path[0];
$subcommand = $command->find_subcommand( $path );
// create an empty container
if ( !$subcommand ) {
$subcommand = new Dispatcher\CompositeCommand( $command, $subcommand_name,
new \EE_CLI\DocParser( '' ) );
$command->add_subcommand( $subcommand_name, $subcommand );
}
$command = $subcommand;
}
$leaf_command = Dispatcher\CommandFactory::create( $leaf_name, $class, $command );
if ( ! $command->can_have_subcommands() ) {
throw new Exception( sprintf( "'%s' can't have subcommands.",
implode( ' ' , Dispatcher\get_path( $command ) ) ) );
}
$command->add_subcommand( $leaf_name, $leaf_command );
}
/**
* Display a message in the CLI and end with a newline
*
* @param string $message
*/
public static function line( $message = '' ) {
echo $message . "\n";
}
/**
* Log an informational message.
*
* @param string $message
*/
public static function log( $message ) {
self::$logger->info( $message );
}
/**
* Display a success in the CLI and end with a newline
*
* @param string $message
*/
public static function success( $message ) {
self::$logger->success( $message );
}
/**
* Log debug information
*
* @param string $message
*/
public static function debug( $message ) {
self::$logger->debug( self::error_to_string( $message ) );
}
/**
* Display a warning in the CLI and end with a newline
*
* @param string $message
*/
public static function warning( $message ) {
self::$logger->warning( self::error_to_string( $message ) );
}
/**
* Display an error in the CLI and end with a newline
*
* @param string|EE_Error $message
* @param bool $exit if true, the script will exit()
*/
public static function error( $message, $exit = true ) {
if ( ! isset( self::get_runner()->assoc_args[ 'completions' ] ) ) {
self::$logger->error( self::error_to_string( $message ) );
}
if ( $exit ) {
exit(1);
}
}
/**
* Display an error in the CLI and end with a newline
*
* @param array $message each element from the array will be printed on its own line
*/
public static function error_multi_line( $message_lines ) {
if ( ! isset( self::get_runner()->assoc_args[ 'completions' ] ) && is_array( $message_lines ) ) {
self::$logger->error_multi_line( array_map( array( __CLASS__, 'error_to_string' ), $message_lines ) );
}
}
/**
* Ask for confirmation before running a destructive operation.
*/
public static function confirm( $question, $assoc_args = array() ) {
if ( ! \EE_CLI\Utils\get_flag_value( $assoc_args, 'yes' ) ) {
fwrite( STDOUT, $question . " [y/n] " );
$answer = trim( fgets( STDIN ) );
if ( 'y' != $answer )
exit;
}
}
/**
* Read value from a positional argument or from STDIN.
*
* @param array $args The list of positional arguments.
* @param int $index At which position to check for the value.
*
* @return string
*/
public static function get_value_from_arg_or_stdin( $args, $index ) {
if ( isset( $args[ $index ] ) ) {
$raw_value = $args[ $index ];
} else {
// We don't use file_get_contents() here because it doesn't handle
// Ctrl-D properly, when typing in the value interactively.
$raw_value = '';
while ( ( $line = fgets( STDIN ) ) !== false ) {
$raw_value .= $line;
}
}
return $raw_value;
}
/**
* Read a value, from various formats.
*
* @param mixed $value
* @param array $assoc_args
*/
public static function read_value( $raw_value, $assoc_args = array() ) {
if ( \EE_CLI\Utils\get_flag_value( $assoc_args, 'format' ) === 'json' ) {
$value = json_decode( $raw_value, true );
if ( null === $value ) {
EE_CLI::error( sprintf( 'Invalid JSON: %s', $raw_value ) );
}
} else {
$value = $raw_value;
}
return $value;
}
/**
* Display a value, in various formats
*
* @param mixed $value
* @param array $assoc_args
*/
public static function print_value( $value, $assoc_args = array() ) {
if ( \EE_CLI\Utils\get_flag_value( $assoc_args, 'format' ) === 'json' ) {
$value = json_encode( $value );
} elseif ( is_array( $value ) || is_object( $value ) ) {
$value = var_export( $value );
}
echo $value . "\n";
}
/**
* Convert a ee_error into a string
*
* @param mixed $errors
* @return string
*/
public static function error_to_string( $errors ) {
if ( is_string( $errors ) ) {
return $errors;
}
if ( is_object( $errors ) && is_a( $errors, 'EE_Error' ) ) {
foreach ( $errors->get_error_messages() as $message ) {
if ( $errors->get_error_data() )
return $message . ' ' . $errors->get_error_data();
else
return $message;
}
}
}
/**
* Launch an external process that takes over I/O.
*
* @param string Command to call
* @param bool Whether to exit if the command returns an error status
* @param bool Whether to return an exit status (default) or detailed execution results
*
* @return int|ProcessRun The command exit status, or a ProcessRun instance
*/
public static function launch( $command, $exit_on_error = true, $return_detailed = false ) {
$proc = Process::create( $command );
$results = $proc->run();
if ( $results->return_code && $exit_on_error )
exit( $results->return_code );
if ( $return_detailed ) {
return $results;
} else {
return $results->return_code;
}
}
/**
* Launch another EE-CLI command using the runtime arguments for the current process
*
* @param string Command to call
* @param array $args Positional arguments to use
* @param array $assoc_args Associative arguments to use
* @param bool Whether to exit if the command returns an error status
* @param bool Whether to return an exit status (default) or detailed execution results
* @param array $runtime_args Override one or more global args (path,url,user,allow-root)
*
* @return int|ProcessRun The command exit status, or a ProcessRun instance
*/
public static function launch_self( $command, $args = array(), $assoc_args = array(), $exit_on_error = true, $return_detailed = false, $runtime_args = array() ) {
$reused_runtime_args = array(
'path',
'url',
'user',
'allow-root',
);
foreach ( $reused_runtime_args as $key ) {
if ( isset( $runtime_args[ $key ] ) ) {
$assoc_args[ $key ] = $runtime_args[ $key ];
} else if ( $value = self::get_runner()->config[ $key ] )
$assoc_args[ $key ] = $value;
}
$php_bin = self::get_php_binary();
$script_path = $GLOBALS['argv'][0];
$args = implode( ' ', array_map( 'escapeshellarg', $args ) );
$assoc_args = \EE_CLI\Utils\assoc_args_to_str( $assoc_args );
$full_command = "{$php_bin} {$script_path} {$command} {$args} {$assoc_args}";
return self::launch( $full_command, $exit_on_error, $return_detailed );
}
/**
* Get the path to the PHP binary used when executing EE-CLI.
* Environment values permit specific binaries to be indicated.
*
* @return string
*/
public static function get_php_binary() {
if ( defined( 'PHP_BINARY' ) )
return PHP_BINARY;
if ( getenv( 'EE_CLI_PHP_USED' ) )
return getenv( 'EE_CLI_PHP_USED' );
if ( getenv( 'EE_CLI_PHP' ) )
return getenv( 'EE_CLI_PHP' );
return 'php';
}
public static function get_config( $key = null ) {
if ( null === $key ) {
return self::get_runner()->config;
}
if ( !isset( self::get_runner()->config[ $key ] ) ) {
self::warning( "Unknown config option '$key'." );
return null;
}
return self::get_runner()->config[ $key ];
}
/**
* Run a given command within the current process using the same global parameters.
*
* To run a command using a new process with the same global parameters, use EE_CLI::launch_self()
* To run a command using a new process with different global parameters, use EE_CLI::launch()
*
* @param array
* @param array
*/
public static function run_command( $args, $assoc_args = array() ) {
self::get_runner()->run_command( $args, $assoc_args );
}
// DEPRECATED STUFF
public static function add_man_dir() {
trigger_error( 'EE_CLI::add_man_dir() is deprecated. Add docs inline.', E_USER_WARNING );
}
// back-compat
public static function out( $str ) {
fwrite( STDOUT, $str );
}
// back-compat
public static function addCommand( $name, $class ) {
trigger_error( sprintf( 'ee %s: %s is deprecated. use EE_CLI::add_command() instead.',
$name, __FUNCTION__ ), E_USER_WARNING );
self::add_command( $name, $class );
}
}