_actionsToGet = $this->_itemsToGet('name'); $entityReflection = new \ReflectionClass('\Civi\Api4\\' . $this->_entityName); foreach ($entityReflection->getMethods(\ReflectionMethod::IS_STATIC | \ReflectionMethod::IS_PUBLIC) as $method) { $actionName = $method->getName(); if ($actionName != 'permissions' && $actionName != 'getInfo' && $actionName[0] != '_') { $this->loadAction($actionName, $method); } } if (!$this->_actionsToGet || count($this->_actionsToGet) > count($this->_actions)) { // Search for entity-specific actions in extensions foreach (\CRM_Extension_System::singleton()->getMapper()->getActiveModuleFiles() as $ext) { $dir = \CRM_Utils_File::addTrailingSlash(dirname($ext['filePath'])); $this->scanDir($dir . 'Civi/Api4/Action/' . $this->_entityName); } // Search for entity-specific actions in core $this->scanDir(\CRM_Utils_File::addTrailingSlash(__DIR__) . $this->_entityName); } ksort($this->_actions); return $this->_actions; } /** * @param $dir */ private function scanDir($dir) { if (is_dir($dir)) { foreach (glob("$dir/*.php") as $file) { $matches = []; preg_match('/(\w*)\.php$/', $file, $matches); $actionName = array_pop($matches); $actionClass = new \ReflectionClass('\\Civi\\Api4\\Action\\' . $this->_entityName . '\\' . $actionName); if ($actionClass->isInstantiable() && $actionClass->isSubclassOf('\\Civi\\Api4\\Generic\\AbstractAction')) { $this->loadAction(lcfirst($actionName)); } } } } /** * @param $actionName * @param \ReflectionMethod $method */ private function loadAction($actionName, $method = NULL) { try { if (!isset($this->_actions[$actionName]) && (!$this->_actionsToGet || in_array($actionName, $this->_actionsToGet))) { $action = \Civi\API\Request::create($this->getEntityName(), $actionName, ['version' => 4]); if (is_object($action) && (!$this->checkPermissions || $action->isAuthorized())) { $this->_actions[$actionName] = ['name' => $actionName]; if ($this->_isFieldSelected('description', 'comment', 'see')) { $vars = ['entity' => $this->getEntityName(), 'action' => $actionName]; // Docblock from action class $actionDocs = ReflectionUtils::getCodeDocs($action->reflect(), NULL, $vars); unset($actionDocs['method']); // Docblock from action factory function in entity class. This takes precedence since most action classes are generic. if ($method) { $methodDocs = ReflectionUtils::getCodeDocs($method, 'Method', $vars); // Allow method doc to inherit class doc if (strpos($method->getDocComment(), '@inheritDoc') !== FALSE && !empty($methodDocs['comment']) && !empty($actionDocs['comment'])) { $methodDocs['comment'] .= "\n\n" . $actionDocs['comment']; } $actionDocs = array_filter($methodDocs) + $actionDocs; } $this->_actions[$actionName] += $actionDocs; } if ($this->_isFieldSelected('params')) { $this->_actions[$actionName]['params'] = $action->getParamInfo(); // Language param is only relevant on multilingual sites $languageLimit = (array) \Civi::settings()->get('languageLimit'); if (count($languageLimit) < 2) { unset($this->_actions[$actionName]['params']['language']); } elseif (isset($this->_actions[$actionName]['params']['language'])) { $this->_actions[$actionName]['params']['language']['options'] = array_keys($languageLimit); } } } } } catch (NotImplementedException $e) { } } public function fields() { return [ [ 'name' => 'name', 'description' => 'Action name', ], [ 'name' => 'description', 'description' => 'Description from docblock', ], [ 'name' => 'comment', 'description' => 'Comments from docblock', ], [ 'name' => 'see', 'data_type' => 'Array', 'description' => 'Any @see annotations from docblock', ], [ 'name' => 'params', 'description' => 'List of all accepted parameters', 'data_type' => 'Array', ], ]; } }