Extractor.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. <?php
  2. namespace app\admin\command\Api\library;
  3. use Exception;
  4. /**
  5. * Class imported from https://github.com/eriknyk/Annotations
  6. * @author Erik Amaru Ortiz https://github.com/eriknyk‎
  7. *
  8. * @license http://opensource.org/licenses/bsd-license.php The BSD License
  9. * @author Calin Rada <rada.calin@gmail.com>
  10. */
  11. class Extractor
  12. {
  13. /**
  14. * Static array to store already parsed annotations
  15. * @var array
  16. */
  17. private static $annotationCache;
  18. /**
  19. * Indicates that annotations should has strict behavior, 'false' by default
  20. * @var boolean
  21. */
  22. private $strict = false;
  23. /**
  24. * Stores the default namespace for Objects instance, usually used on methods like getMethodAnnotationsObjects()
  25. * @var string
  26. */
  27. public $defaultNamespace = '';
  28. /**
  29. * Sets strict variable to true/false
  30. * @param bool $value boolean value to indicate that annotations to has strict behavior
  31. */
  32. public function setStrict($value)
  33. {
  34. $this->strict = (bool)$value;
  35. }
  36. /**
  37. * Sets default namespace to use in object instantiation
  38. * @param string $namespace default namespace
  39. */
  40. public function setDefaultNamespace($namespace)
  41. {
  42. $this->defaultNamespace = $namespace;
  43. }
  44. /**
  45. * Gets default namespace used in object instantiation
  46. * @return string $namespace default namespace
  47. */
  48. public function getDefaultAnnotationNamespace()
  49. {
  50. return $this->defaultNamespace;
  51. }
  52. /**
  53. * Gets all anotations with pattern @SomeAnnotation() from a given class
  54. *
  55. * @param string $className class name to get annotations
  56. * @return array self::$annotationCache all annotated elements
  57. */
  58. public static function getClassAnnotations($className)
  59. {
  60. if (!isset(self::$annotationCache[$className])) {
  61. $class = new \ReflectionClass($className);
  62. self::$annotationCache[$className] = self::parseAnnotations($class->getDocComment());
  63. }
  64. return self::$annotationCache[$className];
  65. }
  66. public static function getAllClassAnnotations($className)
  67. {
  68. $class = new \ReflectionClass($className);
  69. foreach ($class->getMethods() as $object) {
  70. self::$annotationCache['annotations'][$className][$object->name] = self::getMethodAnnotations($className, $object->name);
  71. }
  72. return self::$annotationCache['annotations'];
  73. }
  74. /**
  75. * Gets all anotations with pattern @SomeAnnotation() from a determinated method of a given class
  76. *
  77. * @param string $className class name
  78. * @param string $methodName method name to get annotations
  79. * @return array self::$annotationCache all annotated elements of a method given
  80. */
  81. public static function getMethodAnnotations($className, $methodName)
  82. {
  83. if (!isset(self::$annotationCache[$className . '::' . $methodName])) {
  84. try {
  85. $method = new \ReflectionMethod($className, $methodName);
  86. $class = new \ReflectionClass($className);
  87. if (!$method->isPublic() || $method->isConstructor()) {
  88. $annotations = array();
  89. } else {
  90. $annotations = self::consolidateAnnotations($method, $class);
  91. }
  92. } catch (\ReflectionException $e) {
  93. $annotations = array();
  94. }
  95. self::$annotationCache[$className . '::' . $methodName] = $annotations;
  96. }
  97. return self::$annotationCache[$className . '::' . $methodName];
  98. }
  99. /**
  100. * Gets all anotations with pattern @SomeAnnotation() from a determinated method of a given class
  101. * and instance its abcAnnotation class
  102. *
  103. * @param string $className class name
  104. * @param string $methodName method name to get annotations
  105. * @return array self::$annotationCache all annotated objects of a method given
  106. */
  107. public function getMethodAnnotationsObjects($className, $methodName)
  108. {
  109. $annotations = $this->getMethodAnnotations($className, $methodName);
  110. $objects = array();
  111. $i = 0;
  112. foreach ($annotations as $annotationClass => $listParams) {
  113. $annotationClass = ucfirst($annotationClass);
  114. $class = $this->defaultNamespace . $annotationClass . 'Annotation';
  115. // verify is the annotation class exists, depending if Annotations::strict is true
  116. // if not, just skip the annotation instance creation.
  117. if (!class_exists($class)) {
  118. if ($this->strict) {
  119. throw new Exception(sprintf('Runtime Error: Annotation Class Not Found: %s', $class));
  120. } else {
  121. // silent skip & continue
  122. continue;
  123. }
  124. }
  125. if (empty($objects[$annotationClass])) {
  126. $objects[$annotationClass] = new $class();
  127. }
  128. foreach ($listParams as $params) {
  129. if (is_array($params)) {
  130. foreach ($params as $key => $value) {
  131. $objects[$annotationClass]->set($key, $value);
  132. }
  133. } else {
  134. $objects[$annotationClass]->set($i++, $params);
  135. }
  136. }
  137. }
  138. return $objects;
  139. }
  140. private static function consolidateAnnotations($method, $class)
  141. {
  142. $dockblockClass = $class->getDocComment();
  143. $docblockMethod = $method->getDocComment();
  144. $methodName = $method->getName();
  145. $methodAnnotations = self::parseAnnotations($docblockMethod);
  146. $classAnnotations = self::parseAnnotations($dockblockClass);
  147. if (isset($methodAnnotations['ApiInternal']) || $methodName == '_initialize' || $methodName == '_empty') {
  148. return [];
  149. }
  150. $properties = $class->getDefaultProperties();
  151. $noNeedLogin = isset($properties['noNeedLogin']) ? is_array($properties['noNeedLogin']) ? $properties['noNeedLogin'] : [$properties['noNeedLogin']] : [];
  152. $noNeedRight = isset($properties['noNeedRight']) ? is_array($properties['noNeedRight']) ? $properties['noNeedRight'] : [$properties['noNeedRight']] : [];
  153. preg_match_all("/\*[\s]+(.*)(\\r\\n|\\r|\\n)/U", str_replace('/**', '', $docblockMethod), $methodArr);
  154. preg_match_all("/\*[\s]+(.*)(\\r\\n|\\r|\\n)/U", str_replace('/**', '', $dockblockClass), $classArr);
  155. $methodTitle = isset($methodArr[1]) && isset($methodArr[1][0]) ? $methodArr[1][0] : '';
  156. $classTitle = isset($classArr[1]) && isset($classArr[1][0]) ? $classArr[1][0] : '';
  157. if (!isset($methodAnnotations['ApiMethod'])) {
  158. $methodAnnotations['ApiMethod'] = ['get'];
  159. }
  160. if (!isset($methodAnnotations['ApiSummary'])) {
  161. $methodAnnotations['ApiSummary'] = [$methodTitle];
  162. }
  163. if ($methodAnnotations) {
  164. foreach ($classAnnotations as $name => $valueClass) {
  165. if (count($valueClass) !== 1) {
  166. continue;
  167. }
  168. if ($name === 'ApiRoute') {
  169. if (isset($methodAnnotations[$name])) {
  170. $methodAnnotations[$name] = [rtrim($valueClass[0], '/') . $methodAnnotations[$name][0]];
  171. } else {
  172. $methodAnnotations[$name] = [rtrim($valueClass[0], '/') . '/' . $method->getName()];
  173. }
  174. }
  175. if ($name === 'ApiSector') {
  176. $methodAnnotations[$name] = $valueClass;
  177. }
  178. }
  179. }
  180. if (!isset($methodAnnotations['ApiTitle'])) {
  181. $methodAnnotations['ApiTitle'] = [$methodTitle];
  182. }
  183. if (!isset($methodAnnotations['ApiRoute'])) {
  184. $urlArr = [];
  185. $className = $class->getName();
  186. list($prefix, $suffix) = explode('\\' . \think\Config::get('url_controller_layer') . '\\', $className);
  187. $prefixArr = explode('\\', $prefix);
  188. $suffixArr = explode('\\', $suffix);
  189. if ($prefixArr[0] == \think\Config::get('app_namespace')) {
  190. $prefixArr[0] = '';
  191. }
  192. $urlArr = array_merge($urlArr, $prefixArr);
  193. $urlArr[] = implode('.', array_map(function ($item) {
  194. return \think\Loader::parseName($item);
  195. }, $suffixArr));
  196. $urlArr[] = $method->getName();
  197. $methodAnnotations['ApiRoute'] = [implode('/', $urlArr)];
  198. }
  199. if (!isset($methodAnnotations['ApiSector'])) {
  200. $methodAnnotations['ApiSector'] = isset($classAnnotations['ApiSector']) ? $classAnnotations['ApiSector'] : [$classTitle];
  201. }
  202. if (!isset($methodAnnotations['ApiParams'])) {
  203. $params = self::parseCustomAnnotations($docblockMethod, 'param');
  204. foreach ($params as $k => $v) {
  205. $arr = explode(' ', preg_replace("/[\s]+/", " ", $v));
  206. $methodAnnotations['ApiParams'][] = [
  207. 'name' => isset($arr[1]) ? str_replace('$', '', $arr[1]) : '',
  208. 'nullable' => false,
  209. 'type' => isset($arr[0]) ? $arr[0] : 'string',
  210. 'description' => isset($arr[2]) ? $arr[2] : ''
  211. ];
  212. }
  213. }
  214. $methodAnnotations['ApiPermissionLogin'] = [!in_array('*', $noNeedLogin) && !in_array($methodName, $noNeedLogin)];
  215. $methodAnnotations['ApiPermissionRight'] = [!in_array('*', $noNeedRight) && !in_array($methodName, $noNeedRight)];
  216. return $methodAnnotations;
  217. }
  218. /**
  219. * Parse annotations
  220. *
  221. * @param string $docblock
  222. * @param string $name
  223. * @return array parsed annotations params
  224. */
  225. private static function parseCustomAnnotations($docblock, $name = 'param')
  226. {
  227. $annotations = array();
  228. $docblock = substr($docblock, 3, -2);
  229. if (preg_match_all('/@' . $name . '(?:\s*(?:\(\s*)?(.*?)(?:\s*\))?)??\s*(?:\n|\*\/)/', $docblock, $matches)) {
  230. foreach ($matches[1] as $k => $v) {
  231. $annotations[] = $v;
  232. }
  233. }
  234. return $annotations;
  235. }
  236. /**
  237. * Parse annotations
  238. *
  239. * @param string $docblock
  240. * @return array parsed annotations params
  241. */
  242. private static function parseAnnotations($docblock)
  243. {
  244. $annotations = array();
  245. // Strip away the docblock header and footer to ease parsing of one line annotations
  246. $docblock = substr($docblock, 3, -2);
  247. if (preg_match_all('/@(?<name>[A-Za-z_-]+)[\s\t]*\((?<args>(?:(?!\)).)*)\)\r?/s', $docblock, $matches)) {
  248. $numMatches = count($matches[0]);
  249. for ($i = 0; $i < $numMatches; ++$i) {
  250. $name = $matches['name'][$i];
  251. $value = '';
  252. // annotations has arguments
  253. if (isset($matches['args'][$i])) {
  254. $argsParts = trim($matches['args'][$i]);
  255. if ($name == 'ApiReturn') {
  256. $value = $argsParts;
  257. } else if ($matches['args'][$i] != '') {
  258. $argsParts = preg_replace("/\{(\w+)\}/", '#$1#', $argsParts);
  259. $value = self::parseArgs($argsParts);
  260. if (is_string($value)) {
  261. $value = preg_replace("/\#(\w+)\#/", '{$1}', $argsParts);
  262. }
  263. }
  264. }
  265. $annotations[$name][] = $value;
  266. }
  267. }
  268. if (stripos($docblock, '@ApiInternal') !== false) {
  269. $annotations['ApiInternal'] = [true];
  270. }
  271. return $annotations;
  272. }
  273. /**
  274. * Parse individual annotation arguments
  275. *
  276. * @param string $content arguments string
  277. * @return array annotated arguments
  278. */
  279. private static function parseArgs($content)
  280. {
  281. // Replace initial stars
  282. $content = preg_replace('/^\s*\*/m', '', $content);
  283. $data = array();
  284. $len = strlen($content);
  285. $i = 0;
  286. $var = '';
  287. $val = '';
  288. $level = 1;
  289. $prevDelimiter = '';
  290. $nextDelimiter = '';
  291. $nextToken = '';
  292. $composing = false;
  293. $type = 'plain';
  294. $delimiter = null;
  295. $quoted = false;
  296. $tokens = array('"', '"', '{', '}', ',', '=');
  297. while ($i <= $len) {
  298. $prev_c = substr($content, $i - 1, 1);
  299. $c = substr($content, $i++, 1);
  300. if ($c === '"' && $prev_c !== "\\") {
  301. $delimiter = $c;
  302. //open delimiter
  303. if (!$composing && empty($prevDelimiter) && empty($nextDelimiter)) {
  304. $prevDelimiter = $nextDelimiter = $delimiter;
  305. $val = '';
  306. $composing = true;
  307. $quoted = true;
  308. } else {
  309. // close delimiter
  310. if ($c !== $nextDelimiter) {
  311. throw new Exception(sprintf(
  312. "Parse Error: enclosing error -> expected: [%s], given: [%s]", $nextDelimiter, $c
  313. ));
  314. }
  315. // validating syntax
  316. if ($i < $len) {
  317. if (',' !== substr($content, $i, 1) && '\\' !== $prev_c) {
  318. throw new Exception(sprintf(
  319. "Parse Error: missing comma separator near: ...%s<--", substr($content, ($i - 10), $i)
  320. ));
  321. }
  322. }
  323. $prevDelimiter = $nextDelimiter = '';
  324. $composing = false;
  325. $delimiter = null;
  326. }
  327. } elseif (!$composing && in_array($c, $tokens)) {
  328. switch ($c) {
  329. case '=':
  330. $prevDelimiter = $nextDelimiter = '';
  331. $level = 2;
  332. $composing = false;
  333. $type = 'assoc';
  334. $quoted = false;
  335. break;
  336. case ',':
  337. $level = 3;
  338. // If composing flag is true yet,
  339. // it means that the string was not enclosed, so it is parsing error.
  340. if ($composing === true && !empty($prevDelimiter) && !empty($nextDelimiter)) {
  341. throw new Exception(sprintf(
  342. "Parse Error: enclosing error -> expected: [%s], given: [%s]", $nextDelimiter, $c
  343. ));
  344. }
  345. $prevDelimiter = $nextDelimiter = '';
  346. break;
  347. case '{':
  348. $subc = '';
  349. $subComposing = true;
  350. while ($i <= $len) {
  351. $c = substr($content, $i++, 1);
  352. if (isset($delimiter) && $c === $delimiter) {
  353. throw new Exception(sprintf(
  354. "Parse Error: Composite variable is not enclosed correctly."
  355. ));
  356. }
  357. if ($c === '}') {
  358. $subComposing = false;
  359. break;
  360. }
  361. $subc .= $c;
  362. }
  363. // if the string is composing yet means that the structure of var. never was enclosed with '}'
  364. if ($subComposing) {
  365. throw new Exception(sprintf(
  366. "Parse Error: Composite variable is not enclosed correctly. near: ...%s'", $subc
  367. ));
  368. }
  369. $val = self::parseArgs($subc);
  370. break;
  371. }
  372. } else {
  373. if ($level == 1) {
  374. $var .= $c;
  375. } elseif ($level == 2) {
  376. $val .= $c;
  377. }
  378. }
  379. if ($level === 3 || $i === $len) {
  380. if ($type == 'plain' && $i === $len) {
  381. $data = self::castValue($var);
  382. } else {
  383. $data[trim($var)] = self::castValue($val, !$quoted);
  384. }
  385. $level = 1;
  386. $var = $val = '';
  387. $composing = false;
  388. $quoted = false;
  389. }
  390. }
  391. return $data;
  392. }
  393. /**
  394. * Try determinate the original type variable of a string
  395. *
  396. * @param string $val string containing possibles variables that can be cast to bool or int
  397. * @param boolean $trim indicate if the value passed should be trimmed after to try cast
  398. * @return mixed returns the value converted to original type if was possible
  399. */
  400. private static function castValue($val, $trim = false)
  401. {
  402. if (is_array($val)) {
  403. foreach ($val as $key => $value) {
  404. $val[$key] = self::castValue($value);
  405. }
  406. } elseif (is_string($val)) {
  407. if ($trim) {
  408. $val = trim($val);
  409. }
  410. $val = stripslashes($val);
  411. $tmp = strtolower($val);
  412. if ($tmp === 'false' || $tmp === 'true') {
  413. $val = $tmp === 'true';
  414. } elseif (is_numeric($val)) {
  415. return $val + 0;
  416. }
  417. unset($tmp);
  418. }
  419. return $val;
  420. }
  421. }