toArray(); } if (!is_array($options)) { throw new Zend_Cloud_DocumentService_Exception('Invalid options provided to constructor'); } $this->_simpleDb = new Zend_Service_Amazon_SimpleDb( $options[self::AWS_ACCESS_KEY], $options[self::AWS_SECRET_KEY] ); if (isset($options[self::HTTP_ADAPTER])) { $this->_sqs->getHttpClient()->setAdapter($options[self::HTTP_ADAPTER]); } if (isset($options[self::DOCUMENT_CLASS])) { $this->setDocumentClass($options[self::DOCUMENT_CLASS]); } if (isset($options[self::DOCUMENTSET_CLASS])) { $this->setDocumentSetClass($options[self::DOCUMENTSET_CLASS]); } if (isset($options[self::QUERY_CLASS])) { $this->setQueryClass($options[self::QUERY_CLASS]); } } /** * Create collection. * * @param string $name * @param array $options * @return void */ public function createCollection($name, $options = null) { try { $this->_simpleDb->createDomain($name); } catch(Zend_Service_Amazon_Exception $e) { throw new Zend_Cloud_DocumentService_Exception('Error on domain creation: '.$e->getMessage(), $e->getCode(), $e); } } /** * Delete collection. * * @param string $name * @param array $options * @return void */ public function deleteCollection($name, $options = null) { try { $this->_simpleDb->deleteDomain($name); } catch(Zend_Service_Amazon_Exception $e) { throw new Zend_Cloud_DocumentService_Exception('Error on collection deletion: '.$e->getMessage(), $e->getCode(), $e); } } /** * List collections. * * @param array $options * @return array */ public function listCollections($options = null) { try { // TODO package this in Pages $domains = $this->_simpleDb->listDomains()->getData(); } catch(Zend_Service_Amazon_Exception $e) { throw new Zend_Cloud_DocumentService_Exception('Error on collection deletion: '.$e->getMessage(), $e->getCode(), $e); } return $domains; } /** * List documents * * Returns a key/value array of document names to document objects. * * @param string $collectionName Name of collection for which to list documents * @param array|null $options * @return Zend_Cloud_DocumentService_DocumentSet */ public function listDocuments($collectionName, array $options = null) { $query = $this->select('*')->from($collectionName); $items = $this->query($collectionName, $query, $options); return $items; } /** * Insert document * * @param string $collectionName Collection into which to insert document * @param array|Zend_Cloud_DocumentService_Document $document * @param array $options * @return void */ public function insertDocument($collectionName, $document, $options = null) { if (is_array($document)) { $document = $this->_getDocumentFromArray($document); } if (!$document instanceof Zend_Cloud_DocumentService_Document) { throw new Zend_Cloud_DocumentService_Exception('Invalid document supplied'); } try { $this->_simpleDb->putAttributes( $collectionName, $document->getID(), $this->_makeAttributes($document->getID(), $document->getFields()) ); } catch(Zend_Service_Amazon_Exception $e) { throw new Zend_Cloud_DocumentService_Exception('Error on document insertion: '.$e->getMessage(), $e->getCode(), $e); } } /** * Replace an existing document with a new version * * @param string $collectionName * @param array|Zend_Cloud_DocumentService_Document $document * @param array $options * @return void */ public function replaceDocument($collectionName, $document, $options = null) { if (is_array($document)) { $document = $this->_getDocumentFromArray($document); } if (!$document instanceof Zend_Cloud_DocumentService_Document) { throw new Zend_Cloud_DocumentService_Exception('Invalid document supplied'); } // Delete document first, then insert. PutAttributes always keeps any // fields not referenced in the payload, but present in the document $documentId = $document->getId(); $fields = $document->getFields(); $docClass = get_class($document); $this->deleteDocument($collectionName, $document, $options); $document = new $docClass($fields, $documentId); $this->insertDocument($collectionName, $document); } /** * Update document. The new document replaces the existing document. * * Option 'merge' specifies to add all attributes (if true) or * specific attributes ("attr" => true) instead of replacing them. * By default, attributes are replaced. * * @param string $collectionName * @param mixed|Zend_Cloud_DocumentService_Document $documentId Document ID, adapter-dependent * @param array|Zend_Cloud_DocumentService_Document $fieldset Set of fields to update * @param array $options * @return boolean */ public function updateDocument($collectionName, $documentId, $fieldset = null, $options = null) { if (null === $fieldset && $documentId instanceof Zend_Cloud_DocumentService_Document) { $fieldset = $documentId->getFields(); if (empty($documentId)) { $documentId = $documentId->getId(); } } elseif ($fieldset instanceof Zend_Cloud_DocumentService_Document) { if (empty($documentId)) { $documentId = $fieldset->getId(); } $fieldset = $fieldset->getFields(); } $replace = array(); if (empty($options[self::MERGE_OPTION])) { // no merge option - we replace all foreach ($fieldset as $key => $value) { $replace[$key] = true; } } elseif (is_array($options[self::MERGE_OPTION])) { foreach ($fieldset as $key => $value) { if (empty($options[self::MERGE_OPTION][$key])) { // if there's merge key, we add it, otherwise we replace it $replace[$key] = true; } } } // otherwise $replace is empty - all is merged try { $this->_simpleDb->putAttributes( $collectionName, $documentId, $this->_makeAttributes($documentId, $fieldset), $replace ); } catch(Zend_Service_Amazon_Exception $e) { throw new Zend_Cloud_DocumentService_Exception('Error on document update: '.$e->getMessage(), $e->getCode(), $e); } return true; } /** * Delete document. * * @param string $collectionName Collection from which to delete document * @param mixed $document Document ID or Document object. * @param array $options * @return boolean */ public function deleteDocument($collectionName, $document, $options = null) { if ($document instanceof Zend_Cloud_DocumentService_Document) { $document = $document->getId(); } try { $this->_simpleDb->deleteAttributes($collectionName, $document); } catch(Zend_Service_Amazon_Exception $e) { throw new Zend_Cloud_DocumentService_Exception('Error on document deletion: '.$e->getMessage(), $e->getCode(), $e); } return true; } /** * Fetch single document by ID * * @param string $collectionName Collection name * @param mixed $documentId Document ID, adapter-dependent * @param array $options * @return Zend_Cloud_DocumentService_Document */ public function fetchDocument($collectionName, $documentId, $options = null) { try { $attributes = $this->_simpleDb->getAttributes($collectionName, $documentId); if ($attributes == false || count($attributes) == 0) { return false; } return $this->_resolveAttributes($attributes, true); } catch(Zend_Service_Amazon_Exception $e) { throw new Zend_Cloud_DocumentService_Exception('Error on fetching document: '.$e->getMessage(), $e->getCode(), $e); } } /** * Query for documents stored in the document service. If a string is passed in * $query, the query string will be passed directly to the service. * * @param string $collectionName Collection name * @param string $query * @param array $options * @return array Zend_Cloud_DocumentService_DocumentSet */ public function query($collectionName, $query, $options = null) { $returnDocs = isset($options[self::RETURN_DOCUMENTS]) ? (bool) $options[self::RETURN_DOCUMENTS] : true; try { if ($query instanceof Zend_Cloud_DocumentService_Adapter_SimpleDb_Query) { $query = $query->assemble($collectionName); } $result = $this->_simpleDb->select($query); } catch(Zend_Service_Amazon_Exception $e) { throw new Zend_Cloud_DocumentService_Exception('Error on document query: '.$e->getMessage(), $e->getCode(), $e); } return $this->_getDocumentSetFromResultSet($result, $returnDocs); } /** * Create query statement * * @param string $fields * @return Zend_Cloud_DocumentService_Adapter_SimpleDb_Query */ public function select($fields = null) { $queryClass = $this->getQueryClass(); if (!class_exists($queryClass)) { require_once 'Zend/Loader.php'; Zend_Loader::loadClass($queryClass); } $query = new $queryClass($this); $defaultClass = self::DEFAULT_QUERY_CLASS; if (!$query instanceof $defaultClass) { throw new Zend_Cloud_DocumentService_Exception('Query class must extend ' . self::DEFAULT_QUERY_CLASS); } $query->select($fields); return $query; } /** * Get the concrete service client * * @return Zend_Service_Amazon_SimpleDb */ public function getClient() { return $this->_simpleDb; } /** * Convert array of key-value pairs to array of Amazon attributes * * @param string $name * @param array $attributes * @return array */ protected function _makeAttributes($name, $attributes) { $result = array(); foreach ($attributes as $key => $attr) { $result[] = new Zend_Service_Amazon_SimpleDb_Attribute($name, $key, $attr); } return $result; } /** * Convert array of Amazon attributes to array of key-value pairs * * @param array $attributes * @return array */ protected function _resolveAttributes($attributes, $returnDocument = false) { $result = array(); foreach ($attributes as $attr) { $value = $attr->getValues(); if (count($value) == 0) { $value = null; } elseif (count($value) == 1) { $value = $value[0]; } $result[$attr->getName()] = $value; } // Return as document object? if ($returnDocument) { $documentClass = $this->getDocumentClass(); return new $documentClass($result, $attr->getItemName()); } return $result; } /** * Create suitable document from array of fields * * @param array $document * @return Zend_Cloud_DocumentService_Document */ protected function _getDocumentFromArray($document) { if (!isset($document[Zend_Cloud_DocumentService_Document::KEY_FIELD])) { if (isset($document[self::ITEM_NAME])) { $key = $document[self::ITEM_NAME]; unset($document[self::ITEM_NAME]); } else { throw new Zend_Cloud_DocumentService_Exception('Fields array should contain the key field '.Zend_Cloud_DocumentService_Document::KEY_FIELD); } } else { $key = $document[Zend_Cloud_DocumentService_Document::KEY_FIELD]; unset($document[Zend_Cloud_DocumentService_Document::KEY_FIELD]); } $documentClass = $this->getDocumentClass(); return new $documentClass($document, $key); } /** * Create a DocumentSet from a SimpleDb resultset * * @param Zend_Service_Amazon_SimpleDb_Page $resultSet * @param bool $returnDocs * @return Zend_Cloud_DocumentService_DocumentSet */ protected function _getDocumentSetFromResultSet(Zend_Service_Amazon_SimpleDb_Page $resultSet, $returnDocs = true) { $docs = array(); foreach ($resultSet->getData() as $item) { $docs[] = $this->_resolveAttributes($item, $returnDocs); } $setClass = $this->getDocumentSetClass(); return new $setClass($docs); } }