1
0
Fork 0
mirror of synced 2024-06-02 19:04:49 +12:00
appwrite/app/controllers/api/database.php

932 lines
43 KiB
PHP
Raw Normal View History

2019-05-09 18:54:39 +12:00
<?php
use Utopia\App;
use Utopia\Exception;
use Utopia\Validator\Boolean;
use Utopia\Validator\Numeric;
2019-05-09 18:54:39 +12:00
use Utopia\Validator\Range;
use Utopia\Validator\WhiteList;
use Utopia\Validator\Text;
use Utopia\Validator\ArrayList;
use Utopia\Validator\JSON;
use Appwrite\Database\Database;
use Appwrite\Database\Document;
use Appwrite\Database\Validator\UID;
use Appwrite\Database\Validator\Key;
use Appwrite\Database\Validator\Structure;
use Appwrite\Database\Validator\Collection;
use Appwrite\Database\Validator\Authorization;
use Appwrite\Database\Exception\Authorization as AuthorizationException;
use Appwrite\Database\Exception\Structure as StructureException;
2020-10-31 08:53:27 +13:00
use Appwrite\Utopia\Response;
2021-06-09 08:12:14 +12:00
use Utopia\Database\Database as Database2;
use Utopia\Database\Document as Document2;
use Utopia\Database\Query;
use Utopia\Database\Validator\Authorization as Authorization2;
2021-06-12 06:07:05 +12:00
use function Amp\Promise\wait;
2020-06-29 05:31:21 +12:00
App::post('/v1/database/collections')
2019-05-09 18:54:39 +12:00
->desc('Create Collection')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
2020-10-31 08:53:27 +13:00
->label('event', 'database.collections.create')
->label('scope', 'collections.write')
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.method', 'createCollection')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/database/create-collection.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_COLLECTION)
->param('id', '', new Text(255), 'Collection ID. Max length: 255 chars.')
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('dbForExternal')
2020-12-27 04:05:04 +13:00
->inject('audits')
->action(function ($id, $response, $dbForExternal, $audits) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal*/
2020-07-06 02:19:59 +12:00
/** @var Appwrite\Event\Event $audits */
$collection = $dbForExternal->createCollection($id);
$collection = $dbForExternal->getCollection($id);
2019-05-09 18:54:39 +12:00
2020-07-06 02:19:59 +12:00
$audits
2020-06-30 09:43:34 +12:00
->setParam('event', 'database.collections.create')
->setParam('resource', 'database/collection/'.$collection->getId())
->setParam('data', $collection->getArrayCopy())
2020-06-30 09:43:34 +12:00
;
2021-05-27 22:09:14 +12:00
$response->setStatusCode(Response::STATUS_CODE_CREATED);
$response->dynamic2($collection, Response::MODEL_COLLECTION);
2020-12-27 04:05:04 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::get('/v1/database/collections')
2020-02-01 11:34:07 +13:00
->desc('List Collections')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
2020-02-01 11:34:07 +13:00
->label('scope', 'collections.read')
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2020-02-01 11:34:07 +13:00
->label('sdk.method', 'listCollections')
->label('sdk.description', '/docs/references/database/list-collections.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_COLLECTION_LIST)
2020-09-11 02:40:14 +12:00
->param('limit', 25, new Range(0, 100), 'Results limit value. By default will return maximum 25 results. Maximum of 100 results allowed per request.', true)
->param('offset', 0, new Range(0, 40000), 'Results offset. The default value is 0. Use this param to manage pagination.', true)
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('dbForExternal')
->action(function ($limit, $offset, $response, $dbForExternal) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2020-06-30 09:43:34 +12:00
$collections = $dbForExternal->listCollections($limit, $offset);
2020-06-30 09:43:34 +12:00
$response->dynamic2(new Document2([
'collections' => $collections,
'sum' => \count($collections),
2020-10-31 08:53:27 +13:00
]), Response::MODEL_COLLECTION_LIST);
2020-12-27 04:05:04 +13:00
});
2020-02-01 11:34:07 +13:00
2020-06-29 05:31:21 +12:00
App::get('/v1/database/collections/:collectionId')
2020-02-01 11:34:07 +13:00
->desc('Get Collection')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
2020-02-01 11:34:07 +13:00
->label('scope', 'collections.read')
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2020-02-01 11:34:07 +13:00
->label('sdk.method', 'getCollection')
->label('sdk.description', '/docs/references/database/get-collection.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_COLLECTION)
2020-09-11 02:40:14 +12:00
->param('collectionId', '', new UID(), 'Collection unique ID.')
2020-12-27 04:05:04 +13:00
->inject('response')
2021-06-12 06:07:05 +12:00
->inject('dbForExternal')
->action(function ($collectionId, $response, $dbForExternal) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2021-06-12 06:07:05 +12:00
$collection = $dbForExternal->getCollection($collectionId);
2020-06-30 09:43:34 +12:00
if ($collection->isEmpty()) {
2020-06-30 09:43:34 +12:00
throw new Exception('Collection not found', 404);
2020-02-01 11:34:07 +13:00
}
2020-06-30 09:43:34 +12:00
$response->dynamic2($collection, Response::MODEL_COLLECTION);
2020-12-27 04:05:04 +13:00
});
2020-02-01 11:34:07 +13:00
2020-06-29 05:31:21 +12:00
App::put('/v1/database/collections/:collectionId')
2019-06-09 23:44:58 +12:00
->desc('Update Collection')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
->label('scope', 'collections.write')
2020-10-31 08:53:27 +13:00
->label('event', 'database.collections.update')
2019-06-09 23:44:58 +12:00
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.method', 'updateCollection')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/database/update-collection.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_COLLECTION)
2020-09-11 02:40:14 +12:00
->param('collectionId', '', new UID(), 'Collection unique ID.')
->param('name', null, new Text(128), 'Collection name. Max length: 128 chars.')
2021-03-22 11:17:20 +13:00
->param('read', null, new ArrayList(new Text(64)), 'An array of strings with read permissions. By default inherits the existing read permissions. [learn more about permissions](/docs/permissions) and get a full list of available permissions.', true)
->param('write', null, new ArrayList(new Text(64)), 'An array of strings with write permissions. By default inherits the existing write permissions. [learn more about permissions](/docs/permissions) and get a full list of available permissions.', true)
2020-07-01 03:46:42 +12:00
->param('rules', [], function ($projectDB) { return new ArrayList(new Collection($projectDB, [Database::SYSTEM_COLLECTION_RULES], ['$collection' => Database::SYSTEM_COLLECTION_RULES, '$permissions' => ['read' => [], 'write' => []]])); }, 'Array of [rule objects](/docs/rules). Each rule define a collection field name, data type and validation.', true, ['projectDB'])
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('projectDB')
->inject('audits')
2020-10-31 08:53:27 +13:00
->action(function ($collectionId, $name, $read, $write, $rules, $response, $projectDB, $audits) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
2020-06-30 09:43:34 +12:00
/** @var Appwrite\Database\Database $projectDB */
2020-07-06 02:19:59 +12:00
/** @var Appwrite\Event\Event $audits */
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
$collection = $projectDB->getDocument($collectionId, false);
2019-10-01 17:57:41 +13:00
2020-06-30 09:43:34 +12:00
if (empty($collection->getId()) || Database::SYSTEM_COLLECTION_COLLECTIONS != $collection->getCollection()) {
throw new Exception('Collection not found', 404);
}
2019-10-01 17:57:41 +13:00
2020-06-30 09:43:34 +12:00
$parsedRules = [];
2021-03-22 11:17:20 +13:00
$read = (is_null($read)) ? ($collection->getPermissions()['read'] ?? []) : $read; // By default inherit read permissions
$write = (is_null($write)) ? ($collection->getPermissions()['write'] ?? []) : $write; // By default inherit write permissions
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
foreach ($rules as &$rule) {
$parsedRules[] = \array_merge([
'$collection' => Database::SYSTEM_COLLECTION_RULES,
'$permissions' => [
'read' => $read,
'write' => $write,
],
], $rule);
}
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
try {
$collection = $projectDB->updateDocument(\array_merge($collection->getArrayCopy(), [
'name' => $name,
'structure' => true,
'dateUpdated' => \time(),
'$permissions' => [
'read' => $read,
'write' => $write,
],
'rules' => $parsedRules,
]));
} catch (AuthorizationException $exception) {
2020-08-30 16:49:24 +12:00
throw new Exception('Unauthorized permissions', 401);
2020-06-30 09:43:34 +12:00
} catch (StructureException $exception) {
throw new Exception('Bad structure. '.$exception->getMessage(), 400);
} catch (\Exception $exception) {
throw new Exception('Failed saving document to DB', 500);
}
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
if (false === $collection) {
throw new Exception('Failed saving collection to DB', 500);
}
2020-07-06 02:19:59 +12:00
$audits
2020-06-30 09:43:34 +12:00
->setParam('event', 'database.collections.update')
2020-10-31 08:53:27 +13:00
->setParam('resource', 'database/collections/'.$collection->getId())
->setParam('data', $collection->getArrayCopy())
2020-06-30 09:43:34 +12:00
;
2020-10-31 08:53:27 +13:00
$response->dynamic($collection, Response::MODEL_COLLECTION);
2020-12-27 04:05:04 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::delete('/v1/database/collections/:collectionId')
2019-05-09 18:54:39 +12:00
->desc('Delete Collection')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
->label('scope', 'collections.write')
2020-10-31 08:53:27 +13:00
->label('event', 'database.collections.delete')
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.method', 'deleteCollection')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/database/delete-collection.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.model', Response::MODEL_NONE)
2020-09-11 02:40:14 +12:00
->param('collectionId', '', new UID(), 'Collection unique ID.')
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('dbForExternal')
2020-12-27 04:05:04 +13:00
->inject('events')
->inject('audits')
->inject('deletes')
->action(function ($collectionId, $response, $dbForExternal, $events, $audits, $deletes) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2020-12-07 11:14:57 +13:00
/** @var Appwrite\Event\Event $events */
2020-07-06 02:19:59 +12:00
/** @var Appwrite\Event\Event $audits */
2019-05-09 18:54:39 +12:00
$collection = $dbForExternal->getCollection($collectionId);
2019-05-09 18:54:39 +12:00
if ($collection->isEmpty()) {
2020-06-30 09:43:34 +12:00
throw new Exception('Collection not found', 404);
}
$dbForExternal->deleteCollection($collectionId);
// TODO@kodumbeats use worker to handle this
// $deletes
// ->setParam('type', DELETE_TYPE_DOCUMENT)
// ->setParam('document', $collection)
// ;
2020-12-07 11:14:57 +13:00
$events
->setParam('eventData', $response->output2($collection, Response::MODEL_COLLECTION))
2020-06-30 09:43:34 +12:00
;
2019-05-09 18:54:39 +12:00
2020-07-06 02:19:59 +12:00
$audits
2020-06-30 09:43:34 +12:00
->setParam('event', 'database.collections.delete')
2020-10-31 08:53:27 +13:00
->setParam('resource', 'database/collections/'.$collection->getId())
->setParam('data', $collection->getArrayCopy())
2020-06-30 09:43:34 +12:00
;
$response->noContent();
2020-12-27 04:05:04 +13:00
});
2019-05-09 18:54:39 +12:00
2021-03-25 04:40:33 +13:00
App::post('/v1/database/collections/:collectionId/attributes')
->desc('Create Attribute')
->groups(['api', 'database'])
->label('event', 'database.attributes.create')
->label('scope', 'attributes.write')
->label('sdk.namespace', 'database')
->label('sdk.platform', [APP_PLATFORM_SERVER])
->label('sdk.method', 'createAttribute')
->label('sdk.description', '/docs/references/database/create-attribute.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_ATTRIBUTE)
->param('collectionId', '', new UID(), 'Collection unique ID. You can create a new collection using the Database service [server integration](/docs/server/database#createCollection).')
->param('id', '', new Text(256), 'Attribute ID.')
2021-03-25 04:40:33 +13:00
->param('type', null, new Text(256), 'Attribute type.')
// TODO@kodumbeats add units to description
->param('size', null, new Numeric(), 'Attribute size.')
->param('required', null, new Boolean(), 'Is attribute required?')
2021-06-12 06:07:05 +12:00
->param('signed', true, new Boolean(), 'Is attribute signed?', true)
->param('array', false, new Boolean(), 'Is attribute an array?', true)
// TODO@kodumbeats "We should have here a whitelist of allowed values. We should add a link to a reference in the future documentation. We might be able to hide this option from the public API for now."
2021-06-12 06:07:05 +12:00
->param('filters', [], new ArrayList(new Text(256)), 'Array of filters.', true)
2021-03-25 04:40:33 +13:00
->inject('response')
->inject('dbForExternal')
2021-06-09 08:12:14 +12:00
->inject('audits')
->action(function ($collectionId, $id, $type, $size, $required, $signed, $array, $filters, $response, $dbForExternal, $audits) {
2021-03-25 04:40:33 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal*/
2021-03-25 04:40:33 +13:00
/** @var Appwrite\Event\Event $audits */
2021-06-11 01:15:00 +12:00
$collection = $dbForExternal->getCollection($collectionId);
if ($collection->isEmpty()) {
2021-06-11 01:15:00 +12:00
throw new Exception('Collection not found', 404);
}
// TODO@kodumbeats handle failed attribute creation
$success = $dbForExternal->createAttribute($collectionId, $id, $type, $size, $required, $signed, $array, $filters);
// Database->createAttribute() does not return a document
// So we need to create one for the response
$attribute = new Document2([
'$collection' => $collectionId,
'$id' => $id,
'type' => $type,
'size' => $size,
'required' => $required,
'signed' => $signed,
'array' => $array,
'filters' => $filters
]);
2021-03-25 04:40:33 +13:00
$audits
->setParam('event', 'database.attributes.create')
->setParam('resource', 'database/attributes/'.$attribute->getId())
->setParam('data', $attribute)
2021-03-25 04:40:33 +13:00
;
$response->setStatusCode(Response::STATUS_CODE_CREATED);
$response->dynamic2($attribute, Response::MODEL_ATTRIBUTE);
2021-03-25 04:40:33 +13:00
});
2021-03-26 08:52:57 +13:00
App::get('v1/database/collections/:collectionId/attributes')
->desc('List Attributes')
->groups(['api', 'database'])
->label('scope', 'attributes.read')
->label('sdk.namespace', 'database')
->label('sdk.platform', [APP_PLATFORM_SERVER])
->label('sdk.method', 'listAttributes')
->label('sdk.description', '/docs/references/database/list-attributes.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_ATTRIBUTE_LIST)
->param('collectionId', '', new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
2021-03-26 08:52:57 +13:00
->inject('response')
->inject('dbForExternal')
->action(function ($collectionId, $response, $dbForExternal) {
2021-03-26 08:52:57 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2021-06-09 08:12:14 +12:00
$collection = $dbForExternal->getCollection($collectionId);
2021-03-26 08:52:57 +13:00
if ($collection->isEmpty()) {
2021-06-11 01:15:00 +12:00
throw new Exception('Collection not found', 404);
}
// TODO@kodumbeats array_merge collectionId to each attribute
2021-06-09 08:12:14 +12:00
$attributes = $collection->getAttributes();
2021-03-26 08:52:57 +13:00
$response->dynamic2(new Document2([
'sum' => \count($attributes),
2021-06-09 08:12:14 +12:00
'attributes' => $attributes
]), Response::MODEL_ATTRIBUTE_LIST);
2021-03-26 08:52:57 +13:00
});
App::get('v1/database/collections/:collectionId/attributes/:attributeId')
->desc('Get Attribute')
->groups(['api', 'database'])
->label('scope', 'attributes.read')
->label('sdk.namespace', 'database')
->label('sdk.platform', [APP_PLATFORM_SERVER])
->label('sdk.method', 'listAttributes')
->label('sdk.description', '/docs/references/database/get-attribute.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_ATTRIBUTE)
->param('collectionId', '', new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('attributeId', '', new Text(256), 'Attribute ID.')
2021-03-26 08:52:57 +13:00
->inject('response')
->inject('dbForExternal')
->action(function ($collectionId, $attributeId, $response, $dbForExternal) {
2021-03-26 08:52:57 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2021-06-09 08:12:14 +12:00
$collection = $dbForExternal->getCollection($collectionId);
2021-03-26 08:52:57 +13:00
2021-06-11 01:15:00 +12:00
if (empty($collection)) {
throw new Exception('Collection not found', 404);
}
2021-06-09 08:12:14 +12:00
$attributes = $collection->getAttributes();
2021-03-26 08:52:57 +13:00
// Search for attribute
$attributeIndex = array_search($attributeId, array_column($attributes, '$id'));
if ($attributeIndex === false) {
throw new Exception('Attribute not found', 404);
}
$attribute = new Document2([\array_merge($attributes[$attributeIndex], [
'collectionId' => $collectionId,
])]);
$response->dynamic2($attribute, Response::MODEL_ATTRIBUTE);
2021-03-26 08:52:57 +13:00
});
2021-03-25 04:40:33 +13:00
App::delete('/v1/database/collections/:collectionId/attributes/:attributeId')
->desc('Delete Attribute')
->groups(['api', 'database'])
->label('scope', 'attributes.write')
->label('event', 'database.attributes.delete')
->label('sdk.namespace', 'database')
->label('sdk.platform', [APP_PLATFORM_SERVER])
->label('sdk.method', 'deleteAttribute')
->label('sdk.description', '/docs/references/database/delete-attribute.md')
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.model', Response::MODEL_NONE)
->param('collectionId', '', new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('attributeId', '', new Text(256), 'Attribute ID.')
2021-03-25 04:40:33 +13:00
->inject('response')
->inject('dbForExternal')
2021-03-25 04:40:33 +13:00
->inject('events')
->inject('audits')
->inject('deletes')
->action(function ($collectionId, $attributeId, $response, $dbForExternal, $events, $audits, $deletes) {
2021-03-25 04:40:33 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2021-03-25 04:40:33 +13:00
/** @var Appwrite\Event\Event $events */
/** @var Appwrite\Event\Event $audits */
$collection = $dbForExternal->getCollection($collectionId);
2021-03-25 04:40:33 +13:00
if ($collection->isEmpty()) {
2021-06-11 01:15:00 +12:00
throw new Exception('Collection not found', 404);
}
2021-06-09 08:12:14 +12:00
$attributes = $collection->getAttributes();
2021-03-25 04:40:33 +13:00
// Search for attribute
$attributeIndex = array_search($attributeId, array_column($attributes, '$id'));
2021-03-25 04:40:33 +13:00
if ($attributeIndex === false) {
throw new Exception('Attribute not found', 404);
}
$attribute = new Document2([\array_merge($attributes[$attributeIndex], [
'collectionId' => $collectionId,
])]);
2021-06-09 08:12:14 +12:00
// TODO@kodumbeats use the deletes worker to handle this
$success = $dbForExternal->deleteAttribute($collectionId, $attributeId);
2021-06-09 08:12:14 +12:00
// $deletes
// ->setParam('type', DELETE_TYPE_DOCUMENT)
// ->setParam('document', $attribute)
// ;
$events
->setParam('payload', $response->output2($attribute, Response::MODEL_ATTRIBUTE))
;
2021-03-25 04:40:33 +13:00
$audits
->setParam('event', 'database.attributes.delete')
->setParam('resource', 'database/attributes/'.$attribute->getId())
->setParam('data', $attribute->getArrayCopy())
;
$response->noContent();
});
2021-03-24 10:19:19 +13:00
App::post('/v1/database/collections/:collectionId/indexes')
->desc('Create Index')
->groups(['api', 'database'])
->label('event', 'database.indexes.create')
->label('scope', 'indexes.write')
->label('sdk.namespace', 'database')
2021-03-25 04:40:33 +13:00
->label('sdk.platform', [APP_PLATFORM_SERVER])
2021-03-24 10:19:19 +13:00
->label('sdk.method', 'createIndex')
->label('sdk.description', '/docs/references/database/create-index.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_INDEX)
->param('collectionId', '', new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('id', null, new Text(256), 'Index ID.')
// TODO@kodumbeats type should be a whitelist of supported index types
2021-03-24 10:19:19 +13:00
->param('type', null, new Text(256), 'Index type.')
->param('attributes', null, new ArrayList(new Text(256)), 'Array of attributes to index.')
->param('lengths', null, new ArrayList(new Text(256)), 'Array of index lengths.')
->param('orders', null, new ArrayList(new Text(256)), 'Array of index orders.')
2021-03-24 10:19:19 +13:00
->inject('response')
->inject('dbForExternal')
->inject('audits')
->action(function ($collectionId, $id, $type, $attributes, $lengths, $orders, $response, $dbForExternal, $audits) {
2021-03-24 10:19:19 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2021-03-24 10:19:19 +13:00
/** @var Appwrite\Event\Event $audits */
2021-06-11 01:15:00 +12:00
$collection = $dbForExternal->getCollection($collectionId);
if ($collection->isEmpty()) {
2021-06-11 01:15:00 +12:00
throw new Exception('Collection not found', 404);
}
$success = $dbForExternal->createIndex($collectionId, $id, $type, $attributes, $lengths, $orders);
// Database->createIndex() does not return a document
// So we need to create one for the response
$index = new Document2([
'$collection' => $collectionId,
'$id' => $id,
'type' => $type,
'attributes' => $attributes,
'lengths' => $lengths,
'orders' => $orders,
]);
2021-03-24 10:19:19 +13:00
$audits
->setParam('event', 'database.indexes.create')
->setParam('resource', 'database/indexes/'.$index->getId())
->setParam('data', $index->getArrayCopy())
2021-03-24 10:19:19 +13:00
;
$response->setStatusCode(Response::STATUS_CODE_CREATED);
$response->dynamic2($index, Response::MODEL_INDEX);
2021-03-24 10:19:19 +13:00
});
2021-03-26 08:52:57 +13:00
App::get('v1/database/collections/:collectionId/indexes')
2021-03-25 04:40:33 +13:00
->desc('List Indexes')
->groups(['api', 'database'])
->label('scope', 'indexes.read')
->label('sdk.namespace', 'database')
->label('sdk.platform', [APP_PLATFORM_SERVER])
->label('sdk.method', 'listIndexes')
->label('sdk.description', '/docs/references/database/list-indexes.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_INDEX_LIST)
->param('collectionId', '', new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
2021-03-25 04:40:33 +13:00
->inject('response')
->inject('dbForExternal')
->action(function ($collectionId, $response, $dbForExternal) {
2021-03-25 04:40:33 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
$collection = $dbForExternal->getCollection($collectionId);
2021-03-25 04:40:33 +13:00
if ($collection->isEmpty()) {
2021-06-11 01:15:00 +12:00
throw new Exception('Collection not found', 404);
}
// TODO@kodumbeats decode index string and merge ['$collection' => $collectionId]
$indexes = $collection->getAttribute('indexes');
2021-03-25 04:40:33 +13:00
$response->dynamic2(new Document2([
'sum' => \count($indexes),
'attributes' => $indexes,
]), Response::MODEL_INDEX_LIST);
2021-03-25 04:40:33 +13:00
});
2021-03-26 08:52:57 +13:00
App::get('v1/database/collections/:collectionId/indexes/:indexId')
->desc('Get Index')
->groups(['api', 'database'])
->label('scope', 'indexes.read')
->label('sdk.namespace', 'database')
->label('sdk.platform', [APP_PLATFORM_SERVER])
->label('sdk.method', 'listIndexes')
->label('sdk.description', '/docs/references/database/get-index.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_INDEX)
->param('collectionId', '', new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('indexId', null, new Text(256), 'Index ID.')
2021-03-26 08:52:57 +13:00
->inject('response')
->inject('projectDB')
->action(function ($collectionId, $indexId, $response, $dbForExternal) {
2021-03-26 08:52:57 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2021-03-26 08:52:57 +13:00
$collection = $dbForExternal->getCollection($collectionId);
if ($collection->isEmpty()) {
2021-06-11 01:15:00 +12:00
throw new Exception('Collection not found', 404);
}
// TODO@kodumbeats decode 'indexes' into array
$indexes = $collection->getAttribute('indexes');
2021-03-26 08:52:57 +13:00
// // Search for index
$indexIndex = array_search($indexId, array_column($indexes, '$id'));
if ($indexIndex === false) {
throw new Exception('Index not found', 404);
}
$index = new Document2([\array_merge($indexes[$indexIndex], [
'collectionId' => $collectionId,
])]);
$response->dynamic2($index, Response::MODEL_INDEX);
2021-03-26 08:52:57 +13:00
});
App::delete('/v1/database/collections/:collectionId/indexes/:indexId')
->desc('Delete Index')
->groups(['api', 'database'])
->label('scope', 'indexes.write')
->label('event', 'database.indexes.delete')
->label('sdk.namespace', 'database')
->label('sdk.platform', [APP_PLATFORM_SERVER])
->label('sdk.method', 'deleteIndex')
->label('sdk.description', '/docs/references/database/delete-index.md')
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.model', Response::MODEL_NONE)
->param('collectionId', null, new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('indexId', '', new UID(), 'Index unique ID.')
2021-03-26 08:52:57 +13:00
->inject('response')
->inject('dbForExternal')
2021-03-26 08:52:57 +13:00
->inject('events')
->inject('audits')
->inject('deletes')
->action(function ($collectionId, $indexId, $response, $dbForExternal, $events, $audits, $deletes) {
2021-03-26 08:52:57 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2021-03-26 08:52:57 +13:00
/** @var Appwrite\Event\Event $events */
/** @var Appwrite\Event\Event $audits */
$collection = $dbForExternal->getCollection($collectionId);
2021-06-12 06:07:05 +12:00
if ($collection->isEmpty()) {
2021-06-11 01:15:00 +12:00
throw new Exception('Collection not found', 404);
}
// TODO@kodumbeats decode 'indexes' into array
$indexes = $collection->getAttribute('indexes');
2021-03-26 08:52:57 +13:00
// // Search for index
$indexIndex = array_search($indexId, array_column($indexes, '$id'));
if ($indexIndex === false) {
throw new Exception('Index not found', 404);
2021-03-26 08:52:57 +13:00
}
$index = new Document2([\array_merge($indexes[$indexIndex], [
'collectionId' => $collectionId,
])]);
// TODO@kodumbeats use the deletes worker to handle this
$success = $dbForExternal->deleteIndex($collectionId, $indexId);
// $deletes
// ->setParam('type', DELETE_TYPE_DOCUMENT)
// ->setParam('document', $attribute)
// ;
2021-03-26 08:52:57 +13:00
$events
->setParam('payload', $response->output2($index, Response::MODEL_INDEX))
2021-03-26 08:52:57 +13:00
;
$audits
->setParam('event', 'database.indexes.delete')
->setParam('resource', 'database/indexes/'.$index->getId())
2021-03-26 08:52:57 +13:00
->setParam('data', $index->getArrayCopy())
;
$response->noContent();
});
2020-06-29 05:31:21 +12:00
App::post('/v1/database/collections/:collectionId/documents')
2020-02-01 11:34:07 +13:00
->desc('Create Document')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
2020-10-31 08:53:27 +13:00
->label('event', 'database.documents.create')
2020-02-01 11:34:07 +13:00
->label('scope', 'documents.write')
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_KEY, APP_AUTH_TYPE_JWT])
2020-02-01 11:34:07 +13:00
->label('sdk.method', 'createDocument')
->label('sdk.description', '/docs/references/database/create-document.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_DOCUMENT)
2020-09-11 02:40:14 +12:00
->param('collectionId', null, new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('data', [], new JSON(), 'Document data as JSON object.')
2021-03-22 11:17:20 +13:00
->param('read', null, new ArrayList(new Text(64)), 'An array of strings with read permissions. By default only the current user is granted with read permissions. [learn more about permissions](/docs/permissions) and get a full list of available permissions.', true)
->param('write', null, new ArrayList(new Text(64)), 'An array of strings with write permissions. By default only the current user is granted with write permissions. [learn more about permissions](/docs/permissions) and get a full list of available permissions.', true)
// ->param('parentDocument', '', new UID(), 'Parent document unique ID. Use when you want your new document to be a child of a parent document.', true)
// ->param('parentProperty', '', new Key(), 'Parent document property name. Use when you want your new document to be a child of a parent document.', true)
// ->param('parentPropertyType', Document::SET_TYPE_ASSIGN, new WhiteList([Document::SET_TYPE_ASSIGN, Document::SET_TYPE_APPEND, Document::SET_TYPE_PREPEND], true), 'Parent document property connection type. You can set this value to **assign**, **append** or **prepend**, default value is assign. Use when you want your new document to be a child of a parent document.', true)
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('dbForExternal')
2021-03-22 11:17:20 +13:00
->inject('user')
2020-12-27 04:05:04 +13:00
->inject('audits')
->action(function ($collectionId, $data, $read, $write, /*$parentDocument, $parentProperty, $parentPropertyType,*/ $response, $dbForExternal, $user, $audits) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
/** @var Utopia\Database\Document $user */
2020-07-06 02:19:59 +12:00
/** @var Appwrite\Event\Event $audits */
2020-06-30 09:43:34 +12:00
$data = (\is_string($data)) ? \json_decode($data, true) : $data; // Cast to JSON array
if (empty($data)) {
throw new Exception('Missing payload', 400);
}
2020-02-01 11:34:07 +13:00
2020-06-30 09:43:34 +12:00
if (isset($data['$id'])) {
throw new Exception('$id is not allowed for creating new documents, try update instead', 400);
}
$collection = $dbForExternal->getCollection($collectionId);
2020-02-01 11:34:07 +13:00
2021-06-12 06:07:05 +12:00
if ($collection->isEmpty()) {
2020-06-30 09:43:34 +12:00
throw new Exception('Collection not found', 404);
}
2020-02-01 11:34:07 +13:00
2020-06-30 09:43:34 +12:00
$data['$collection'] = $collectionId; // Adding this param to make API easier for developers
$data['$read'] = (is_null($read) && !$user->isEmpty()) ? ['user:'.$user->getId()] : $read ?? []; // By default set read permissions for user
$data['$write'] = (is_null($write) && !$user->isEmpty()) ? ['user:'.$user->getId()] : $write ?? []; // By default set write permissions for user
2020-02-01 11:34:07 +13:00
2020-06-30 09:43:34 +12:00
/**
* TODO@kodumbeats Set default collection values
2020-06-30 09:43:34 +12:00
*/
// foreach ($collection->getAttribute('rules') as $key => $rule) {
// $key = $rule['key'] ?? '';
// $default = $rule['default'] ?? null;
// if (!isset($data[$key])) {
// $data[$key] = $default;
// }
// }
$document = $dbForExternal->createDocument($collectionId, new Document2($data));
2020-02-01 11:34:07 +13:00
2020-07-06 02:19:59 +12:00
$audits
2020-06-30 09:43:34 +12:00
->setParam('event', 'database.documents.create')
->setParam('resource', 'database/document/'.$document->getId())
->setParam('data', $document->getArrayCopy())
2020-06-30 09:43:34 +12:00
;
2020-02-01 11:34:07 +13:00
2021-05-27 22:09:14 +12:00
$response->setStatusCode(Response::STATUS_CODE_CREATED);
$response->dynamic2($document, Response::MODEL_DOCUMENT);
2020-12-27 04:05:04 +13:00
});
2020-02-01 11:34:07 +13:00
2020-06-29 05:31:21 +12:00
App::get('/v1/database/collections/:collectionId/documents')
2019-05-09 18:54:39 +12:00
->desc('List Documents')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
->label('scope', 'documents.read')
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_KEY, APP_AUTH_TYPE_JWT])
2019-05-09 18:54:39 +12:00
->label('sdk.method', 'listDocuments')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/database/list-documents.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_DOCUMENT_LIST)
2020-09-11 02:40:14 +12:00
->param('collectionId', null, new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('filters', [], new ArrayList(new Text(128)), 'Array of filter strings. Each filter is constructed from a key name, comparison operator (=, !=, >, <, <=, >=) and a value. You can also use a dot (.) separator in attribute names to filter by child document attributes. Examples: \'name=John Doe\' or \'category.$id>=5bed2d152c362\'.', true)
2020-11-06 05:02:25 +13:00
->param('limit', 25, new Range(0, 100), 'Maximum number of documents to return in response. Use this value to manage pagination. By default will return maximum 25 results. Maximum of 100 results allowed per request.', true)
->param('offset', 0, new Range(0, 900000000), 'Offset value. The default value is 0. Use this param to manage pagination.', true)
2020-12-13 12:22:55 +13:00
->param('orderField', '', new Text(128), 'Document field that results will be sorted by.', true)
2020-09-11 02:40:14 +12:00
->param('orderType', 'ASC', new WhiteList(['DESC', 'ASC'], true), 'Order direction. Possible values are DESC for descending order, or ASC for ascending order.', true)
->param('orderCast', 'string', new WhiteList(['int', 'string', 'date', 'time', 'datetime'], true), 'Order field type casting. Possible values are int, string, date, time or datetime. The database will attempt to cast the order field to the value you pass here. The default value is a string.', true)
->param('search', '', new Text(256), 'Search query. Enter any free text search. The database will try to find a match against all document attributes and children. Max length: 256 chars.', true)
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('projectDB')
2020-07-09 02:28:01 +12:00
->action(function ($collectionId, $filters, $limit, $offset, $orderField, $orderType, $orderCast, $search, $response, $projectDB) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
2020-06-30 09:43:34 +12:00
/** @var Appwrite\Database\Database $projectDB */
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
$collection = $projectDB->getDocument($collectionId, false);
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
if (\is_null($collection->getId()) || Database::SYSTEM_COLLECTION_COLLECTIONS != $collection->getCollection()) {
throw new Exception('Collection not found', 404);
}
2020-06-22 00:12:13 +12:00
2020-06-30 09:43:34 +12:00
$list = $projectDB->getCollection([
'limit' => $limit,
'offset' => $offset,
'orderField' => $orderField,
'orderType' => $orderType,
'orderCast' => $orderCast,
'search' => $search,
'filters' => \array_merge($filters, [
'$collection='.$collectionId,
]),
]);
2020-10-31 08:53:27 +13:00
// if (App::isDevelopment()) {
// $collection
// ->setAttribute('debug', $projectDB->getDebug())
// ->setAttribute('limit', $limit)
// ->setAttribute('offset', $offset)
// ->setAttribute('orderField', $orderField)
// ->setAttribute('orderType', $orderType)
// ->setAttribute('orderCast', $orderCast)
// ->setAttribute('filters', $filters)
// ;
// }
2020-06-30 09:43:34 +12:00
$collection
->setAttribute('sum', $projectDB->getSum())
->setAttribute('documents', $list)
;
2020-10-31 08:53:27 +13:00
$response->dynamic($collection, Response::MODEL_DOCUMENT_LIST);
2020-12-27 04:05:04 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::get('/v1/database/collections/:collectionId/documents/:documentId')
2019-05-09 18:54:39 +12:00
->desc('Get Document')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
->label('scope', 'documents.read')
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_KEY, APP_AUTH_TYPE_JWT])
2019-05-09 18:54:39 +12:00
->label('sdk.method', 'getDocument')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/database/get-document.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_DOCUMENT)
2020-09-11 02:40:14 +12:00
->param('collectionId', null, new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('documentId', null, new UID(), 'Document unique ID.')
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('dbForExternal')
->action(function ($collectionId, $documentId, $response, $dbForExternal) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2019-05-09 18:54:39 +12:00
$collection = $dbForExternal->getCollection($collectionId);
2019-05-09 18:54:39 +12:00
2021-06-12 06:07:05 +12:00
if ($collection->isEmpty()) {
throw new Exception('Collection not found', 404);
}
$document = $dbForExternal->getDocument($collectionId, $documentId);
if ($document->isEmpty()) {
2020-06-30 09:43:34 +12:00
throw new Exception('No document found', 404);
}
2019-05-09 18:54:39 +12:00
$response->dynamic2($document, Response::MODEL_DOCUMENT);
2020-12-27 04:05:04 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::patch('/v1/database/collections/:collectionId/documents/:documentId')
2019-05-09 18:54:39 +12:00
->desc('Update Document')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
2020-10-31 08:53:27 +13:00
->label('event', 'database.documents.update')
->label('scope', 'documents.write')
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_KEY, APP_AUTH_TYPE_JWT])
2019-05-09 18:54:39 +12:00
->label('sdk.method', 'updateDocument')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/database/update-document.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_DOCUMENT)
2020-09-11 02:40:14 +12:00
->param('collectionId', null, new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('documentId', null, new UID(), 'Document unique ID.')
->param('data', [], new JSON(), 'Document data as JSON object.')
2021-03-22 11:17:20 +13:00
->param('read', null, new ArrayList(new Text(64)), 'An array of strings with read permissions. By default inherits the existing read permissions. [learn more about permissions](/docs/permissions) and get a full list of available permissions.', true)
->param('write', null, new ArrayList(new Text(64)), 'An array of strings with write permissions. By default inherits the existing write permissions. [learn more about permissions](/docs/permissions) and get a full list of available permissions.', true)
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('projectDB')
->inject('audits')
2020-10-31 08:53:27 +13:00
->action(function ($collectionId, $documentId, $data, $read, $write, $response, $projectDB, $audits) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
2020-06-30 09:43:34 +12:00
/** @var Appwrite\Database\Database $projectDB */
2020-07-06 02:19:59 +12:00
/** @var Appwrite\Event\Event $audits */
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
$collection = $projectDB->getDocument($collectionId, false);
$document = $projectDB->getDocument($documentId, false);
2019-09-17 07:03:24 +12:00
2020-06-30 09:43:34 +12:00
$data = (\is_string($data)) ? \json_decode($data, true) : $data; // Cast to JSON array
2021-03-22 11:17:20 +13:00
2020-06-30 09:43:34 +12:00
if (!\is_array($data)) {
2020-10-31 08:53:27 +13:00
throw new Exception('Data param should be a valid JSON object', 400);
2020-06-30 09:43:34 +12:00
}
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
if (\is_null($collection->getId()) || Database::SYSTEM_COLLECTION_COLLECTIONS != $collection->getCollection()) {
throw new Exception('Collection not found', 404);
}
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
if (empty($document->getArrayCopy()) || $document->getCollection() != $collectionId) { // Check empty
throw new Exception('No document found', 404);
}
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
$data = \array_merge($document->getArrayCopy(), $data);
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
$data['$collection'] = $collection->getId(); // Make sure user don't switch collectionID
$data['$id'] = $document->getId(); // Make sure user don't switch document unique ID
2021-03-22 11:17:20 +13:00
$data['$permissions']['read'] = (is_null($read)) ? ($document->getPermissions()['read'] ?? []) : $read; // By default inherit read permissions
$data['$permissions']['write'] = (is_null($write)) ? ($document->getPermissions()['write'] ?? []) : $write; // By default inherit write permissions
2020-06-30 09:43:34 +12:00
if (empty($data)) {
throw new Exception('Missing payload', 400);
}
2020-10-31 08:53:27 +13:00
2020-06-30 09:43:34 +12:00
try {
$data = $projectDB->updateDocument($data);
} catch (AuthorizationException $exception) {
2020-08-30 16:49:24 +12:00
throw new Exception('Unauthorized permissions', 401);
2020-06-30 09:43:34 +12:00
} catch (StructureException $exception) {
throw new Exception('Bad structure. '.$exception->getMessage(), 400);
} catch (\Exception $exception) {
throw new Exception('Failed saving document to DB', 500);
}
2019-05-09 18:54:39 +12:00
2020-07-06 02:19:59 +12:00
$audits
2020-06-30 09:43:34 +12:00
->setParam('event', 'database.documents.update')
2020-10-31 08:53:27 +13:00
->setParam('resource', 'database/document/'.$data->getId())
->setParam('data', $data->getArrayCopy())
2020-06-30 09:43:34 +12:00
;
2019-05-09 18:54:39 +12:00
$response->dynamic($data, Response::MODEL_DOCUMENT);
2020-12-27 04:05:04 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::delete('/v1/database/collections/:collectionId/documents/:documentId')
2019-05-09 18:54:39 +12:00
->desc('Delete Document')
2020-06-26 06:32:12 +12:00
->groups(['api', 'database'])
->label('scope', 'documents.write')
2020-10-31 08:53:27 +13:00
->label('event', 'database.documents.delete')
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'database')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_KEY, APP_AUTH_TYPE_JWT])
2019-05-09 18:54:39 +12:00
->label('sdk.method', 'deleteDocument')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/database/delete-document.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.model', Response::MODEL_NONE)
2020-09-11 02:40:14 +12:00
->param('collectionId', null, new UID(), 'Collection unique ID. You can create a new collection with validation rules using the Database service [server integration](/docs/server/database#createCollection).')
->param('documentId', null, new UID(), 'Document unique ID.')
2020-12-27 04:05:04 +13:00
->inject('response')
->inject('projectDB')
->inject('events')
->inject('audits')
->action(function ($collectionId, $documentId, $response, $dbForExternal, $events, $audits) {
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
/** @var Utopia\Database\Database $dbForExternal */
2020-12-07 11:14:57 +13:00
/** @var Appwrite\Event\Event $events */
2020-07-06 02:19:59 +12:00
/** @var Appwrite\Event\Event $audits */
2019-05-09 18:54:39 +12:00
$collection = $dbForExternal->getCollection($collectionId);
2019-05-09 18:54:39 +12:00
2021-06-12 06:07:05 +12:00
if ($collection->isEmpty()) {
2020-06-30 09:43:34 +12:00
throw new Exception('Collection not found', 404);
}
$document = $dbForExternal->getDocument($collectionId, $documentId);
if ($document->isEmpty()) {
throw new Exception('No document found', 404);
2020-06-30 09:43:34 +12:00
}
2019-05-09 18:54:39 +12:00
$success = $dbForExternal->deleteDocument($collectionId, $documentId);
2020-12-07 11:14:57 +13:00
$events
->setParam('eventData', $response->output2($document, Response::MODEL_DOCUMENT))
2020-06-30 09:43:34 +12:00
;
2020-07-06 02:19:59 +12:00
$audits
2020-06-30 09:43:34 +12:00
->setParam('event', 'database.documents.delete')
2020-10-31 08:53:27 +13:00
->setParam('resource', 'database/document/'.$document->getId())
->setParam('data', $document->getArrayCopy()) // Audit document in case of malicious or disastrous action
2020-06-30 09:43:34 +12:00
;
2019-05-09 18:54:39 +12:00
2020-06-30 09:43:34 +12:00
$response->noContent();
2021-03-26 08:52:57 +13:00
});