369 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			369 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | ThinkPHP [ WE CAN DO IT JUST THINK IT ]
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Copyright (c) 2006-2016 http://thinkphp.cn All rights reserved.
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Author: yunwuxin <448901948@qq.com>
 | ||
| // +----------------------------------------------------------------------
 | ||
| 
 | ||
| namespace think\console\output\driver;
 | ||
| 
 | ||
| use think\console\Output;
 | ||
| use think\console\output\Formatter;
 | ||
| 
 | ||
| class Console
 | ||
| {
 | ||
| 
 | ||
|     /** @var  Resource */
 | ||
|     private $stdout;
 | ||
| 
 | ||
|     /** @var  Formatter */
 | ||
|     private $formatter;
 | ||
| 
 | ||
|     private $terminalDimensions;
 | ||
| 
 | ||
|     /** @var  Output */
 | ||
|     private $output;
 | ||
| 
 | ||
|     public function __construct(Output $output)
 | ||
|     {
 | ||
|         $this->output    = $output;
 | ||
|         $this->formatter = new Formatter();
 | ||
|         $this->stdout    = $this->openOutputStream();
 | ||
|         $decorated       = $this->hasColorSupport($this->stdout);
 | ||
|         $this->formatter->setDecorated($decorated);
 | ||
|     }
 | ||
| 
 | ||
|     public function setDecorated($decorated)
 | ||
|     {
 | ||
|         $this->formatter->setDecorated($decorated);
 | ||
|     }
 | ||
| 
 | ||
|     public function write($messages, $newline = false, $type = Output::OUTPUT_NORMAL, $stream = null)
 | ||
|     {
 | ||
|         if (Output::VERBOSITY_QUIET === $this->output->getVerbosity()) {
 | ||
|             return;
 | ||
|         }
 | ||
| 
 | ||
|         $messages = (array) $messages;
 | ||
| 
 | ||
|         foreach ($messages as $message) {
 | ||
|             switch ($type) {
 | ||
|                 case Output::OUTPUT_NORMAL:
 | ||
|                     $message = $this->formatter->format($message);
 | ||
|                     break;
 | ||
|                 case Output::OUTPUT_RAW:
 | ||
|                     break;
 | ||
|                 case Output::OUTPUT_PLAIN:
 | ||
|                     $message = strip_tags($this->formatter->format($message));
 | ||
|                     break;
 | ||
|                 default:
 | ||
|                     throw new \InvalidArgumentException(sprintf('Unknown output type given (%s)', $type));
 | ||
|             }
 | ||
| 
 | ||
|             $this->doWrite($message, $newline, $stream);
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     public function renderException(\Exception $e)
 | ||
|     {
 | ||
|         $stderr    = $this->openErrorStream();
 | ||
|         $decorated = $this->hasColorSupport($stderr);
 | ||
|         $this->formatter->setDecorated($decorated);
 | ||
| 
 | ||
|         do {
 | ||
|             $title = sprintf('  [%s]  ', get_class($e));
 | ||
| 
 | ||
|             $len = $this->stringWidth($title);
 | ||
| 
 | ||
|             $width = $this->getTerminalWidth() ? $this->getTerminalWidth() - 1 : PHP_INT_MAX;
 | ||
| 
 | ||
|             if (defined('HHVM_VERSION') && $width > 1 << 31) {
 | ||
|                 $width = 1 << 31;
 | ||
|             }
 | ||
|             $lines = [];
 | ||
|             foreach (preg_split('/\r?\n/', $e->getMessage()) as $line) {
 | ||
|                 foreach ($this->splitStringByWidth($line, $width - 4) as $line) {
 | ||
| 
 | ||
|                     $lineLength = $this->stringWidth(preg_replace('/\[[^m]*m/', '', $line)) + 4;
 | ||
|                     $lines[]    = [$line, $lineLength];
 | ||
| 
 | ||
|                     $len = max($lineLength, $len);
 | ||
|                 }
 | ||
|             }
 | ||
| 
 | ||
|             $messages   = ['', ''];
 | ||
|             $messages[] = $emptyLine = sprintf('<error>%s</error>', str_repeat(' ', $len));
 | ||
|             $messages[] = sprintf('<error>%s%s</error>', $title, str_repeat(' ', max(0, $len - $this->stringWidth($title))));
 | ||
|             foreach ($lines as $line) {
 | ||
|                 $messages[] = sprintf('<error>  %s  %s</error>', $line[0], str_repeat(' ', $len - $line[1]));
 | ||
|             }
 | ||
|             $messages[] = $emptyLine;
 | ||
|             $messages[] = '';
 | ||
|             $messages[] = '';
 | ||
| 
 | ||
|             $this->write($messages, true, Output::OUTPUT_NORMAL, $stderr);
 | ||
| 
 | ||
|             if (Output::VERBOSITY_VERBOSE <= $this->output->getVerbosity()) {
 | ||
|                 $this->write('<comment>Exception trace:</comment>', true, Output::OUTPUT_NORMAL, $stderr);
 | ||
| 
 | ||
|                 // exception related properties
 | ||
|                 $trace = $e->getTrace();
 | ||
|                 array_unshift($trace, [
 | ||
|                     'function' => '',
 | ||
|                     'file'     => $e->getFile() !== null ? $e->getFile() : 'n/a',
 | ||
|                     'line'     => $e->getLine() !== null ? $e->getLine() : 'n/a',
 | ||
|                     'args'     => [],
 | ||
|                 ]);
 | ||
| 
 | ||
|                 for ($i = 0, $count = count($trace); $i < $count; ++$i) {
 | ||
|                     $class    = isset($trace[$i]['class']) ? $trace[$i]['class'] : '';
 | ||
|                     $type     = isset($trace[$i]['type']) ? $trace[$i]['type'] : '';
 | ||
|                     $function = $trace[$i]['function'];
 | ||
|                     $file     = isset($trace[$i]['file']) ? $trace[$i]['file'] : 'n/a';
 | ||
|                     $line     = isset($trace[$i]['line']) ? $trace[$i]['line'] : 'n/a';
 | ||
| 
 | ||
|                     $this->write(sprintf(' %s%s%s() at <info>%s:%s</info>', $class, $type, $function, $file, $line), true, Output::OUTPUT_NORMAL, $stderr);
 | ||
|                 }
 | ||
| 
 | ||
|                 $this->write('', true, Output::OUTPUT_NORMAL, $stderr);
 | ||
|                 $this->write('', true, Output::OUTPUT_NORMAL, $stderr);
 | ||
|             }
 | ||
|         } while ($e = $e->getPrevious());
 | ||
| 
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 获取终端宽度
 | ||
|      * @return int|null
 | ||
|      */
 | ||
|     protected function getTerminalWidth()
 | ||
|     {
 | ||
|         $dimensions = $this->getTerminalDimensions();
 | ||
| 
 | ||
|         return $dimensions[0];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 获取终端高度
 | ||
|      * @return int|null
 | ||
|      */
 | ||
|     protected function getTerminalHeight()
 | ||
|     {
 | ||
|         $dimensions = $this->getTerminalDimensions();
 | ||
| 
 | ||
|         return $dimensions[1];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 获取当前终端的尺寸
 | ||
|      * @return array
 | ||
|      */
 | ||
|     public function getTerminalDimensions()
 | ||
|     {
 | ||
|         if ($this->terminalDimensions) {
 | ||
|             return $this->terminalDimensions;
 | ||
|         }
 | ||
| 
 | ||
|         if ('\\' === DIRECTORY_SEPARATOR) {
 | ||
|             if (preg_match('/^(\d+)x\d+ \(\d+x(\d+)\)$/', trim(getenv('ANSICON')), $matches)) {
 | ||
|                 return [(int) $matches[1], (int) $matches[2]];
 | ||
|             }
 | ||
|             if (preg_match('/^(\d+)x(\d+)$/', $this->getMode(), $matches)) {
 | ||
|                 return [(int) $matches[1], (int) $matches[2]];
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         if ($sttyString = $this->getSttyColumns()) {
 | ||
|             if (preg_match('/rows.(\d+);.columns.(\d+);/i', $sttyString, $matches)) {
 | ||
|                 return [(int) $matches[2], (int) $matches[1]];
 | ||
|             }
 | ||
|             if (preg_match('/;.(\d+).rows;.(\d+).columns/i', $sttyString, $matches)) {
 | ||
|                 return [(int) $matches[2], (int) $matches[1]];
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         return [null, null];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 获取stty列数
 | ||
|      * @return string
 | ||
|      */
 | ||
|     private function getSttyColumns()
 | ||
|     {
 | ||
|         if (!function_exists('proc_open')) {
 | ||
|             return;
 | ||
|         }
 | ||
| 
 | ||
|         $descriptorspec = [1 => ['pipe', 'w'], 2 => ['pipe', 'w']];
 | ||
|         $process        = proc_open('stty -a | grep columns', $descriptorspec, $pipes, null, null, ['suppress_errors' => true]);
 | ||
|         if (is_resource($process)) {
 | ||
|             $info = stream_get_contents($pipes[1]);
 | ||
|             fclose($pipes[1]);
 | ||
|             fclose($pipes[2]);
 | ||
|             proc_close($process);
 | ||
| 
 | ||
|             return $info;
 | ||
|         }
 | ||
|         return;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 获取终端模式
 | ||
|      * @return string <width>x<height> 或 null
 | ||
|      */
 | ||
|     private function getMode()
 | ||
|     {
 | ||
|         if (!function_exists('proc_open')) {
 | ||
|             return;
 | ||
|         }
 | ||
| 
 | ||
|         $descriptorspec = [1 => ['pipe', 'w'], 2 => ['pipe', 'w']];
 | ||
|         $process        = proc_open('mode CON', $descriptorspec, $pipes, null, null, ['suppress_errors' => true]);
 | ||
|         if (is_resource($process)) {
 | ||
|             $info = stream_get_contents($pipes[1]);
 | ||
|             fclose($pipes[1]);
 | ||
|             fclose($pipes[2]);
 | ||
|             proc_close($process);
 | ||
| 
 | ||
|             if (preg_match('/--------+\r?\n.+?(\d+)\r?\n.+?(\d+)\r?\n/', $info, $matches)) {
 | ||
|                 return $matches[2] . 'x' . $matches[1];
 | ||
|             }
 | ||
|         }
 | ||
|         return;
 | ||
|     }
 | ||
| 
 | ||
|     private function stringWidth($string)
 | ||
|     {
 | ||
|         if (!function_exists('mb_strwidth')) {
 | ||
|             return strlen($string);
 | ||
|         }
 | ||
| 
 | ||
|         if (false === $encoding = mb_detect_encoding($string)) {
 | ||
|             return strlen($string);
 | ||
|         }
 | ||
| 
 | ||
|         return mb_strwidth($string, $encoding);
 | ||
|     }
 | ||
| 
 | ||
|     private function splitStringByWidth($string, $width)
 | ||
|     {
 | ||
|         if (!function_exists('mb_strwidth')) {
 | ||
|             return str_split($string, $width);
 | ||
|         }
 | ||
| 
 | ||
|         if (false === $encoding = mb_detect_encoding($string)) {
 | ||
|             return str_split($string, $width);
 | ||
|         }
 | ||
| 
 | ||
|         $utf8String = mb_convert_encoding($string, 'utf8', $encoding);
 | ||
|         $lines      = [];
 | ||
|         $line       = '';
 | ||
|         foreach (preg_split('//u', $utf8String) as $char) {
 | ||
|             if (mb_strwidth($line . $char, 'utf8') <= $width) {
 | ||
|                 $line .= $char;
 | ||
|                 continue;
 | ||
|             }
 | ||
|             $lines[] = str_pad($line, $width);
 | ||
|             $line    = $char;
 | ||
|         }
 | ||
|         if (strlen($line)) {
 | ||
|             $lines[] = count($lines) ? str_pad($line, $width) : $line;
 | ||
|         }
 | ||
| 
 | ||
|         mb_convert_variables($encoding, 'utf8', $lines);
 | ||
| 
 | ||
|         return $lines;
 | ||
|     }
 | ||
| 
 | ||
|     private function isRunningOS400()
 | ||
|     {
 | ||
|         $checks = [
 | ||
|             function_exists('php_uname') ? php_uname('s') : '',
 | ||
|             getenv('OSTYPE'),
 | ||
|             PHP_OS,
 | ||
|         ];
 | ||
|         return false !== stripos(implode(';', $checks), 'OS400');
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 当前环境是否支持写入控制台输出到stdout.
 | ||
|      *
 | ||
|      * @return bool
 | ||
|      */
 | ||
|     protected function hasStdoutSupport()
 | ||
|     {
 | ||
|         return false === $this->isRunningOS400();
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 当前环境是否支持写入控制台输出到stderr.
 | ||
|      *
 | ||
|      * @return bool
 | ||
|      */
 | ||
|     protected function hasStderrSupport()
 | ||
|     {
 | ||
|         return false === $this->isRunningOS400();
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @return resource
 | ||
|      */
 | ||
|     private function openOutputStream()
 | ||
|     {
 | ||
|         if (!$this->hasStdoutSupport()) {
 | ||
|             return fopen('php://output', 'w');
 | ||
|         }
 | ||
|         return @fopen('php://stdout', 'w') ?: fopen('php://output', 'w');
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @return resource
 | ||
|      */
 | ||
|     private function openErrorStream()
 | ||
|     {
 | ||
|         return fopen($this->hasStderrSupport() ? 'php://stderr' : 'php://output', 'w');
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 将消息写入到输出。
 | ||
|      * @param string $message 消息
 | ||
|      * @param bool   $newline 是否另起一行
 | ||
|      * @param null   $stream
 | ||
|      */
 | ||
|     protected function doWrite($message, $newline, $stream = null)
 | ||
|     {
 | ||
|         if (null === $stream) {
 | ||
|             $stream = $this->stdout;
 | ||
|         }
 | ||
|         if (false === @fwrite($stream, $message . ($newline ? PHP_EOL : ''))) {
 | ||
|             throw new \RuntimeException('Unable to write output.');
 | ||
|         }
 | ||
| 
 | ||
|         fflush($stream);
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 是否支持着色
 | ||
|      * @param $stream
 | ||
|      * @return bool
 | ||
|      */
 | ||
|     protected function hasColorSupport($stream)
 | ||
|     {
 | ||
|         if (DIRECTORY_SEPARATOR === '\\') {
 | ||
|             return
 | ||
|             '10.0.10586' === PHP_WINDOWS_VERSION_MAJOR . '.' . PHP_WINDOWS_VERSION_MINOR . '.' . PHP_WINDOWS_VERSION_BUILD
 | ||
|             || false !== getenv('ANSICON')
 | ||
|             || 'ON' === getenv('ConEmuANSI')
 | ||
|             || 'xterm' === getenv('TERM');
 | ||
|         }
 | ||
| 
 | ||
|         return function_exists('posix_isatty') && @posix_isatty($stream);
 | ||
|     }
 | ||
| 
 | ||
| }
 | 
