: Limit the result set * - &relationdepth=: Displays links to existing has-one and has-many relationships to a certain depth (Default: 1) * - &fields=: Comma-separated list of fields on the output object (defaults to all database-columns) * * @todo Finish RestfulServer_Item and RestfulServer_List implementation and re-enable $url_handlers * @todo Implement PUT/POST/DELETE for relations * @todo Make SearchContext specification customizeable for each class * @todo Allow for range-searches (e.g. on Created column) * @todo Allow other authentication methods (currently only HTTP BasicAuth) */ class RestfulServer extends Controller { static $url_handlers = array( '$ClassName/$ID/$Relation' => 'handleAction' #'$ClassName/#ID' => 'handleItem', #'$ClassName' => 'handleList', ); protected static $api_base = "api/v1/"; /** * If no extension is given in the request, resolve to this extension * (and subsequently the {@link self::$default_mimetype}. * * @var string */ public static $default_extension = "xml"; /** * If no extension is given, resolve the request to this mimetype. * * @var string */ protected static $default_mimetype = "text/xml"; /* function handleItem($request) { return new RestfulServer_Item(DataObject::get_by_id($request->param("ClassName"), $request->param("ID"))); } function handleList($request) { return new RestfulServer_List(DataObject::get($request->param("ClassName"),"")); } */ /** * This handler acts as the switchboard for the controller. * Since no $Action url-param is set, all requests are sent here. */ function index() { ContentNegotiator::disable(); if(!isset($this->urlParams['ClassName'])) return $this->notFound(); $className = $this->urlParams['ClassName']; $id = (isset($this->urlParams['ID'])) ? $this->urlParams['ID'] : null; $relation = (isset($this->urlParams['Relation'])) ? $this->urlParams['Relation'] : null; // if api access is disabled, don't proceed if(!singleton($className)->stat('api_access')) return $this->permissionFailure(); // authenticate through HTTP BasicAuth $member = $this->authenticate(); // handle different HTTP verbs if($this->request->isGET()) return $this->getHandler($className, $id, $relation); if($this->request->isPOST()) return $this->postHandler($className, $id, $relation); if($this->request->isPUT()) return $this->putHandler($className, $id, $relation); if($this->request->isDELETE()) return $this->deleteHandler($className, $id, $relation); // if no HTTP verb matches, return error return $this->methodNotAllowed(); } /** * Handler for object read. * * The data object will be returned in the following format: * * * Value * ... * * ... * * * * * ... * * * * * * * Access is controlled by two variables: * * - static $api_access must be set. This enables the API on a class by class basis * - $obj->canView() must return true. This lets you implement record-level security * * @todo Access checking * * @param String $className * @param Int $id * @param String $relation * @return String The serialized representation of the requested object(s) - usually XML or JSON. */ protected function getHandler($className, $id, $relation) { $sort = array( 'sort' => $this->request->getVar('sort'), 'dir' => $this->request->getVar('dir') ); $limit = array( 'start' => $this->request->getVar('start'), 'limit' => $this->request->getVar('limit') ); $responseFormatter = $this->getResponseDataFormatter(); if(!$responseFormatter) return $this->unsupportedMediaType(); if($id) { $obj = DataObject::get_by_id($className, $id); if(!$obj) return $this->notFound(); if(!$obj->canView()) return $this->permissionFailure(); if($relation) { if($relationClass = $obj->many_many($relation)) { $query = $obj->getManyManyComponentsQuery($relation); } elseif($relationClass = $obj->has_many($relation)) { $query = $obj->getComponentsQuery($relation); } elseif($relationClass = $obj->has_one($relation)) { $query = null; } elseif($obj->hasMethod("{$relation}Query")) { // @todo HACK Switch to ComponentSet->getQuery() once we implement it (and lazy loading) $query = $obj->{"{$relation}Query"}(null, $sort, null, $limit); $relationClass = $obj->{"{$relation}Class"}(); } else { return $this->notFound(); } // get all results $obj = $this->search($relationClass, $this->request->getVars(), $sort, $limit, $query); if(!$obj) $obj = new DataObjectSet(); } } else { $obj = $this->search($className, $this->request->getVars(), $sort, $limit); // show empty serialized result when no records are present if(!$obj) $obj = new DataObjectSet(); } $this->getResponse()->addHeader('Content-Type', $responseFormatter->getOutputContentType()); if($obj instanceof DataObjectSet) return $responseFormatter->convertDataObjectSet($obj); else return $responseFormatter->convertDataObject($obj); } /** * Uses the default {@link SearchContext} specified through * {@link DataObject::getDefaultSearchContext()} to augument * an existing query object (mostly a component query from {@link DataObject}) * with search clauses. * * @todo Allow specifying of different searchcontext getters on model-by-model basis * * @param string $className * @param array $params * @return DataObjectSet */ protected function search($className, $params = null, $sort = null, $limit = null, $existingQuery = null) { if(singleton($className)->hasMethod('getRestfulSearchContext')) { $searchContext = singleton($className)->{'getRestfulSearchContext'}(); } else { $searchContext = singleton($className)->getDefaultSearchContext(); } $query = $searchContext->getQuery($params, $sort, $limit, $existingQuery); return singleton($className)->buildDataObjectSet($query->execute()); } /** * Returns a dataformatter instance based on the request * extension or mimetype. Falls back to {@link self::$default_extension}. * * @param boolean $includeAcceptHeader Determines wether to inspect and prioritize any HTTP Accept headers * @return DataFormatter */ protected function getDataFormatter($includeAcceptHeader = false) { $extension = $this->request->getExtension(); $contentType = $this->request->getHeader('Content-Type'); $accept = $this->request->getHeader('Accept'); // get formatter if(!empty($extension)) { $formatter = DataFormatter::for_extension($extension); }elseif($includeAcceptHeader && !empty($accept) && $accept != '*/*') { $formatter = DataFormatter::for_mimetypes(explode(',',$accept)); } elseif(!empty($contentType)) { $formatter = DataFormatter::for_mimetype($contentType); } else { $formatter = DataFormatter::for_extension(self::$default_extension); } // set custom fields if($customFields = $this->request->getVar('fields')) $formatter->setCustomFields(explode(',',$customFields)); // set relation depth $relationDepth = $this->request->getVar('relationdepth'); if(is_numeric($relationDepth)) $formatter->relationDepth = (int)$relationDepth; return $formatter; } protected function getRequestDataFormatter() { return $this->getDataFormatter(false); } protected function getResponseDataFormatter() { return $this->getDataFormatter(true); } /** * Handler for object delete */ protected function deleteHandler($className, $id) { $obj = DataObject::get_by_id($className, $id); if(!$obj) return $this->notFound(); if(!$obj->canDelete()) return $this->permissionFailure(); $obj->delete(); $this->getResponse()->setStatusCode(204); // No Content return true; } /** * Handler for object write */ protected function putHandler($className, $id) { $obj = DataObject::get_by_id($className, $id); if(!$obj) return $this->notFound(); if(!$obj->canEdit()) return $this->permissionFailure(); $reqFormatter = $this->getRequestDataFormatter(); if(!$reqFormatter) return $this->unsupportedMediaType(); $responseFormatter = $this->getResponseDataFormatter(); if(!$responseFormatter) return $this->unsupportedMediaType(); $obj = $this->updateDataObject($obj, $reqFormatter); $this->getResponse()->setStatusCode(200); // Success $this->getResponse()->addHeader('Content-Type', $responseFormatter->getOutputContentType()); $objHref = Director::absoluteURL(self::$api_base . "$obj->class/$obj->ID"); $this->getResponse()->addHeader('Location', $objHref); return $responseFormatter->convertDataObject($obj); } /** * Handler for object append / method call. * * @todo Posting to an existing URL (without a relation) * current resolves in creatig a new element, * rather than a "Conflict" message. */ protected function postHandler($className, $id) { if(!singleton($className)->canCreate()) return $this->permissionFailure(); $obj = new $className(); $reqFormatter = $this->getRequestDataFormatter(); if(!$reqFormatter) return $this->unsupportedMediaType(); $responseFormatter = $this->getResponseDataFormatter(); $obj = $this->updateDataObject($obj, $reqFormatter); $this->getResponse()->setStatusCode(201); // Created $this->getResponse()->addHeader('Content-Type', $responseFormatter->getOutputContentType()); $objHref = Director::absoluteURL(self::$api_base . "$obj->class/$obj->ID"); $this->getResponse()->addHeader('Location', $objHref); return $responseFormatter->convertDataObject($obj); } /** * Converts either the given HTTP Body into an array * (based on the DataFormatter instance), or returns * the POST variables. * Automatically filters out certain critical fields * that shouldn't be set by the client (e.g. ID). * * @param DataObject $obj * @param DataFormatter $formatter * @return DataObject The passed object */ protected function updateDataObject($obj, $formatter) { // if neither an http body nor POST data is present, return error $body = $this->request->getBody(); if(!$body && !$this->request->postVars()) { $this->getResponse()->setStatusCode(204); // No Content return 'No Content'; } if(!empty($body)) { $data = $formatter->convertStringToArray($body); } else { // assume application/x-www-form-urlencoded which is automatically parsed by PHP $data = $this->request->postVars(); } // @todo Disallow editing of certain keys in database $data = array_diff_key($data, array('ID','Created')); $obj->update($data); $obj->write(); return $obj; } protected function permissionFailure() { // return a 401 $this->getResponse()->setStatusCode(403); return "You don't have access to this item through the API."; } protected function notFound() { // return a 404 $this->getResponse()->setStatusCode(404); return "That object wasn't found"; } protected function methodNotAllowed() { $this->getResponse()->setStatusCode(405); return "Method Not Allowed"; } protected function unsupportedMediaType() { $this->response->setStatusCode(415); // Unsupported Media Type return "Unsupported Media Type"; } protected function authenticate() { if(!isset($_SERVER['PHP_AUTH_USER']) || !isset($_SERVER['PHP_AUTH_PW'])) return false; if($member = Member::currentMember()) return $member; $member = MemberAuthenticator::authenticate(array( 'Email' => $_SERVER['PHP_AUTH_USER'], 'Password' => $_SERVER['PHP_AUTH_PW'], ), null); if($member) { $member->LogIn(false); return $member; } else { return false; } } } /** * Restful server handler for a DataObjectSet */ class RestfulServer_List { static $url_handlers = array( '#ID' => 'handleItem', ); function __construct($list) { $this->list = $list; } function handleItem($request) { return new RestulServer_Item($this->list->getById($request->param('ID'))); } } /** * Restful server handler for a single DataObject */ class RestfulServer_Item { static $url_handlers = array( '$Relation' => 'handleRelation', ); function __construct($item) { $this->item = $item; } function handleRelation($request) { $funcName = $request('Relation'); $relation = $this->item->$funcName(); if($relation instanceof DataObjectSet) return new RestfulServer_List($relation); else return new RestfulServer_Item($relation); } }