. * */ class OC_API { /** * api actions */ protected static $actions = array(); /** * registers an api call * @param string $method the http method * @param string $url the url to match * @param callable $action the function to run * @param string $app the id of the app registering the call */ public static function register($method, $url, $action, $app, $defaults = array(), $requirements = array()){ $name = strtolower($method).$url; $name = str_replace(array('/', '{', '}'), '_', $name); if(!isset(self::$actions[$name])){ OC::$router->create($name, $url.'.{_format}') ->method($method) ->defaults(array('_format' => 'xml') + $defaults) ->requirements(array('_format' => 'xml|json') + $requirements) ->action('OC_API', 'call'); self::$actions[$name] = array(); } self::$actions[$name][] = array('app' => $app, 'action' => $action); } /** * handles an api call * @param array $parameters */ public static function call($parameters){ $name = $parameters['_route']; // Loop through registered actions foreach(self::$actions[$name] as $action){ $app = $action['app']; if(is_callable($action['action'])){ $responses[] = array('app' => $app, 'response' => call_user_func($action['action'], $parameters)); } else { $responses[] = array('app' => $app, 'response' => 501); } } // Merge the responses $response = self::mergeResponses($responses); // Send the response if(isset($parameters['_format'])){ self::respond($response, $parameters['_format']); } else { self::respond($response); } // logout the user to be stateles OC_User::logout(); } /** * intelligently merges the different responses * @param array $responses * @return array the final merged response */ private static function mergeResponses($responses){ $finalresponse = array(); $numresponses = count($responses); foreach($responses as $response){ if(is_int($response) && empty($finalresponse)){ $finalresponse = $response; continue; } if(is_array($response)){ // Shipped apps win if(OC_App::isShipped($response['app'])){ $finalresponse = array_merge_recursive($finalresponse, $response); } else { $finalresponse = array_merge_recursive($response, $finalresponse); } } } return $finalresponse; } /** * respond to a call * @param int|array $response the response * @param string $format the format xml|json */ private static function respond($response, $format='json'){ if ($format == 'json') { OC_JSON::encodedPrint($response); } else if ($format == 'xml') { header('Content-type: text/xml; charset=UTF-8'); $writer = new XMLWriter(); $writer->openMemory(); $writer->setIndent( true ); $writer->startDocument(); self::toXML($response, $writer); $writer->endDocument(); echo $writer->outputMemory(true); } else { var_dump($format, $response); } } private static function toXML($array, $writer){ foreach($array as $k => $v) { if (is_array($v)) { $writer->startElement($k); self::toXML($v, $writer); $writer->endElement(); } else { $writer->writeElement($k, $v); } } } /** * check if the user is authenticated */ public static function checkLoggedIn(){ // Check OAuth if(!OC_OAuth::isAuthorised()){ OC_Response::setStatus(401); die(); } } }