CRM-14370 - API Kernel - Extract WrapperAdapter
authorTim Otten <totten@civicrm.org>
Sat, 22 Mar 2014 01:37:59 +0000 (18:37 -0700)
committerTim Otten <totten@civicrm.org>
Sun, 6 Apr 2014 04:15:56 +0000 (21:15 -0700)
Civi/API/Kernel.php
Civi/API/Subscriber/WrapperAdapter.php [new file with mode: 0644]
Civi/Core/Container.php

index 89ca67707c76e45b841a5ac5f0464c8f3943d1cc..ed13ecb28052385128bf77a3507dfd9195a5f62f 100644 (file)
@@ -75,15 +75,6 @@ class Kernel {
     $apiRequest['extra'] = $extra;
     $apiRequest['fields'] = NULL;
 
-    /** @var $apiWrappers array<\API_Wrapper> */
-    $apiWrappers = array(
-      \CRM_Utils_API_HTMLInputCoder::singleton(),
-      \CRM_Utils_API_NullOutputCoder::singleton(),
-      \CRM_Utils_API_ReloadOption::singleton(),
-      \CRM_Utils_API_MatchOption::singleton(),
-    );
-    \CRM_Utils_Hook::apiWrappers($apiWrappers, $apiRequest);
-
     try {
       if (!is_array($params)) {
         throw new \API_Exception('Input variable `params` is not an array', 2000);
@@ -101,11 +92,6 @@ class Kernel {
 
       $apiRequest = $this->dispatcher->dispatch(Events::PREPARE, new PrepareEvent(NULL, $apiRequest))->getApiRequest();
 
-      // For input filtering, process $apiWrappers in forward order
-      foreach ($apiWrappers as $apiWrapper) {
-        $apiRequest = $apiWrapper->fromApiInput($apiRequest);
-      }
-
       $function = $apiRequest['function'];
       if ($apiRequest['function'] && $apiRequest['is_generic']) {
         // Unlike normal API implementations, generic implementations require explicit
@@ -120,11 +106,6 @@ class Kernel {
         throw new \API_Exception("API (" . $apiRequest['entity'] . ", " . $apiRequest['action'] . ") does not exist (join the API team and implement it!)");
       }
 
-      // For output filtering, process $apiWrappers in reverse order
-      foreach (array_reverse($apiWrappers) as $apiWrapper) {
-        $result = $apiWrapper->toApiOutput($apiRequest, $result);
-      }
-
       if (\CRM_Utils_Array::value('is_error', $result, 0) == 0) {
         _civicrm_api_call_nested_api($apiRequest['params'], $result, $apiRequest['action'], $apiRequest['entity'], $apiRequest['version']);
       }
diff --git a/Civi/API/Subscriber/WrapperAdapter.php b/Civi/API/Subscriber/WrapperAdapter.php
new file mode 100644 (file)
index 0000000..b94da6c
--- /dev/null
@@ -0,0 +1,89 @@
+<?php
+/*
+ +--------------------------------------------------------------------+
+ | CiviCRM version 4.4                                                |
+ +--------------------------------------------------------------------+
+ | Copyright CiviCRM LLC (c) 2004-2013                                |
+ +--------------------------------------------------------------------+
+ | This file is a part of CiviCRM.                                    |
+ |                                                                    |
+ | CiviCRM is free software; you can copy, modify, and distribute it  |
+ | under the terms of the GNU Affero General Public License           |
+ | Version 3, 19 November 2007 and the CiviCRM Licensing Exception.   |
+ |                                                                    |
+ | CiviCRM is distributed in the hope that it will be useful, but     |
+ | WITHOUT ANY WARRANTY; without even the implied warranty of         |
+ | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.               |
+ | See the GNU Affero General Public License for more details.        |
+ |                                                                    |
+ | You should have received a copy of the GNU Affero General Public   |
+ | License and the CiviCRM Licensing Exception along                  |
+ | with this program; if not, contact CiviCRM LLC                     |
+ | at info[AT]civicrm[DOT]org. If you have questions about the        |
+ | GNU Affero General Public License or the licensing of CiviCRM,     |
+ | see the CiviCRM license FAQ at http://civicrm.org/licensing        |
+ +--------------------------------------------------------------------+
+*/
+
+namespace Civi\API\Subscriber;
+use Civi\API\Events;
+use Symfony\Component\EventDispatcher\EventSubscriberInterface;
+
+/**
+ * This is a wrapper for the legacy "API Wrapper" interface which allows
+ * wrappers to run through the new kernel. It translates from dispatcher events
+ * ('api.prepare', 'api.respond') to wrapper calls ('fromApiInput', 'toApiOutput').
+ */
+class WrapperAdapter implements EventSubscriberInterface {
+
+  public static function getSubscribedEvents() {
+    return array(
+      Events::PREPARE => array('onApiPrepare', Events::W_MIDDLE),
+      Events::RESPOND => array('onApiRespond', Events::W_EARLY),
+    );
+  }
+
+  /**
+   * @var array(\API_Wrapper)
+   */
+  protected $defaults;
+
+  function __construct($defaults = array()) {
+    $this->defaults = $defaults;
+  }
+
+  public function onApiPrepare(\Civi\API\Event\PrepareEvent $event) {
+    $apiRequest = $event->getApiRequest();
+
+    // For input filtering, process $apiWrappers in forward order
+    foreach ($this->getWrappers($apiRequest) as $apiWrapper) {
+      $apiRequest = $apiWrapper->fromApiInput($apiRequest);
+    }
+
+    $event->setApiRequest($apiRequest);
+  }
+
+  public function onApiRespond(\Civi\API\Event\RespondEvent $event) {
+    $apiRequest = $event->getApiRequest();
+    $result = $event->getResponse();
+
+    // For output filtering, process $apiWrappers in reverse order
+    foreach (array_reverse($this->getWrappers($apiRequest)) as $apiWrapper) {
+      $result = $apiWrapper->toApiOutput($apiRequest, $result);
+    }
+
+    $event->setResponse($result);
+  }
+
+  /**
+   * @param array $apiRequest
+   * @return array<\API_Wrapper>
+   */
+  public function getWrappers($apiRequest) {
+    if (!isset($apiRequest['wrappers'])) {
+      $apiRequest['wrappers'] = $this->defaults;
+      \CRM_Utils_Hook::apiWrappers($apiRequest['wrappers'], $apiRequest);
+    }
+    return $apiRequest['wrappers'];
+  }
+}
\ No newline at end of file
index 6c22738f2d0ec71696b90f9954b5736078170c23..c6019c3a29b1a7772afc1a7e7791aea76038952d 100644 (file)
@@ -89,6 +89,12 @@ class Container {
     $dispatcher->addSubscriber(new \Civi\API\Subscriber\TransactionSubscriber());
     $dispatcher->addSubscriber(new \Civi\API\Subscriber\I18nSubscriber());
     $dispatcher->addSubscriber(new \Civi\API\Subscriber\APIv3SchemaAdapter());
+    $dispatcher->addSubscriber(new \Civi\API\Subscriber\WrapperAdapter(array(
+      \CRM_Utils_API_HTMLInputCoder::singleton(),
+      \CRM_Utils_API_NullOutputCoder::singleton(),
+      \CRM_Utils_API_ReloadOption::singleton(),
+      \CRM_Utils_API_MatchOption::singleton(),
+    )));
     $dispatcher->addSubscriber(new \Civi\API\Subscriber\XDebugSubscriber());
     $dispatcher->addListener(\Civi\API\Events::AUTHORIZE, function(\Civi\API\Event\AuthorizeEvent $event) {
       $apiRequest = $event->getApiRequest();