| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544 | <?phpnamespace app\admin\command\Api\library;use Exception;/** * Class imported from https://github.com/eriknyk/Annotations * @author  Erik Amaru Ortiz https://github.com/eriknyk * * @license http://opensource.org/licenses/bsd-license.php The BSD License * @author  Calin Rada <rada.calin@gmail.com> */class Extractor{    /**     * Static array to store already parsed annotations     * @var array     */    private static $annotationCache;    private static $classAnnotationCache;    private static $classMethodAnnotationCache;    private static $classPropertyValueCache;    /**     * Indicates that annotations should has strict behavior, 'false' by default     * @var boolean     */    private $strict = false;    /**     * Stores the default namespace for Objects instance, usually used on methods like getMethodAnnotationsObjects()     * @var string     */    public $defaultNamespace = '';    /**     * Sets strict variable to true/false     * @param bool $value boolean value to indicate that annotations to has strict behavior     */    public function setStrict($value)    {        $this->strict = (bool)$value;    }    /**     * Sets default namespace to use in object instantiation     * @param string $namespace default namespace     */    public function setDefaultNamespace($namespace)    {        $this->defaultNamespace = $namespace;    }    /**     * Gets default namespace used in object instantiation     * @return string $namespace default namespace     */    public function getDefaultAnnotationNamespace()    {        return $this->defaultNamespace;    }    /**     * Gets all anotations with pattern @SomeAnnotation() from a given class     *     * @param string $className class name to get annotations     * @return array  self::$classAnnotationCache all annotated elements     */    public static function getClassAnnotations($className)    {        if (!isset(self::$classAnnotationCache[$className])) {            $class = new \ReflectionClass($className);            $annotationArr = self::parseAnnotations($class->getDocComment());            $annotationArr['ApiTitle'] = !isset($annotationArr['ApiTitle'][0]) || !trim($annotationArr['ApiTitle'][0]) ? [$class->getShortName()] : $annotationArr['ApiTitle'];            self::$classAnnotationCache[$className] = $annotationArr;        }        return self::$classAnnotationCache[$className];    }    /**     * 获取类所有方法的属性配置     * @param $className     * @return mixed     * @throws \ReflectionException     */    public static function getClassMethodAnnotations($className)    {        $class = new \ReflectionClass($className);        foreach ($class->getMethods() as $object) {            self::$classMethodAnnotationCache[$className][$object->name] = self::getMethodAnnotations($className, $object->name);        }        return self::$classMethodAnnotationCache[$className];    }    public static function getClassPropertyValues($className)    {        $class = new \ReflectionClass($className);        foreach ($class->getProperties() as $object) {            self::$classPropertyValueCache[$className][$object->name] = self::getClassPropertyValue($className, $object->name);        }        return self::$classMethodAnnotationCache[$className];    }    public static function getAllClassAnnotations()    {        return self::$classAnnotationCache;    }    public static function getAllClassMethodAnnotations()    {        return self::$classMethodAnnotationCache;    }    public static function getAllClassPropertyValues()    {        return self::$classPropertyValueCache;    }    public static function getClassPropertyValue($className, $property)    {        $_SERVER['REQUEST_METHOD'] = 'GET';        $reflectionClass = new \ReflectionClass($className);        $reflectionProperty = $reflectionClass->getProperty($property);        $reflectionProperty->setAccessible(true);        return $reflectionProperty->getValue($reflectionClass->newInstanceWithoutConstructor());    }    /**     * Gets all anotations with pattern @SomeAnnotation() from a determinated method of a given class     *     * @param string $className  class name     * @param string $methodName method name to get annotations     * @return array  self::$annotationCache all annotated elements of a method given     */    public static function getMethodAnnotations($className, $methodName)    {        if (!isset(self::$annotationCache[$className . '::' . $methodName])) {            try {                $method = new \ReflectionMethod($className, $methodName);                $class = new \ReflectionClass($className);                if (!$method->isPublic() || $method->isConstructor()) {                    $annotations = array();                } else {                    $annotations = self::consolidateAnnotations($method, $class);                }            } catch (\ReflectionException $e) {                $annotations = array();            }            self::$annotationCache[$className . '::' . $methodName] = $annotations;        }        return self::$annotationCache[$className . '::' . $methodName];    }    /**     * Gets all anotations with pattern @SomeAnnotation() from a determinated method of a given class     * and instance its abcAnnotation class     *     * @param string $className  class name     * @param string $methodName method name to get annotations     * @return array  self::$annotationCache all annotated objects of a method given     */    public function getMethodAnnotationsObjects($className, $methodName)    {        $annotations = $this->getMethodAnnotations($className, $methodName);        $objects = array();        $i = 0;        foreach ($annotations as $annotationClass => $listParams) {            $annotationClass = ucfirst($annotationClass);            $class = $this->defaultNamespace . $annotationClass . 'Annotation';            // verify is the annotation class exists, depending if Annotations::strict is true            // if not, just skip the annotation instance creation.            if (!class_exists($class)) {                if ($this->strict) {                    throw new Exception(sprintf('Runtime Error: Annotation Class Not Found: %s', $class));                } else {                    // silent skip & continue                    continue;                }            }            if (empty($objects[$annotationClass])) {                $objects[$annotationClass] = new $class();            }            foreach ($listParams as $params) {                if (is_array($params)) {                    foreach ($params as $key => $value) {                        $objects[$annotationClass]->set($key, $value);                    }                } else {                    $objects[$annotationClass]->set($i++, $params);                }            }        }        return $objects;    }    private static function consolidateAnnotations($method, $class)    {        $dockblockClass = $class->getDocComment();        $docblockMethod = $method->getDocComment();        $methodName = $method->getName();        $methodAnnotations = self::parseAnnotations($docblockMethod);        $methodAnnotations['ApiTitle'] = !isset($methodAnnotations['ApiTitle'][0]) || !trim($methodAnnotations['ApiTitle'][0]) ? [$method->getName()] : $methodAnnotations['ApiTitle'];        $classAnnotations = self::parseAnnotations($dockblockClass);        $classAnnotations['ApiTitle'] = !isset($classAnnotations['ApiTitle'][0]) || !trim($classAnnotations['ApiTitle'][0]) ? [$class->getShortName()] : $classAnnotations['ApiTitle'];        if (isset($methodAnnotations['ApiInternal']) || $methodName == '_initialize' || $methodName == '_empty') {            return [];        }        $properties = $class->getDefaultProperties();        $noNeedLogin = isset($properties['noNeedLogin']) ? (is_array($properties['noNeedLogin']) ? $properties['noNeedLogin'] : [$properties['noNeedLogin']]) : [];        $noNeedRight = isset($properties['noNeedRight']) ? (is_array($properties['noNeedRight']) ? $properties['noNeedRight'] : [$properties['noNeedRight']]) : [];        preg_match_all("/\*[\s]+(.*)(\\r\\n|\\r|\\n)/U", str_replace('/**', '', $docblockMethod), $methodArr);        preg_match_all("/\*[\s]+(.*)(\\r\\n|\\r|\\n)/U", str_replace('/**', '', $dockblockClass), $classArr);        if (!isset($methodAnnotations['ApiMethod'])) {            $methodAnnotations['ApiMethod'] = ['get'];        }        if (!isset($methodAnnotations['ApiWeigh'])) {            $methodAnnotations['ApiWeigh'] = [0];        }        if (!isset($methodAnnotations['ApiSummary'])) {            $methodAnnotations['ApiSummary'] = $methodAnnotations['ApiTitle'];        }        if ($methodAnnotations) {            foreach ($classAnnotations as $name => $valueClass) {                if (count($valueClass) !== 1) {                    continue;                }                if ($name === 'ApiRoute') {                    if (isset($methodAnnotations[$name])) {                        $methodAnnotations[$name] = [rtrim($valueClass[0], '/') . $methodAnnotations[$name][0]];                    } else {                        $methodAnnotations[$name] = [rtrim($valueClass[0], '/') . '/' . $method->getName()];                    }                }                if ($name === 'ApiSector') {                    $methodAnnotations[$name] = $valueClass;                }            }        }        if (!isset($methodAnnotations['ApiRoute'])) {            $urlArr = [];            $className = $class->getName();            list($prefix, $suffix) = explode('\\' . \think\Config::get('url_controller_layer') . '\\', $className);            $prefixArr = explode('\\', $prefix);            $suffixArr = explode('\\', $suffix);            if ($prefixArr[0] == \think\Config::get('app_namespace')) {                $prefixArr[0] = '';            }            $urlArr = array_merge($urlArr, $prefixArr);            $urlArr[] = implode('.', array_map(function ($item) {                return \think\Loader::parseName($item);            }, $suffixArr));            $urlArr[] = $method->getName();            $methodAnnotations['ApiRoute'] = [implode('/', $urlArr)];        }        if (!isset($methodAnnotations['ApiSector'])) {            $methodAnnotations['ApiSector'] = isset($classAnnotations['ApiSector']) ? $classAnnotations['ApiSector'] : $classAnnotations['ApiTitle'];        }        if (!isset($methodAnnotations['ApiParams'])) {            $params = self::parseCustomAnnotations($docblockMethod, 'param');            foreach ($params as $k => $v) {                $arr = explode(' ', preg_replace("/[\s]+/", " ", $v));                $methodAnnotations['ApiParams'][] = [                    'name'        => isset($arr[1]) ? str_replace('$', '', $arr[1]) : '',                    'nullable'    => false,                    'type'        => isset($arr[0]) ? $arr[0] : 'string',                    'description' => isset($arr[2]) ? $arr[2] : ''                ];            }        }        $methodAnnotations['ApiPermissionLogin'] = [!in_array('*', $noNeedLogin) && !in_array($methodName, $noNeedLogin)];        $methodAnnotations['ApiPermissionRight'] = !$methodAnnotations['ApiPermissionLogin'][0] ? [false] : [!in_array('*', $noNeedRight) && !in_array($methodName, $noNeedRight)];        return $methodAnnotations;    }    /**     * Parse annotations     *     * @param string $docblock     * @param string $name     * @return array  parsed annotations params     */    private static function parseCustomAnnotations($docblock, $name = 'param')    {        $annotations = array();        $docblock = substr($docblock, 3, -2);        if (preg_match_all('/@' . $name . '(?:\s*(?:\(\s*)?(.*?)(?:\s*\))?)??\s*(?:\n|\*\/)/', $docblock, $matches)) {            foreach ($matches[1] as $k => $v) {                $annotations[] = $v;            }        }        return $annotations;    }    /**     * Parse annotations     *     * @param string $docblock     * @return array  parsed annotations params     */    private static function parseAnnotations($docblock)    {        $annotations = array();        // Strip away the docblock header and footer to ease parsing of one line annotations        $docblock = substr($docblock, 3, -2);        if (preg_match_all('/@(?<name>[A-Za-z_-]+)[\s\t]*\((?<args>(?:(?!\)).)*)\)\r?/s', $docblock, $matches)) {            $numMatches = count($matches[0]);            for ($i = 0; $i < $numMatches; ++$i) {                $name = $matches['name'][$i];                $value = '';                // annotations has arguments                if (isset($matches['args'][$i])) {                    $argsParts = trim($matches['args'][$i]);                    if ($name == 'ApiReturn') {                        $value = $argsParts;                    } elseif ($matches['args'][$i] != '') {                        $argsParts = preg_replace("/\{(\w+)\}/", '#$1#', $argsParts);                        $value = self::parseArgs($argsParts);                        if (is_string($value)) {                            $value = preg_replace("/\#(\w+)\#/", '{$1}', $argsParts);                        }                    }                }                $annotations[$name][] = $value;            }        }        if (stripos($docblock, '@ApiInternal') !== false) {            $annotations['ApiInternal'] = [true];        }        if (!isset($annotations['ApiTitle'])) {            preg_match_all("/\*[\s]+(.*)(\\r\\n|\\r|\\n)/U", str_replace('/**', '', $docblock), $matchArr);            $title = isset($matchArr[1]) && isset($matchArr[1][0]) ? $matchArr[1][0] : '';            $annotations['ApiTitle'] = [$title];        }        return $annotations;    }    /**     * Parse individual annotation arguments     *     * @param string $content arguments string     * @return array  annotated arguments     */    private static function parseArgs($content)    {        // Replace initial stars        $content = preg_replace('/^\s*\*/m', '', $content);        $data = array();        $len = strlen($content);        $i = 0;        $var = '';        $val = '';        $level = 1;        $prevDelimiter = '';        $nextDelimiter = '';        $nextToken = '';        $composing = false;        $type = 'plain';        $delimiter = null;        $quoted = false;        $tokens = array('"', '"', '{', '}', ',', '=');        while ($i <= $len) {            $prev_c = substr($content, $i - 1, 1);            $c = substr($content, $i++, 1);            if ($c === '"' && $prev_c !== "\\") {                $delimiter = $c;                //open delimiter                if (!$composing && empty($prevDelimiter) && empty($nextDelimiter)) {                    $prevDelimiter = $nextDelimiter = $delimiter;                    $val = '';                    $composing = true;                    $quoted = true;                } else {                    // close delimiter                    if ($c !== $nextDelimiter) {                        throw new Exception(sprintf(                            "Parse Error: enclosing error -> expected: [%s], given: [%s]",                            $nextDelimiter,                            $c                        ));                    }                    // validating syntax                    if ($i < $len) {                        if (',' !== substr($content, $i, 1) && '\\' !== $prev_c) {                            throw new Exception(sprintf(                                "Parse Error: missing comma separator near: ...%s<--",                                substr($content, ($i - 10), $i)                            ));                        }                    }                    $prevDelimiter = $nextDelimiter = '';                    $composing = false;                    $delimiter = null;                }            } elseif (!$composing && in_array($c, $tokens)) {                switch ($c) {                    case '=':                        $prevDelimiter = $nextDelimiter = '';                        $level = 2;                        $composing = false;                        $type = 'assoc';                        $quoted = false;                        break;                    case ',':                        $level = 3;                        // If composing flag is true yet,                        // it means that the string was not enclosed, so it is parsing error.                        if ($composing === true && !empty($prevDelimiter) && !empty($nextDelimiter)) {                            throw new Exception(sprintf(                                "Parse Error: enclosing error -> expected: [%s], given: [%s]",                                $nextDelimiter,                                $c                            ));                        }                        $prevDelimiter = $nextDelimiter = '';                        break;                    case '{':                        $subc = '';                        $subComposing = true;                        while ($i <= $len) {                            $c = substr($content, $i++, 1);                            if (isset($delimiter) && $c === $delimiter) {                                throw new Exception(sprintf(                                    "Parse Error: Composite variable is not enclosed correctly."                                ));                            }                            if ($c === '}') {                                $subComposing = false;                                break;                            }                            $subc .= $c;                        }                        // if the string is composing yet means that the structure of var. never was enclosed with '}'                        if ($subComposing) {                            throw new Exception(sprintf(                                "Parse Error: Composite variable is not enclosed correctly. near: ...%s'",                                $subc                            ));                        }                        $val = self::parseArgs($subc);                        break;                }            } else {                if ($level == 1) {                    $var .= $c;                } elseif ($level == 2) {                    $val .= $c;                }            }            if ($level === 3 || $i === $len) {                if ($type == 'plain' && $i === $len) {                    $data = self::castValue($var);                } else {                    $data[trim($var)] = self::castValue($val, !$quoted);                }                $level = 1;                $var = $val = '';                $composing = false;                $quoted = false;            }        }        return $data;    }    /**     * Try determinate the original type variable of a string     *     * @param string  $val  string containing possibles variables that can be cast to bool or int     * @param boolean $trim indicate if the value passed should be trimmed after to try cast     * @return mixed   returns the value converted to original type if was possible     */    private static function castValue($val, $trim = false)    {        if (is_array($val)) {            foreach ($val as $key => $value) {                $val[$key] = self::castValue($value);            }        } elseif (is_string($val)) {            if ($trim) {                $val = trim($val);            }            $val = stripslashes($val);            $tmp = strtolower($val);            if ($tmp === 'false' || $tmp === 'true') {                $val = $tmp === 'true';            } elseif (is_numeric($val)) {                return $val + 0;            }            unset($tmp);        }        return $val;    }}
 |