Newer
Older
<?php
/**
* @author Jaap Jansma <jaap.jansma@civicoop.org>
* @license AGPL-3.0
*/
namespace Civi\DataProcessor\Output;
use Civi\API\Event\ResolveEvent;
use Civi\API\Event\RespondEvent;
use Civi\API\Events;
use Civi\API\Provider\ProviderInterface as API_ProviderInterface;
use Civi\DataProcessor\ProcessorType\AbstractProcessorType;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
class Api implements OutputInterface, API_ProviderInterface, EventSubscriberInterface{
public function __construct() {
}

jaapjansma
committed
/**
* Returns true when this filter has additional configuration
*
* @return bool
*/
public function hasConfiguration() {
return true;
}
/**
* When this filter type has additional configuration you can add
* the fields on the form with this function.

jaapjansma
committed
*
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
* @param \CRM_Core_Form $form
* @param array $filter
*/
public function buildConfigurationForm(\CRM_Core_Form $form, $output=array()) {
$form->add('select','permission', E::ts('Permission'), \CRM_Core_Permission::basicPermissions(), true, array(
'style' => 'min-width:250px',
'class' => 'crm-select2 huge',
'placeholder' => E::ts('- select -'),
));
$form->add('text', 'api_entity', E::ts('API Entity'), true);
$form->add('text', 'api_action', E::ts('API Action Name'), true);
$form->add('text', 'api_count_action', E::ts('API GetCount Action Name'), true);
if ($output) {
$defaults['permission'] = $output['permission'];
$defaults['api_entity'] = $output['api_entity'];
$defaults['api_action'] = $output['api_action'];
$defaults['api_count_action'] = $output['api_count_action'];
} else {
$defaults['permission'] = 'access CiviCRM';
}
$form->setDefaults($defaults);
}
/**
* When this filter type has configuration specify the template file name
* for the configuration form.
*
* @return false|string
*/
public function getConfigurationTemplateFileName() {
return "CRM/Dataprocessor/Form/Output/API.tpl";
}
/**
* Process the submitted values and create a configuration array
*
* @param $submittedValues
* @param array $output
* @return array

jaapjansma
committed
*/
public function processConfiguration($submittedValues, &$output) {
$output['permission'] = $submittedValues['permission'];
$output['api_entity'] = $submittedValues['api_entity'];
$output['api_action'] = $submittedValues['api_action'];
$output['api_count_action'] = $submittedValues['api_count_action'];
return array();

jaapjansma
committed
}
/**
* @return array
*/
public static function getSubscribedEvents() {
// Some remarks on the solution to implement the getFields method.
// We would like to have implemented it as a normal api call. Meaning
// it would been processed in the invoke method.
//
// However the reflection provider has a stop propegation so we cannot use
// getfields here unles we are earlier then the reflection provider.
// We should then use a weight lower than Events::W_EARLY and do a
// stop propegation in our event. But setting an early flag is not a neat way to do stuff.
// So instead we use the Respond event and check in the respond event whether the action is getFields and
// if so do our getfields stuff there.
return array(
Events::RESOLVE => array('onApiResolve'),
Events::RESPOND => array('onGetFieldsRepsonse'), // we use this method to add our field definition to the getFields action.
);
}
public function onApiResolve(ResolveEvent $event) {

jaapjansma
committed
$entities = $this->getEntityNames();

jaapjansma
committed
foreach($entities as $entity) {
if (strtolower($apiRequest['entity']) == strtolower($entity)) {
$event->setApiProvider($this);
}
/**
* Event listener on the ResponddEvent to handle the getfields actions.
* So the fields defined by the user are availble in the api explorer for example.
*/
public function onGetFieldsRepsonse(RespondEvent $event) {
$apiRequest = $event->getApiRequest();
$params = $apiRequest['params'];
$result = $event->getResponse();
// First check whether the entity is dataprocessorapi and the action is getfields.
// If not return this function.

jaapjansma
committed
if (strtolower($apiRequest['action']) != 'getfields') {
return;
}
// Now check whether the action param is set. With the action param we can find the data processor.
if (isset($params['action'])) {
try {

jaapjansma
committed
// Find the data processor
$dataProcessorIdSql = "
SELECT *
FROM civicrm_data_processor_output o
INNER JOIN civicrm_data_processor p ON o.data_processor_id = p.id
WHERE p.is_active = 1 AND LOWER(o.api_entity) = LOWER(%1)
AND (LOWER(o.api_action) = LOWER(%2) OR LOWER(o.api_count_action) = LOWER(%2))
";
$dataProcessorIdParams[1] = array($apiRequest['entity'], 'String');
$dataProcessorIdParams[2] = array($params['action'], 'String');
$dao = \CRM_Core_DAO::executeQuery($dataProcessorIdSql, $dataProcessorIdParams);
if (!$dao->fetch()) {
throw new \API_Exception("Could not find a data processor");
}
$dataProcessor = civicrm_api3('DataProcessor', 'getsingle', array('id' => $dao->data_processor_id));
$dataProcessorClass = \CRM_Dataprocessor_BAO_DataProcessor::dataProcessorToClass($dataProcessor);

jaapjansma
committed
foreach ($dataProcessorClass->getDataFlow()->getOutputFieldHandlers() as $outputFieldHandler) {
$fieldSpec = $outputFieldHandler->getOutputFieldSpecification();
$type = \CRM_Utils_Type::T_STRING;
if (isset($types[$fieldSpec->type])) {
$type = $types[$fieldSpec->type];
}
$field = [
'name' => $fieldSpec->alias,
'title' => $fieldSpec->title,
'description' => '',
];
if ($fieldSpec->getOptions()) {
$field['options'] = $fieldSpec->getOptions();
}
$result['values'][$fieldSpec->alias] = $field;
}
foreach($dataProcessorClass->getFilterHandlers() as $filterHandler) {
$type = \CRM_Utils_Type::T_STRING;
if (isset($types[$fieldSpec->type])) {
$type = $types[$fieldSpec->type];
}
$field = [
'name' => $fieldSpec->alias,
'title' => $fieldSpec->title,
'description' => '',
'api.required' => $filterHandler->isRequired(),
'api.aliases' => [],
'api.filter' => TRUE,
'api.return' => isset($result['values'][$fieldSpec->alias]) ? $result['values'][$fieldSpec->alias]['api.return'] : FALSE,
];
if ($fieldSpec->getOptions()) {
$field['options'] = $fieldSpec->getOptions();
}
if (!isset($result['values'][$fieldSpec->alias])) {
$result['values'][$fieldSpec->alias] = $field;
} else {
$result['values'][$fieldSpec->alias] = array_merge($result['values'][$fieldSpec->alias], $field);
}
}
} catch(\Exception $e) {
// Do nothing.
}
$result['count'] = count($result['values']);
$event->setResponse($result);
}
}
/**
* @param array $apiRequest
* The full description of the API request.
* @return array
* structured response data (per civicrm_api3_create_success)
* @see civicrm_api3_create_success

jaapjansma
committed
* @throws \Exception

jaapjansma
committed
$isCountAction = FALSE;
$dataProcessorIdSql = "
SELECT *
FROM civicrm_data_processor_output o
INNER JOIN civicrm_data_processor p ON o.data_processor_id = p.id
WHERE p.is_active = 1 AND LOWER(o.api_entity) = LOWER(%1)
AND (LOWER(o.api_action) = LOWER(%2) OR LOWER(o.api_count_action) = LOWER(%2))
";
$dataProcessorIdParams[1] = array($apiRequest['entity'], 'String');
$dataProcessorIdParams[2] = array($apiRequest['action'], 'String');
$dao = \CRM_Core_DAO::executeQuery($dataProcessorIdSql, $dataProcessorIdParams);
if (!$dao->fetch()) {
throw new \API_Exception("Could not find a data processor");

jaapjansma
committed
if (strtolower($dao->api_count_action) == $apiRequest['action']) {
$isCountAction = TRUE;
$dataProcessor = civicrm_api3('DataProcessor', 'getsingle', array('id' => $dao->data_processor_id));
$dataProcessorClass = \CRM_Dataprocessor_BAO_DataProcessor::dataProcessorToClass($dataProcessor);
foreach($dataProcessorClass->getFilterHandlers() as $filter) {
$filterSpec = $filter->getFieldSpecification();
if ($filter->isRequired() && !isset($params[$filterSpec->alias])) {
throw new \API_Exception('Field '.$filterSpec->alias.' is required');
}
if (isset($params[$filterSpec->alias])) {
if (!is_array($params[$filterSpec->alias])) {
$filterParams = [
'op' => '=',
'value' => $params[$filterSpec->alias],
];
} else {
$value = reset($params[$filterSpec->alias]);
$filterParams = [
'op' => key($params[$filterSpec->alias]),
'value' => $value,
];
}
$filter->setFilter($filterParams);
}
}
$count = $dataProcessorClass->getDataFlow()->recordCount();
} else {
$options = _civicrm_api3_get_options_from_params($apiRequest['params']);
if (isset($options['limit']) && $options['limit'] > 0) {
$dataProcessorClass->getDataFlow()->setLimit($options['limit']);
$dataProcessorClass->getDataFlow()->setOffset($options['offset']);
if (isset($options['sort'])) {
$sort = explode(', ', $options['sort']);
foreach ($sort as $index => &$sortString) {
// Get sort field and direction
list($sortField, $dir) = array_pad(explode(' ', $sortString), 2, 'ASC');
$dataProcessorClass->getDataFlow()->addSort($sortField, $dir);
$records = $dataProcessorClass->getDataFlow()->allRecords();
$values = array();
foreach($records as $idx => $record) {
foreach($record as $fieldname => $field) {
$values[$idx][$fieldname] = $field->formattedValue;
}
}
$return = array(
'values' => $values,
'count' => count($values),
'is_error' => 0,
);
if (isset($apiRequest['params']['debug']) && $apiRequest['params']['debug']) {
$return['debug_info'] = $dataProcessorClass->getDataFlow()->getDebugInformation();
}
}
/**
* @param int $version
* API version.
* @return array<string>
*/

jaapjansma
committed
$dao = \CRM_Core_DAO::executeQuery("
SELECT DISTINCT o.api_entity
FROM civicrm_data_processor_output o
INNER JOIN civicrm_data_processor p ON o.data_processor_id = p.id
WHERE p.is_active = 1
");
$entities = array();
while($dao->fetch()) {
if ($dao->api_entity) {
$entities[] = $dao->api_entity;
}
}
return $entities;
}
/**
* @param int $version
* API version.
* @param string $entity
* API entity.
* @return array<string>
*/
public function getActionNames($version, $entity) {

jaapjansma
committed
$actions = array();
$dao = \CRM_Core_DAO::executeQuery("
SELECT api_action, api_count_action
FROM civicrm_data_processor_output o
INNER JOIN civicrm_data_processor p ON o.data_processor_id = p.id
WHERE p.is_active = 1 AND LOWER(o.api_entity) = LOWER(%1)",
array(1=>array($entity, 'String'))
);
while($dao->fetch()) {
$actions[] = $dao->api_action;
$actions[] = $dao->api_count_action;

jaapjansma
committed