| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321 | <?php/* * This file is part of Psy Shell. * * (c) 2012-2018 Justin Hileman * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */namespace Psy;/** * The Shell execution context. * * This class encapsulates the current variables, most recent return value and * exception, and the current namespace. */class Context{    private static $specialNames = ['_', '_e', '__out', '__psysh__', 'this'];    // Whitelist a very limited number of command-scope magic variable names.    // This might be a bad idea, but future me can sort it out.    private static $commandScopeNames = [        '__function', '__method', '__class', '__namespace', '__file', '__line', '__dir',    ];    private $scopeVariables = [];    private $commandScopeVariables = [];    private $returnValue;    private $lastException;    private $lastStdout;    private $boundObject;    private $boundClass;    /**     * Get a context variable.     *     * @throws InvalidArgumentException If the variable is not found in the current context     *     * @param string $name     *     * @return mixed     */    public function get($name)    {        switch ($name) {            case '_':                return $this->returnValue;            case '_e':                if (isset($this->lastException)) {                    return $this->lastException;                }                break;            case '__out':                if (isset($this->lastStdout)) {                    return $this->lastStdout;                }                break;            case 'this':                if (isset($this->boundObject)) {                    return $this->boundObject;                }                break;            case '__function':            case '__method':            case '__class':            case '__namespace':            case '__file':            case '__line':            case '__dir':                if (\array_key_exists($name, $this->commandScopeVariables)) {                    return $this->commandScopeVariables[$name];                }                break;            default:                if (\array_key_exists($name, $this->scopeVariables)) {                    return $this->scopeVariables[$name];                }                break;        }        throw new \InvalidArgumentException('Unknown variable: $' . $name);    }    /**     * Get all defined variables.     *     * @return array     */    public function getAll()    {        return \array_merge($this->scopeVariables, $this->getSpecialVariables());    }    /**     * Get all defined magic variables: $_, $_e, $__out, $__class, $__file, etc.     *     * @return array     */    public function getSpecialVariables()    {        $vars = [            '_' => $this->returnValue,        ];        if (isset($this->lastException)) {            $vars['_e'] = $this->lastException;        }        if (isset($this->lastStdout)) {            $vars['__out'] = $this->lastStdout;        }        if (isset($this->boundObject)) {            $vars['this'] = $this->boundObject;        }        return \array_merge($vars, $this->commandScopeVariables);    }    /**     * Set all scope variables.     *     * This method does *not* set any of the magic variables: $_, $_e, $__out,     * $__class, $__file, etc.     *     * @param array $vars     */    public function setAll(array $vars)    {        foreach (self::$specialNames as $key) {            unset($vars[$key]);        }        foreach (self::$commandScopeNames as $key) {            unset($vars[$key]);        }        $this->scopeVariables = $vars;    }    /**     * Set the most recent return value.     *     * @param mixed $value     */    public function setReturnValue($value)    {        $this->returnValue = $value;    }    /**     * Get the most recent return value.     *     * @return mixed     */    public function getReturnValue()    {        return $this->returnValue;    }    /**     * Set the most recent Exception.     *     * @param \Exception $e     */    public function setLastException(\Exception $e)    {        $this->lastException = $e;    }    /**     * Get the most recent Exception.     *     * @throws \InvalidArgumentException If no Exception has been caught     *     * @return null|\Exception     */    public function getLastException()    {        if (!isset($this->lastException)) {            throw new \InvalidArgumentException('No most-recent exception');        }        return $this->lastException;    }    /**     * Set the most recent output from evaluated code.     *     * @param string $lastStdout     */    public function setLastStdout($lastStdout)    {        $this->lastStdout = $lastStdout;    }    /**     * Get the most recent output from evaluated code.     *     * @throws \InvalidArgumentException If no output has happened yet     *     * @return null|string     */    public function getLastStdout()    {        if (!isset($this->lastStdout)) {            throw new \InvalidArgumentException('No most-recent output');        }        return $this->lastStdout;    }    /**     * Set the bound object ($this variable) for the interactive shell.     *     * Note that this unsets the bound class, if any exists.     *     * @param object|null $boundObject     */    public function setBoundObject($boundObject)    {        $this->boundObject = \is_object($boundObject) ? $boundObject : null;        $this->boundClass = null;    }    /**     * Get the bound object ($this variable) for the interactive shell.     *     * @return object|null     */    public function getBoundObject()    {        return $this->boundObject;    }    /**     * Set the bound class (self) for the interactive shell.     *     * Note that this unsets the bound object, if any exists.     *     * @param string|null $boundClass     */    public function setBoundClass($boundClass)    {        $this->boundClass = (\is_string($boundClass) && $boundClass !== '') ? $boundClass : null;        $this->boundObject = null;    }    /**     * Get the bound class (self) for the interactive shell.     *     * @return string|null     */    public function getBoundClass()    {        return $this->boundClass;    }    /**     * Set command-scope magic variables: $__class, $__file, etc.     *     * @param array $commandScopeVariables     */    public function setCommandScopeVariables(array $commandScopeVariables)    {        $vars = [];        foreach ($commandScopeVariables as $key => $value) {            // kind of type check            if (\is_scalar($value) && \in_array($key, self::$commandScopeNames)) {                $vars[$key] = $value;            }        }        $this->commandScopeVariables = $vars;    }    /**     * Get command-scope magic variables: $__class, $__file, etc.     *     * @return array     */    public function getCommandScopeVariables()    {        return $this->commandScopeVariables;    }    /**     * Get unused command-scope magic variables names: __class, __file, etc.     *     * This is used by the shell to unset old command-scope variables after a     * new batch is set.     *     * @return array Array of unused variable names     */    public function getUnusedCommandScopeVariableNames()    {        return \array_diff(self::$commandScopeNames, \array_keys($this->commandScopeVariables));    }    /**     * Check whether a variable name is a magic variable.     *     * @param string $name     *     * @return bool     */    public static function isSpecialVariableName($name)    {        return \in_array($name, self::$specialNames) || \in_array($name, self::$commandScopeNames);    }}
 |