1
0
Fork 0
mirror of synced 2024-07-07 23:46:11 +12:00
appwrite/app/controllers/api/messaging.php

1762 lines
76 KiB
PHP
Raw Normal View History

<?php
2023-10-18 06:23:26 +13:00
use Appwrite\Event\Delete;
2023-09-06 22:10:56 +12:00
use Appwrite\Event\Messaging;
use Appwrite\Extend\Exception;
2023-09-06 22:10:56 +12:00
use Appwrite\Permission;
use Appwrite\Role;
2023-08-25 08:15:17 +12:00
use Appwrite\Utopia\Database\Validator\CustomId;
2023-10-14 01:56:54 +13:00
use Appwrite\Utopia\Database\Validator\Queries\Messages;
2023-08-23 04:45:31 +12:00
use Appwrite\Utopia\Database\Validator\Queries\Providers;
2023-08-25 08:15:17 +12:00
use Appwrite\Utopia\Database\Validator\Queries\Topics;
use Appwrite\Utopia\Response;
use Utopia\App;
use Utopia\Database\Database;
use Utopia\Database\Document;
2023-08-29 03:11:05 +12:00
use Utopia\Database\Exception\Duplicate as DuplicateException;
use Utopia\Database\Helpers\ID;
2023-08-23 04:45:31 +12:00
use Utopia\Database\Query;
use Utopia\Database\Validator\Authorization;
2023-09-06 22:10:56 +12:00
use Utopia\Database\Validator\Datetime as DatetimeValidator;
2023-08-23 04:45:31 +12:00
use Utopia\Database\Validator\UID;
use Utopia\Validator\ArrayList;
2023-08-30 04:59:48 +12:00
use Utopia\Validator\Boolean;
use Utopia\Validator\Text;
use Utopia\Validator\WhiteList;
App::post('/v1/messaging/providers/mailgun')
2023-08-24 21:01:22 +12:00
->desc('Create Mailgun Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.create')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'createMailgunProvider')
->label('sdk.description', '/docs/references/messaging/create-mailgun-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.')
2023-08-30 04:59:48 +12:00
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
->param('isEuRegion', false, new Boolean(), 'Set as EU region.', true)
->param('from', '', new Text(256), 'Sender Email Address.')
2023-08-24 21:01:22 +12:00
->param('apiKey', '', new Text(0), 'Mailgun API Key.')
->param('domain', '', new Text(0), 'Mailgun Domain.')
->inject('dbForProject')
->inject('response')
->action(function (string $providerId, string $name, bool $default, bool $enabled, bool $isEuRegion, string $from, string $apiKey, string $domain, Database $dbForProject, Response $response) {
2023-08-25 11:16:15 +12:00
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
2023-08-30 04:59:48 +12:00
$provider = new Document([
2023-08-25 11:16:15 +12:00
'$id' => $providerId,
'name' => $name,
2023-08-24 21:01:22 +12:00
'provider' => 'mailgun',
'type' => 'email',
2023-08-30 04:59:48 +12:00
'default' => $default,
'enabled' => $enabled,
'search' => $providerId . ' ' . $name . ' ' . 'mailgun' . ' ' . 'email',
2023-08-24 21:01:22 +12:00
'credentials' => [
'apiKey' => $apiKey,
'domain' => $domain,
'isEuRegion' => $isEuRegion,
2023-08-24 21:01:22 +12:00
],
'options' => [
'from' => $from,
]
2023-08-30 04:59:48 +12:00
]);
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-06 00:27:48 +13:00
Query::equal('default', [true]),
Query::equal('type', ['email'])
2023-08-30 04:59:48 +12:00
]))
) {
$provider->setAttribute('default', true);
}
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
2023-08-30 04:59:48 +12:00
}
2023-08-24 21:01:22 +12:00
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
App::post('/v1/messaging/providers/sendgrid')
2023-08-24 21:01:22 +12:00
->desc('Create Sendgrid Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.create')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'createSendgridProvider')
->label('sdk.description', '/docs/references/messaging/create-sengrid-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.')
2023-08-30 04:59:48 +12:00
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
2023-08-24 21:01:22 +12:00
->param('apiKey', '', new Text(0), 'Sendgrid API key.')
->inject('dbForProject')
->inject('response')
->action(function (string $providerId, string $name, bool $default, bool $enabled, string $apiKey, Database $dbForProject, Response $response) {
2023-08-25 11:16:15 +12:00
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
2023-08-30 04:59:48 +12:00
$provider = new Document([
2023-08-25 11:16:15 +12:00
'$id' => $providerId,
'name' => $name,
2023-08-24 21:01:22 +12:00
'provider' => 'sendgrid',
'type' => 'email',
2023-08-30 04:59:48 +12:00
'default' => $default,
'enabled' => $enabled,
'options' => [],
'search' => $providerId . ' ' . $name . ' ' . 'sendgrid' . ' ' . 'email',
2023-08-24 21:01:22 +12:00
'credentials' => [
'apiKey' => $apiKey,
],
2023-08-30 04:59:48 +12:00
]);
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-06 00:27:48 +13:00
Query::equal('default', [true]),
Query::equal('type', ['sms'])
2023-08-30 04:59:48 +12:00
]))
) {
$provider->setAttribute('default', true);
}
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
2023-08-30 04:59:48 +12:00
}
2023-08-24 21:01:22 +12:00
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
App::post('/v1/messaging/providers/msg91')
2023-08-24 21:01:22 +12:00
->desc('Create Msg91 Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.create')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'createMsg91Provider')
->label('sdk.description', '/docs/references/messaging/create-msg91-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-08-25 11:16:15 +12:00
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.')
2023-08-30 04:59:48 +12:00
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
->param('from', '', new Text(256), 'Sender Number.')
2023-08-24 21:01:22 +12:00
->param('senderId', '', new Text(0), 'Msg91 Sender ID.')
->param('authKey', '', new Text(0), 'Msg91 Auth Key.')
->inject('dbForProject')
->inject('response')
->action(function (string $providerId, string $name, bool $default, bool $enabled, string $from, string $senderId, string $authKey, Database $dbForProject, Response $response) {
2023-08-25 11:16:15 +12:00
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
2023-08-30 04:59:48 +12:00
$provider = new Document([
2023-08-25 11:16:15 +12:00
'$id' => $providerId,
'name' => $name,
2023-08-24 21:01:22 +12:00
'provider' => 'msg91',
'type' => 'sms',
'search' => $providerId . ' ' . $name . ' ' . 'msg91' . ' ' . 'sms',
2023-08-30 04:59:48 +12:00
'default' => $default,
'enabled' => $enabled,
2023-08-24 21:01:22 +12:00
'credentials' => [
'senderId' => $senderId,
'authKey' => $authKey,
],
'options' => [
'from' => $from,
]
2023-08-30 04:59:48 +12:00
]);
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-04 23:45:59 +13:00
Query::equal('default', [true]),
Query::equal('type', ['sms'])
2023-08-30 04:59:48 +12:00
]))
) {
$provider->setAttribute('default', true);
}
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
2023-08-30 04:59:48 +12:00
}
2023-08-24 21:01:22 +12:00
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-10-06 00:27:48 +13:00
2023-08-23 04:45:31 +12:00
App::post('/v1/messaging/providers/telesign')
2023-08-24 21:01:22 +12:00
->desc('Create Telesign Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.create')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'createTelesignProvider')
->label('sdk.description', '/docs/references/messaging/create-telesign-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.')
2023-08-30 04:59:48 +12:00
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
2023-08-24 21:01:22 +12:00
->param('username', '', new Text(0), 'Telesign username.')
->param('password', '', new Text(0), 'Telesign password.')
->inject('dbForProject')
->inject('response')
->action(function (string $providerId, string $name, bool $default, bool $enabled, string $username, string $password, Database $dbForProject, Response $response) {
2023-08-25 11:16:15 +12:00
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
2023-08-30 04:59:48 +12:00
$provider = new Document([
2023-08-25 11:16:15 +12:00
'$id' => $providerId,
'name' => $name,
2023-08-24 21:01:22 +12:00
'provider' => 'telesign',
'type' => 'sms',
'search' => $providerId . ' ' . $name . ' ' . 'telesign' . ' ' . 'sms',
2023-08-30 04:59:48 +12:00
'default' => $default,
'enabled' => $enabled,
2023-08-24 21:01:22 +12:00
'credentials' => [
'username' => $username,
'password' => $password,
],
2023-08-30 04:59:48 +12:00
]);
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-06 00:27:48 +13:00
Query::equal('default', [true]),
Query::equal('type', ['sms'])
2023-08-30 04:59:48 +12:00
]))
) {
$provider->setAttribute('default', true);
}
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
2023-08-30 04:59:48 +12:00
}
2023-08-24 21:01:22 +12:00
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
2023-10-04 23:45:59 +13:00
App::post('/v1/messaging/providers/textmagic')
->desc('Create Textmagic Provider')
2023-08-24 21:01:22 +12:00
->groups(['api', 'messaging'])
2023-10-04 23:45:59 +13:00
->label('audits.event', 'providers.create')
2023-08-24 21:01:22 +12:00
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'createTextmagicProvider')
->label('sdk.description', '/docs/references/messaging/create-textmagic-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
2023-08-24 21:01:22 +12:00
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-04 23:45:59 +13:00
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
->param('name', '', new Text(128), 'Provider name.')
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
->param('username', '', new Text(0), 'Textmagic username.')
->param('apiKey', '', new Text(0), 'Textmagic apiKey.')
2023-08-24 21:01:22 +12:00
->inject('dbForProject')
->inject('response')
2023-10-04 23:45:59 +13:00
->action(function (string $providerId, string $name, bool $default, bool $enabled, string $username, string $apiKey, Database $dbForProject, Response $response) {
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
$provider = new Document([
'$id' => $providerId,
'name' => $name,
'provider' => 'text-magic',
'type' => 'sms',
'search' => $providerId . ' ' . $name . ' ' . 'text-magic' . ' ' . 'sms',
'default' => $default,
'enabled' => $enabled,
'credentials' => [
'username' => $username,
'apiKey' => $apiKey,
],
]);
2023-08-24 21:01:22 +12:00
2023-10-04 23:45:59 +13:00
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-06 00:27:48 +13:00
Query::equal('default', [true]),
Query::equal('type', ['sms'])
2023-10-04 23:45:59 +13:00
]))
) {
$provider->setAttribute('default', true);
2023-08-24 21:01:22 +12:00
}
2023-10-04 23:45:59 +13:00
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
}
2023-10-04 23:45:59 +13:00
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($provider, Response::MODEL_PROVIDER);
});
App::post('/v1/messaging/providers/twilio')
->desc('Create Twilio Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.create')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'createTwilioProvider')
->label('sdk.description', '/docs/references/messaging/create-twilio-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
->param('name', '', new Text(128), 'Provider name.')
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
->param('accountSid', '', new Text(0), 'Twilio account secret ID.')
->param('authToken', '', new Text(0), 'Twilio authentication token.')
->inject('dbForProject')
->inject('response')
->action(function (string $providerId, string $name, bool $default, bool $enabled, string $accountSid, string $authToken, Database $dbForProject, Response $response) {
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
$provider = new Document([
'$id' => $providerId,
'name' => $name,
'provider' => 'twilio',
'type' => 'sms',
'search' => $providerId . ' ' . $name . ' ' . 'twilio' . ' ' . 'sms',
'default' => $default,
'enabled' => $enabled,
'credentials' => [
'accountSid' => $accountSid,
'authToken' => $authToken,
],
]);
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-06 00:27:48 +13:00
Query::equal('default', [true]),
Query::equal('type', ['sms'])
2023-10-04 23:45:59 +13:00
]))
) {
$provider->setAttribute('default', true);
2023-08-24 21:01:22 +12:00
}
2023-10-04 23:45:59 +13:00
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
2023-10-04 23:45:59 +13:00
}
2023-08-24 21:01:22 +12:00
$response
2023-10-04 23:45:59 +13:00
->setStatusCode(Response::STATUS_CODE_CREATED)
2023-08-24 21:01:22 +12:00
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
2023-10-04 23:45:59 +13:00
App::post('/v1/messaging/providers/vonage')
->desc('Create Vonage Provider')
2023-08-24 21:01:22 +12:00
->groups(['api', 'messaging'])
->label('audits.event', 'providers.create')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'createVonageProvider')
->label('sdk.description', '/docs/references/messaging/create-vonage-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.')
2023-08-30 04:59:48 +12:00
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
2023-10-04 23:45:59 +13:00
->param('apiKey', '', new Text(0), 'Vonage API key.')
->param('apiSecret', '', new Text(0), 'Vonage API secret.')
2023-08-24 21:01:22 +12:00
->inject('dbForProject')
->inject('response')
2023-10-04 23:45:59 +13:00
->action(function (string $providerId, string $name, bool $default, bool $enabled, string $apiKey, string $apiSecret, Database $dbForProject, Response $response) {
2023-08-25 11:16:15 +12:00
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
2023-08-30 04:59:48 +12:00
$provider = new Document([
2023-08-25 11:16:15 +12:00
'$id' => $providerId,
'name' => $name,
2023-10-04 23:45:59 +13:00
'provider' => 'vonage',
2023-08-24 21:01:22 +12:00
'type' => 'sms',
2023-10-04 23:45:59 +13:00
'search' => $providerId . ' ' . $name . ' ' . 'vonage' . ' ' . 'sms',
2023-08-30 04:59:48 +12:00
'default' => $default,
'enabled' => $enabled,
2023-08-24 21:01:22 +12:00
'credentials' => [
'apiKey' => $apiKey,
2023-10-04 23:45:59 +13:00
'apiSecret' => $apiSecret,
2023-08-24 21:01:22 +12:00
],
2023-08-30 04:59:48 +12:00
]);
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-06 00:27:48 +13:00
Query::equal('default', [true]),
2023-10-04 23:45:59 +13:00
Query::equal('type', ['sms'])
2023-08-30 04:59:48 +12:00
]))
) {
$provider->setAttribute('default', true);
}
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
2023-08-30 04:59:48 +12:00
}
2023-08-24 21:01:22 +12:00
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-10-04 23:45:59 +13:00
App::post('/v1/messaging/providers/fcm')
->desc('Create FCM Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.create')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'createFcmProvider')
2023-10-04 23:45:59 +13:00
->label('sdk.description', '/docs/references/messaging/create-fcm-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
->param('name', '', new Text(128), 'Provider name.')
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
->param('serverKey', '', new Text(0), 'FCM Server Key.')
->inject('dbForProject')
->inject('response')
->action(function (string $providerId, string $name, bool $default, bool $enabled, string $serverKey, Database $dbForProject, Response $response) {
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
$provider = new Document([
'$id' => $providerId,
'name' => $name,
'provider' => 'fcm',
'type' => 'push',
'search' => $providerId . ' ' . $name . ' ' . 'fcm' . ' ' . 'push',
'default' => $default,
'enabled' => $enabled,
'credentials' => [
'serverKey' => $serverKey,
2023-08-24 21:01:22 +12:00
],
2023-08-30 04:59:48 +12:00
]);
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-06 00:27:48 +13:00
Query::equal('default', [true]),
Query::equal('type', ['push'])
2023-08-30 04:59:48 +12:00
]))
) {
$provider->setAttribute('default', true);
}
2023-10-04 23:45:59 +13:00
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
2023-10-04 23:45:59 +13:00
}
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($provider, Response::MODEL_PROVIDER);
});
App::post('/v1/messaging/providers/apns')
->desc('Create APNS Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.create')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'createApnsProvider')
2023-10-04 23:45:59 +13:00
->label('sdk.description', '/docs/references/messaging/create-apns-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
->param('providerId', '', new CustomId(), 'Provider ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
->param('name', '', new Text(128), 'Provider name.')
->param('default', false, new Boolean(), 'Set as default provider.', true)
->param('enabled', true, new Boolean(), 'Set as enabled.', true)
->param('authKey', '', new Text(0), 'APNS authentication key.')
->param('authKeyId', '', new Text(0), 'APNS authentication key ID.')
->param('teamId', '', new Text(0), 'APNS team ID.')
->param('bundleId', '', new Text(0), 'APNS bundle ID.')
->param('endpoint', '', new Text(0), 'APNS endpoint.')
->inject('dbForProject')
->inject('response')
->action(function (string $providerId, string $name, bool $default, bool $enabled, string $authKey, string $authKeyId, string $teamId, string $bundleId, string $endpoint, Database $dbForProject, Response $response) {
$providerId = $providerId == 'unique()' ? ID::unique() : $providerId;
$provider = new Document([
'$id' => $providerId,
'name' => $name,
'provider' => 'apns',
'type' => 'push',
'search' => $providerId . ' ' . $name . ' ' . 'apns' . ' ' . 'push',
'default' => $default,
'enabled' => $enabled,
'credentials' => [
'authKey' => $authKey,
'authKeyId' => $authKeyId,
'teamId' => $teamId,
'bundleId' => $bundleId,
'endpoint' => $endpoint,
],
]);
// Check if a default provider exists, if not, set this one as default
if (
empty($dbForProject->findOne('providers', [
2023-10-06 00:27:48 +13:00
Query::equal('default', [true]),
Query::equal('type', ['push'])
2023-10-04 23:45:59 +13:00
]))
) {
$provider->setAttribute('default', true);
}
try {
$provider = $dbForProject->createDocument('providers', $provider);
} catch (DuplicateException) {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_ALREADY_EXISTS);
2023-10-04 23:45:59 +13:00
}
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($provider, Response::MODEL_PROVIDER);
});
App::get('/v1/messaging/providers')
->desc('List Providers')
->groups(['api', 'messaging'])
->label('scope', 'providers.read')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'listProviders')
->label('sdk.description', '/docs/references/messaging/list-providers.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER_LIST)
->param('queries', [], new Providers(), 'Array of query strings generated using the Query class provided by the SDK. [Learn more about queries](https://appwrite.io/docs/queries). Maximum of ' . APP_LIMIT_ARRAY_PARAMS_SIZE . ' queries are allowed, each ' . APP_LIMIT_ARRAY_ELEMENT_SIZE . ' characters long. You may filter on the following attributes: ' . implode(', ', Providers::ALLOWED_ATTRIBUTES), true)
->inject('dbForProject')
->inject('response')
->action(function (array $queries, Database $dbForProject, Response $response) {
$queries = Query::parseQueries($queries);
// Get cursor document if there was a cursor query
$cursor = Query::getByType($queries, [Query::TYPE_CURSORAFTER, Query::TYPE_CURSORBEFORE]);
$cursor = reset($cursor);
if ($cursor) {
$providerId = $cursor->getValue();
2023-10-18 06:23:26 +13:00
$cursorDocument = Authorization::skip(fn () => $dbForProject->getDocument('providers', $providerId));
2023-10-04 23:45:59 +13:00
2023-10-18 06:23:26 +13:00
if ($cursorDocument->isEmpty()) {
2023-10-04 23:45:59 +13:00
throw new Exception(Exception::GENERAL_CURSOR_NOT_FOUND, "Provider '{$providerId}' for the 'cursor' value not found.");
}
2023-10-07 02:53:46 +13:00
$cursor->setValue($cursorDocument);
2023-10-04 23:45:59 +13:00
}
$filterQueries = Query::groupByType($queries)['filters'];
$response->dynamic(new Document([
'total' => $dbForProject->count('providers', $filterQueries, APP_LIMIT_COUNT),
'providers' => $dbForProject->find('providers', $queries),
]), Response::MODEL_PROVIDER_LIST);
});
2023-10-06 11:00:47 +13:00
App::get('/v1/messaging/providers/:providerId')
2023-10-04 23:45:59 +13:00
->desc('Get Provider')
->groups(['api', 'messaging'])
->label('scope', 'providers.read')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'getProvider')
->label('sdk.description', '/docs/references/messaging/get-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-06 11:00:47 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-10-04 23:45:59 +13:00
->inject('dbForProject')
->inject('response')
2023-10-06 11:00:47 +13:00
->action(function (string $providerId, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-10-04 23:45:59 +13:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$response->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/mailgun/:providerId')
2023-10-04 23:45:59 +13:00
->desc('Update Mailgun Provider')
2023-08-24 21:01:22 +12:00
->groups(['api', 'messaging'])
->label('audits.event', 'providers.update')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'updateMailgunProvider')
->label('sdk.description', '/docs/references/messaging/update-mailgun-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.', true)
2023-09-20 22:25:22 +12:00
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
2023-10-04 23:45:59 +13:00
->param('isEuRegion', null, new Boolean(), 'Set as eu region.', true)
->param('from', '', new Text(256), 'Sender Email Address.', true)
->param('apiKey', '', new Text(0), 'Mailgun API Key.', true)
->param('domain', '', new Text(0), 'Mailgun Domain.', true)
2023-08-24 21:01:22 +12:00
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, ?bool $isEuRegion, string $from, string $apiKey, string $domain, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-08-24 21:01:22 +12:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
2023-08-24 21:01:22 +12:00
$providerAttr = $provider->getAttribute('provider');
2023-10-04 23:45:59 +13:00
if ($providerAttr !== 'mailgun') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
2023-08-24 21:01:22 +12:00
}
2023-09-20 22:25:22 +12:00
if (!empty($name)) {
2023-08-24 21:01:22 +12:00
$provider->setAttribute('name', $name);
2023-10-04 23:45:59 +13:00
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'mailgun' . ' ' . 'email');
}
if (!empty($from)) {
$provider->setAttribute('options', [
'from' => $from,
]);
2023-08-24 21:01:22 +12:00
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
2023-10-04 23:45:59 +13:00
$credentials = $provider->getAttribute('credentials');
if ($isEuRegion === true || $isEuRegion === false) {
$credentials['isEuRegion'] = $isEuRegion;
}
if (!empty($apiKey)) {
$credentials['apiKey'] = $apiKey;
}
if (!empty($domain)) {
$credentials['domain'] = $domain;
2023-08-24 21:01:22 +12:00
}
2023-10-04 23:45:59 +13:00
$provider->setAttribute('credentials', $credentials);
2023-08-24 21:01:22 +12:00
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/sendgrid/:providerId')
2023-10-04 23:45:59 +13:00
->desc('Update Sendgrid Provider')
2023-08-24 21:01:22 +12:00
->groups(['api', 'messaging'])
2023-10-04 23:45:59 +13:00
->label('audits.event', 'providers.update')
2023-08-24 21:01:22 +12:00
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'updateSendgridProvider')
->label('sdk.description', '/docs/references/messaging/update-sendgrid-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
2023-08-24 21:01:22 +12:00
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-10-04 23:45:59 +13:00
->param('name', '', new Text(128), 'Provider name.', true)
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
->param('apiKey', '', new Text(0), 'Sendgrid API key.', true)
2023-08-24 21:01:22 +12:00
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, string $apiKey, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-10-04 23:45:59 +13:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$providerAttr = $provider->getAttribute('provider');
if ($providerAttr !== 'sendgrid') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
2023-10-04 23:45:59 +13:00
}
if (!empty($name)) {
$provider->setAttribute('name', $name);
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'sendgrid' . ' ' . 'email');
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
if (!empty($apiKey)) {
$provider->setAttribute('credentials', [
'apiKey' => $apiKey,
]);
}
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/msg91/:providerId')
2023-10-04 23:45:59 +13:00
->desc('Update Msg91 Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.update')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'updateMsg91Provider')
->label('sdk.description', '/docs/references/messaging/update-msg91-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-10-04 23:45:59 +13:00
->param('name', '', new Text(128), 'Provider name.', true)
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
->param('senderId', '', new Text(0), 'Msg91 Sender ID.', true)
->param('authKey', '', new Text(0), 'Msg91 Auth Key.', true)
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, string $senderId, string $authKey, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-10-04 23:45:59 +13:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$providerAttr = $provider->getAttribute('provider');
if ($providerAttr !== 'msg91') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
2023-10-04 23:45:59 +13:00
}
if (!empty($name)) {
$provider->setAttribute('name', $name);
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'msg91' . ' ' . 'sms');
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
$credentials = $provider->getAttribute('credentials');
if (!empty($senderId)) {
$credentials['senderId'] = $senderId;
}
if (!empty($authKey)) {
$credentials['authKey'] = $authKey;
}
$provider->setAttribute('credentials', $credentials);
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/telesign/:providerId')
2023-10-04 23:45:59 +13:00
->desc('Update Telesign Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.update')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'updateTelesignProvider')
->label('sdk.description', '/docs/references/messaging/update-telesign-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-10-04 23:45:59 +13:00
->param('name', '', new Text(128), 'Provider name.', true)
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
->param('username', '', new Text(0), 'Telesign username.', true)
->param('password', '', new Text(0), 'Telesign password.', true)
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, string $username, string $password, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-10-04 23:45:59 +13:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$providerAttr = $provider->getAttribute('provider');
if ($providerAttr !== 'telesign') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
2023-10-04 23:45:59 +13:00
}
if (!empty($name)) {
$provider->setAttribute('name', $name);
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'telesign' . ' ' . 'sms');
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
$credentials = $provider->getAttribute('credentials');
if (!empty($username)) {
$credentials['username'] = $username;
}
if (!empty($password)) {
$credentials['password'] = $password;
2023-08-30 04:59:48 +12:00
}
2023-10-04 23:45:59 +13:00
$provider->setAttribute('credentials', $credentials);
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
2023-08-24 21:01:22 +12:00
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/textmagic/:providerId')
2023-08-24 21:01:22 +12:00
->desc('Update Textmagic Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.update')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'updateTextmagicProvider')
->label('sdk.description', '/docs/references/messaging/update-textmagic-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.', true)
2023-09-20 22:25:22 +12:00
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
2023-08-24 21:01:22 +12:00
->param('username', '', new Text(0), 'Textmagic username.', true)
->param('apiKey', '', new Text(0), 'Textmagic apiKey.', true)
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, string $username, string $apiKey, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-08-24 21:01:22 +12:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
2023-08-24 21:01:22 +12:00
$providerAttr = $provider->getAttribute('provider');
2023-08-24 21:01:22 +12:00
if ($providerAttr !== 'text-magic') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
2023-08-24 21:01:22 +12:00
}
2023-09-20 22:25:22 +12:00
if (!empty($name)) {
2023-08-24 21:01:22 +12:00
$provider->setAttribute('name', $name);
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'textmagic' . ' ' . 'sms');
2023-08-24 21:01:22 +12:00
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
2023-10-04 23:45:59 +13:00
$credentials = $provider->getAttribute('credentials');
2023-08-30 04:59:48 +12:00
2023-10-04 23:45:59 +13:00
if (!empty($username)) {
$credentials['username'] = $username;
2023-08-30 04:59:48 +12:00
}
2023-10-04 23:45:59 +13:00
if (!empty($apiKey)) {
$credentials['apiKey'] = $apiKey;
2023-08-30 04:59:48 +12:00
}
2023-10-04 23:45:59 +13:00
$provider->setAttribute('credentials', $credentials);
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
2023-08-24 21:01:22 +12:00
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/twilio/:providerId')
2023-08-24 21:01:22 +12:00
->desc('Update Twilio Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.update')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'updateTwilioProvider')
->label('sdk.description', '/docs/references/messaging/update-twilio-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.', true)
2023-09-20 22:25:22 +12:00
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
2023-08-24 21:01:22 +12:00
->param('accountSid', null, new Text(0), 'Twilio account secret ID.', true)
->param('authToken', null, new Text(0), 'Twilio authentication token.', true)
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, string $accountSid, string $authToken, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-08-24 21:01:22 +12:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
2023-08-24 21:01:22 +12:00
$providerAttr = $provider->getAttribute('provider');
2023-08-24 21:01:22 +12:00
if ($providerAttr !== 'twilio') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
2023-08-24 21:01:22 +12:00
}
2023-09-20 22:25:22 +12:00
if (!empty($name)) {
2023-08-24 21:01:22 +12:00
$provider->setAttribute('name', $name);
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'twilio' . ' ' . 'sms');
2023-08-24 21:01:22 +12:00
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
2023-10-04 23:45:59 +13:00
$credentials = $provider->getAttribute('credentials');
2023-08-30 04:59:48 +12:00
2023-10-04 23:45:59 +13:00
if (!empty($accountSid)) {
$credentials['accountSid'] = $accountSid;
2023-08-30 04:59:48 +12:00
}
2023-10-04 23:45:59 +13:00
if (!empty($authToken)) {
$credentials['authToken'] = $authToken;
2023-08-30 04:59:48 +12:00
}
2023-10-04 23:45:59 +13:00
$provider->setAttribute('credentials', $credentials);
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
2023-08-24 21:01:22 +12:00
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/vonage/:providerId')
2023-08-24 21:01:22 +12:00
->desc('Update Vonage Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.update')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
2023-10-04 23:45:59 +13:00
->label('sdk.method', 'updateVonageProvider')
->label('sdk.description', '/docs/references/messaging/update-vonage-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.', true)
2023-09-20 22:25:22 +12:00
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
2023-08-24 21:01:22 +12:00
->param('apiKey', '', new Text(0), 'Vonage API key.', true)
->param('apiSecret', '', new Text(0), 'Vonage API secret.', true)
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, string $apiKey, string $apiSecret, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-08-24 21:01:22 +12:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$providerAttr = $provider->getAttribute('provider');
if ($providerAttr !== 'vonage') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
}
2023-09-20 22:25:22 +12:00
if (!empty($name)) {
2023-08-24 21:01:22 +12:00
$provider->setAttribute('name', $name);
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'vonage' . ' ' . 'sms');
2023-08-24 21:01:22 +12:00
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
2023-10-04 23:45:59 +13:00
$credentials = $provider->getAttribute('credentials');
2023-08-30 04:59:48 +12:00
2023-10-04 23:45:59 +13:00
if (!empty($apiKey)) {
$credentials['apiKey'] = $apiKey;
2023-08-30 04:59:48 +12:00
}
2023-10-04 23:45:59 +13:00
if (!empty($apiSecret)) {
$credentials['apiSecret'] = $apiSecret;
2023-08-30 04:59:48 +12:00
}
2023-10-04 23:45:59 +13:00
$provider->setAttribute('credentials', $credentials);
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
2023-08-24 21:01:22 +12:00
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/fcm/:providerId')
2023-08-24 21:01:22 +12:00
->desc('Update FCM Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.update')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'updateFcmProvider')
2023-10-04 23:45:59 +13:00
->label('sdk.description', '/docs/references/messaging/update-fcm-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.', true)
2023-09-20 22:25:22 +12:00
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
2023-08-24 21:01:22 +12:00
->param('serverKey', '', new Text(0), 'FCM Server Key.', true)
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, string $serverKey, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-08-24 21:01:22 +12:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$providerAttr = $provider->getAttribute('provider');
if ($providerAttr !== 'fcm') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
2023-08-24 21:01:22 +12:00
}
2023-09-20 22:25:22 +12:00
if (!empty($name)) {
2023-08-24 21:01:22 +12:00
$provider->setAttribute('name', $name);
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'fcm' . ' ' . 'push');
2023-08-24 21:01:22 +12:00
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
2023-10-04 23:45:59 +13:00
if (!empty($serverKey)) {
2023-08-24 21:01:22 +12:00
$provider->setAttribute('credentials', ['serverKey' => $serverKey]);
}
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
2023-10-07 02:53:46 +13:00
App::patch('/v1/messaging/providers/apns/:providerId')
2023-08-24 21:01:22 +12:00
->desc('Update APNS Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.update')
->label('audits.resource', 'providers/{response.$id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'updateApnsProvider')
2023-10-04 23:45:59 +13:00
->label('sdk.description', '/docs/references/messaging/update-apns-provider.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_PROVIDER)
2023-10-07 02:53:46 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-08-24 21:01:22 +12:00
->param('name', '', new Text(128), 'Provider name.', true)
2023-09-20 22:25:22 +12:00
->param('enabled', null, new Boolean(), 'Set as enabled.', true)
2023-08-24 21:01:22 +12:00
->param('authKey', '', new Text(0), 'APNS authentication key.', true)
->param('authKeyId', '', new Text(0), 'APNS authentication key ID.', true)
->param('teamId', '', new Text(0), 'APNS team ID.', true)
->param('bundleId', '', new Text(0), 'APNS bundle ID.', true)
->param('endpoint', '', new Text(0), 'APNS endpoint.', true)
->inject('dbForProject')
->inject('response')
2023-10-07 02:53:46 +13:00
->action(function (string $providerId, string $name, ?bool $enabled, string $authKey, string $authKeyId, string $teamId, string $bundleId, string $endpoint, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-08-24 21:01:22 +12:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
2023-08-24 21:01:22 +12:00
$providerAttr = $provider->getAttribute('provider');
2023-08-24 21:01:22 +12:00
if ($providerAttr !== 'apns') {
2023-10-06 00:27:48 +13:00
throw new Exception(Exception::PROVIDER_INCORRECT_TYPE);
2023-08-24 21:01:22 +12:00
}
2023-09-20 22:25:22 +12:00
if (!empty($name)) {
2023-08-24 21:01:22 +12:00
$provider->setAttribute('name', $name);
$provider->setAttribute('search', $provider->getId() . ' ' . $name . ' ' . 'apns' . ' ' . 'push');
2023-08-24 21:01:22 +12:00
}
if ($enabled === true || $enabled === false) {
$provider->setAttribute('enabled', $enabled);
}
2023-10-04 23:45:59 +13:00
$credentials = $provider->getAttribute('credentials');
if (!empty($authKey)) {
$credentials['authKey'] = $authKey;
}
if (!empty($authKeyId)) {
$credentials['authKeyId'] = $authKeyId;
}
if (!empty($teamId)) {
$credentials['teamId'] = $teamId;
}
if (!empty($bundleId)) {
$credentials['bundle'] = $bundleId;
}
if (!empty($endpoint)) {
$credentials['endpoint'] = $endpoint;
2023-08-24 21:01:22 +12:00
}
2023-10-04 23:45:59 +13:00
$provider->setAttribute('credentials', $credentials);
2023-08-24 21:01:22 +12:00
$provider = $dbForProject->updateDocument('providers', $provider->getId(), $provider);
$response
->dynamic($provider, Response::MODEL_PROVIDER);
});
2023-08-23 04:45:31 +12:00
2023-10-06 11:00:47 +13:00
App::delete('/v1/messaging/providers/:providerId')
2023-08-24 21:01:22 +12:00
->desc('Delete Provider')
->groups(['api', 'messaging'])
->label('audits.event', 'providers.delete')
->label('audits.resource', 'providers/{request.id}')
->label('scope', 'providers.write')
2023-09-21 22:48:20 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-24 21:01:22 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'deleteProvider')
->label('sdk.description', '/docs/references/messaging/delete-provider.md')
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_NONE)
2023-10-06 11:00:47 +13:00
->param('providerId', '', new UID(), 'Provider ID.')
2023-08-24 21:01:22 +12:00
->inject('dbForProject')
->inject('response')
2023-10-06 11:00:47 +13:00
->action(function (string $providerId, Database $dbForProject, Response $response) {
$provider = $dbForProject->getDocument('providers', $providerId);
2023-08-24 21:01:22 +12:00
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$dbForProject->deleteDocument('providers', $provider->getId());
$response
->setStatusCode(Response::STATUS_CODE_NOCONTENT)
->noContent();
2023-08-24 21:01:22 +12:00
});
2023-08-23 04:45:31 +12:00
App::post('/v1/messaging/topics')
->desc('Create a topic.')
->groups(['api', 'messaging'])
->label('audits.event', 'topics.create')
->label('audits.resource', 'topics/{response.$id}')
->label('scope', 'topics.write')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'createTopic')
->label('sdk.description', '/docs/references/messaging/create-topic.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_TOPIC)
->param('topicId', '', new CustomId(), 'Topic ID. Choose a custom Topic ID or a new Topic ID.')
->param('providerId', '', new UID(), 'Provider ID.')
->param('name', '', new Text(128), 'Topic Name.')
->param('description', '', new Text(2048), 'Topic Description.', true)
->inject('dbForProject')
->inject('response')
->action(function (string $topicId, string $providerId, string $name, string $description, Database $dbForProject, Response $response) {
$topicId = $topicId == 'unique()' ? ID::unique() : $topicId;
$provider = $dbForProject->getDocument('providers', $providerId);
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$topic = new Document([
'$id' => $topicId,
'providerId' => $providerId,
'providerInternalId' => $provider->getInternalId(),
'name' => $name,
]);
if ($description) {
$topic->setAttribute('description', $description);
}
try {
$topic = $dbForProject->createDocument('topics', $topic);
} catch (DuplicateException) {
throw new Exception(Exception::TOPIC_ALREADY_EXISTS);
}
2023-10-14 01:56:54 +13:00
$response
2023-10-18 06:23:26 +13:00
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($topic, Response::MODEL_TOPIC);
});
2023-08-25 08:15:17 +12:00
App::get('/v1/messaging/topics')
->desc('List topics.')
->groups(['api', 'messaging'])
->label('scope', 'topics.read')
2023-08-25 11:06:41 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-25 08:15:17 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'listTopics')
->label('sdk.description', '/docs/references/messaging/list-topics.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_TOPIC_LIST)
->param('queries', [], new Topics(), 'Array of query strings generated using the Query class provided by the SDK. [Learn more about queries](https://appwrite.io/docs/queries). Maximum of ' . APP_LIMIT_ARRAY_PARAMS_SIZE . ' queries are allowed, each ' . APP_LIMIT_ARRAY_ELEMENT_SIZE . ' characters long. You may filter on the following attributes: ' . implode(', ', Topics::ALLOWED_ATTRIBUTES), true)
->inject('dbForProject')
->inject('response')
->action(function (array $queries, Database $dbForProject, Response $response) {
$queries = Query::parseQueries($queries);
// Get cursor document if there was a cursor query
$cursor = Query::getByType($queries, [Query::TYPE_CURSORAFTER, Query::TYPE_CURSORBEFORE]);
$cursor = reset($cursor);
if ($cursor) {
$topicId = $cursor->getValue();
2023-10-18 06:23:26 +13:00
$cursorDocument = Authorization::skip(fn () => $dbForProject->getDocument('topics', $topicId));
2023-08-25 08:15:17 +12:00
2023-10-18 06:23:26 +13:00
if ($cursorDocument->isEmpty()) {
2023-08-25 08:15:17 +12:00
throw new Exception(Exception::GENERAL_CURSOR_NOT_FOUND, "Topic '{$topicId}' for the 'cursor' value not found.");
}
$cursor->setValue($cursorDocument[0]);
}
$filterQueries = Query::groupByType($queries)['filters'];
$response->dynamic(new Document([
'total' => $dbForProject->count('topics', $filterQueries, APP_LIMIT_COUNT),
2023-08-25 11:06:41 +12:00
'topics' => $dbForProject->find('topics', $queries),
2023-08-25 08:15:17 +12:00
]), Response::MODEL_TOPIC_LIST);
});
2023-08-29 03:11:05 +12:00
App::get('/v1/messaging/topics/:topicId')
2023-08-25 08:15:17 +12:00
->desc('Get a topic.')
->groups(['api', 'messaging'])
->label('scope', 'topics.read')
2023-08-25 11:06:41 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-25 08:15:17 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'getTopic')
->label('sdk.description', '/docs/references/messaging/get-topic.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_TOPIC)
2023-08-29 03:11:05 +12:00
->param('topicId', '', new UID(), 'Topic ID.')
2023-08-25 08:15:17 +12:00
->inject('dbForProject')
->inject('response')
2023-08-29 03:11:05 +12:00
->action(function (string $topicId, Database $dbForProject, Response $response) {
$topic = $dbForProject->getDocument('topics', $topicId);
2023-08-25 08:15:17 +12:00
if ($topic->isEmpty()) {
throw new Exception(Exception::TOPIC_NOT_FOUND);
}
2023-08-29 03:11:05 +12:00
$topic = $dbForProject->getDocument('topics', $topicId);
2023-08-25 08:15:17 +12:00
$response
->dynamic($topic, Response::MODEL_TOPIC);
});
2023-08-25 11:06:41 +12:00
App::patch('/v1/messaging/topics/:topicId')
2023-08-25 08:15:17 +12:00
->desc('Update a topic.')
->groups(['api', 'messaging'])
->label('audits.event', 'topics.update')
->label('audits.resource', 'topics/{response.$id}')
->label('scope', 'topics.write')
2023-08-25 11:06:41 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-25 08:15:17 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'updateTopic')
->label('sdk.description', '/docs/references/messaging/update-topic.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_TOPIC)
2023-08-25 11:06:41 +12:00
->param('topicId', '', new UID(), 'Topic ID.')
2023-08-25 08:15:17 +12:00
->param('name', '', new Text(128), 'Topic Name.', true)
2023-10-18 06:23:26 +13:00
->param('description', '', new Text(2048), 'Topic Description.', true)
2023-08-25 08:15:17 +12:00
->inject('dbForProject')
->inject('response')
2023-08-25 11:06:41 +12:00
->action(function (string $topicId, string $name, string $description, Database $dbForProject, Response $response) {
$topic = $dbForProject->getDocument('topics', $topicId);
2023-08-25 08:15:17 +12:00
if ($topic->isEmpty()) {
throw new Exception(Exception::TOPIC_NOT_FOUND);
}
2023-10-18 06:23:26 +13:00
if (!empty($name)) {
2023-08-25 08:15:17 +12:00
$topic->setAttribute('name', $name);
}
2023-10-18 06:23:26 +13:00
if (!empty($description)) {
2023-08-25 08:15:17 +12:00
$topic->setAttribute('description', $description);
}
2023-08-25 11:06:41 +12:00
$topic = $dbForProject->updateDocument('topics', $topicId, $topic);
2023-08-25 08:15:17 +12:00
$response
->dynamic($topic, Response::MODEL_TOPIC);
});
2023-08-25 11:06:41 +12:00
App::delete('/v1/messaging/topics/:topicId')
2023-08-25 08:15:17 +12:00
->desc('Delete a topic.')
->groups(['api', 'messaging'])
->label('audits.event', 'topics.delete')
2023-08-25 11:06:41 +12:00
->label('audits.resource', 'topics/{request.topicId}')
2023-08-25 08:15:17 +12:00
->label('scope', 'topics.write')
2023-08-25 11:06:41 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-25 08:15:17 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'deleteTopic')
->label('sdk.description', '/docs/references/messaging/delete-topic.md')
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_NONE)
2023-08-25 11:06:41 +12:00
->param('topicId', '', new UID(), 'Topic ID.')
2023-08-25 08:15:17 +12:00
->inject('dbForProject')
2023-10-18 06:23:26 +13:00
->inject('deletes')
2023-08-25 08:15:17 +12:00
->inject('response')
2023-10-18 06:23:26 +13:00
->action(function (string $topicId, Database $dbForProject, Delete $deletes, Response $response) {
2023-08-25 11:06:41 +12:00
$topic = $dbForProject->getDocument('topics', $topicId);
2023-08-25 08:15:17 +12:00
if ($topic->isEmpty()) {
throw new Exception(Exception::TOPIC_NOT_FOUND);
}
2023-10-18 06:23:26 +13:00
$dbForProject->deleteDocument('topics', $topicId);
$deletes
->setType(DELETE_TYPE_SUBSCRIBERS)
->setDocument($topic);
$response
->setStatusCode(Response::STATUS_CODE_NOCONTENT)
->noContent();
2023-08-25 08:15:17 +12:00
});
App::post('/v1/messaging/topics/:topicId/subscribers')
->desc('Adds a Subscriber to a Topic.')
->groups(['api', 'messaging'])
->label('audits.event', 'subscribers.create')
->label('audits.resource', 'subscribers/{response.$id}')
->label('scope', 'subscribers.write')
->label('sdk.auth', [APP_AUTH_TYPE_JWT, APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
2023-10-14 01:56:54 +13:00
->label('sdk.method', 'createSubscriber')
->label('sdk.description', '/docs/references/messaging/create-subscriber.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_SUBSCRIBER)
->param('subscriberId', '', new CustomId(), 'Subscriber ID. Choose a custom Topic ID or a new Topic ID.')
->param('topicId', '', new UID(), 'Topic ID.')
->param('targetId', '', new UID(), 'Target ID.')
->inject('dbForProject')
->inject('response')
->action(function (string $subscriberId, string $topicId, string $targetId, Database $dbForProject, Response $response) {
$subscriberId = $subscriberId == 'unique()' ? ID::unique() : $subscriberId;
$topic = Authorization::skip(fn () => $dbForProject->getDocument('topics', $topicId));
if ($topic->isEmpty()) {
throw new Exception(Exception::TOPIC_NOT_FOUND);
}
$target = Authorization::skip(fn () => $dbForProject->getDocument('targets', $targetId));
if ($target->isEmpty()) {
throw new Exception(Exception::USER_TARGET_NOT_FOUND);
}
$subscriber = new Document([
'$id' => $subscriberId,
'$permissions' => [
Permission::read(Role::user($target->getAttribute('userId'))),
Permission::delete(Role::user($target->getAttribute('userId'))),
],
'topicId' => $topicId,
'topicInternalId' => $topic->getInternalId(),
'targetId' => $targetId,
'targetInternalId' => $target->getInternalId(),
]);
try {
$subscriber = $dbForProject->createDocument('subscribers', $subscriber);
$dbForProject->deleteCachedDocument('topics', $topicId);
} catch (DuplicateException) {
throw new Exception(Exception::SUBSCRIBER_ALREADY_EXISTS);
}
2023-10-14 01:56:54 +13:00
$response
2023-10-18 06:23:26 +13:00
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($subscriber, Response::MODEL_SUBSCRIBER);
});
2023-08-29 03:11:05 +12:00
App::get('/v1/messaging/topics/:topicId/subscribers')
->desc('List topic\'s subscribers.')
->groups(['api', 'messaging'])
->label('scope', 'subscribers.read')
2023-10-14 01:56:54 +13:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-29 03:11:05 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'listSubscribers')
->label('sdk.description', '/docs/references/messaging/list-subscribers.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_SUBSCRIBER_LIST)
->param('topicId', '', new UID(), 'Topic ID.')
->inject('dbForProject')
->inject('response')
->action(function (string $topicId, Database $dbForProject, Response $response) {
$topic = Authorization::skip(fn () => $dbForProject->getDocument('topics', $topicId));
if ($topic->isEmpty()) {
throw new Exception(Exception::TOPIC_NOT_FOUND);
}
$subscribers = $dbForProject->find('subscribers', [
Query::equal('topicInternalId', [$topic->getInternalId()])
]);
2023-08-29 03:13:18 +12:00
2023-08-29 03:11:05 +12:00
$response
->dynamic(new Document([
'subscribers' => $subscribers,
'total' => \count($subscribers),
]), Response::MODEL_SUBSCRIBER_LIST);
2023-08-29 03:13:18 +12:00
});
2023-08-29 03:11:05 +12:00
App::get('/v1/messaging/topics/:topicId/subscriber/:subscriberId')
->desc('Get a topic\'s subscriber.')
->groups(['api', 'messaging'])
->label('scope', 'subscribers.read')
2023-10-14 01:56:54 +13:00
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
2023-08-29 03:11:05 +12:00
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'getSubscriber')
->label('sdk.description', '/docs/references/messaging/get-subscriber.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_SUBSCRIBER)
->param('topicId', '', new UID(), 'Topic ID.')
->param('subscriberId', '', new UID(), 'Subscriber ID.')
->inject('dbForProject')
->inject('response')
->action(function (string $topicId, string $subscriberId, Database $dbForProject, Response $response) {
$topic = Authorization::skip(fn () => $dbForProject->getDocument('topics', $topicId));
if ($topic->isEmpty()) {
throw new Exception(Exception::TOPIC_NOT_FOUND);
}
2023-08-29 03:13:18 +12:00
2023-08-29 03:11:05 +12:00
$subscriber = $dbForProject->getDocument('subscribers', $subscriberId);
2023-08-29 03:13:18 +12:00
if ($subscriber->isEmpty() || $subscriber->getAttribute('topicId') !== $topicId) {
2023-08-29 03:11:05 +12:00
throw new Exception(Exception::SUBSCRIBER_NOT_FOUND);
}
$response
->dynamic($subscriber, Response::MODEL_SUBSCRIBER);
});
App::delete('/v1/messaging/topics/:topicId/subscriber/:subscriberId')
->desc('Delete a Subscriber from a Topic.')
->groups(['api', 'messaging'])
->label('audits.event', 'subscribers.delete')
->label('audits.resource', 'subscribers/{request.subscriberId}')
->label('scope', 'subscribers.write')
->label('sdk.auth', [APP_AUTH_TYPE_JWT, APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'deleteSubscriber')
->label('sdk.description', '/docs/references/messaging/delete-subscriber.md')
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_NONE)
->param('topicId', '', new UID(), 'Topic ID.')
->param('subscriberId', '', new UID(), 'Subscriber ID.')
->inject('dbForProject')
->inject('response')
->action(function (string $topicId, string $subscriberId, Database $dbForProject, Response $response) {
$topic = Authorization::skip(fn () => $dbForProject->getDocument('topics', $topicId));
if ($topic->isEmpty()) {
throw new Exception(Exception::TOPIC_NOT_FOUND);
}
$subscriber = $dbForProject->getDocument('subscribers', $subscriberId);
if ($subscriber->isEmpty() || $subscriber->getAttribute('topicId') !== $topicId) {
throw new Exception(Exception::SUBSCRIBER_NOT_FOUND);
}
$subscriber = $dbForProject->deleteDocument('subscribers', $subscriberId);
$dbForProject->deleteCachedDocument('topics', $topicId);
$response
->setStatusCode(Response::STATUS_CODE_NOCONTENT)
->noContent();
});
App::post('/v1/messaging/messages/email')
2023-10-14 01:56:54 +13:00
->desc('Create an email.')
2023-08-24 21:01:22 +12:00
->groups(['api', 'messaging'])
->label('audits.event', 'messages.create')
->label('audits.resource', 'messages/{response.$id}')
->label('scope', 'messages.write')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
2023-10-14 01:56:54 +13:00
->label('sdk.method', 'createEmail')
->label('sdk.description', '/docs/references/messaging/create-email.md')
2023-08-24 21:01:22 +12:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_MESSAGE)
->param('messageId', '', new CustomId(), 'Message ID. Choose a custom ID or generate a random ID with `ID.unique()`. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
2023-10-04 23:45:59 +13:00
->param('providerId', '', new UID(), 'Email Provider ID.')
2023-10-04 23:54:29 +13:00
->param('to', [], new ArrayList(new Text(Database::LENGTH_KEY)), 'List of Topic IDs or List of User IDs or List of Target IDs.')
2023-10-04 23:45:59 +13:00
->param('subject', '', new Text(998), 'Email Subject.')
->param('description', '', new Text(256), 'Description for Message.', true)
2023-10-06 00:27:48 +13:00
->param('content', '', new Text(64230), 'Email Content.')
2023-10-07 02:53:46 +13:00
->param('status', 'processing', new WhiteList(['draft', 'processing']), 'Message Status.', true)
2023-10-04 23:45:59 +13:00
->param('html', false, new Boolean(), 'Is content of type HTML', true)
2023-10-18 06:23:26 +13:00
->param('deliveryTime', null, new DatetimeValidator(false), 'Delivery time for message.', true)
2023-08-24 21:01:22 +12:00
->inject('dbForProject')
2023-09-06 22:10:56 +12:00
->inject('project')
->inject('messaging')
2023-08-24 21:01:22 +12:00
->inject('response')
2023-10-14 01:56:54 +13:00
->action(function (string $messageId, string $providerId, array $to, string $subject, string $description, string $content, string $status, bool $html, ?string $deliveryTime, Database $dbForProject, Document $project, Messaging $messaging, Response $response) {
$messageId = $messageId == 'unique()' ? ID::unique() : $messageId;
2023-08-24 21:01:22 +12:00
$provider = $dbForProject->getDocument('providers', $providerId);
if ($provider->isEmpty()) {
throw new Exception(Exception::PROVIDER_NOT_FOUND);
}
$message = $dbForProject->createDocument('messages', new Document([
'$id' => $messageId,
2023-08-24 21:01:22 +12:00
'providerId' => $provider->getId(),
'providerInternalId' => $provider->getInternalId(),
'to' => $to,
'data' => [
'subject' => $subject,
'content' => $content,
'html' => $html,
'description' => $description,
2023-08-24 21:01:22 +12:00
],
2023-10-07 02:53:46 +13:00
'status' => $status,
'search' => $messageId . ' ' . $description . ' ' . $subject,
2023-08-24 21:01:22 +12:00
]));
2023-10-07 02:53:46 +13:00
if ($status === 'processing') {
$messaging
->setMessageId($message->getId())
2023-10-14 01:56:54 +13:00
->setProject($project);
if (!empty($deliveryTime)) {
$messaging
->setDeliveryTime($deliveryTime)
->schedule();
} else {
$messaging->trigger();
}
2023-10-07 02:53:46 +13:00
}
2023-08-24 21:01:22 +12:00
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->dynamic($message, Response::MODEL_MESSAGE);
2023-08-24 21:01:22 +12:00
});
2023-10-04 23:45:59 +13:00
2023-10-07 02:53:46 +13:00
App::get('/v1/messaging/messages')
->desc('List Messages')
->groups(['api', 'messaging'])
->label('scope', 'messages.read')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'listMessages')
->label('sdk.description', '/docs/references/messaging/list-messages.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_MESSAGE_LIST)
2023-10-14 01:56:54 +13:00
->param('queries', [], new Messages(), 'Array of query strings generated using the Query class provided by the SDK. [Learn more about queries](https://appwrite.io/docs/queries). Maximum of ' . APP_LIMIT_ARRAY_PARAMS_SIZE . ' queries are allowed, each ' . APP_LIMIT_ARRAY_ELEMENT_SIZE . ' characters long. You may filter on the following attributes: ' . implode(', ', Providers::ALLOWED_ATTRIBUTES), true)
2023-10-07 02:53:46 +13:00
->inject('dbForProject')
->inject('response')
->action(function (array $queries, Database $dbForProject, Response $response) {
$queries = Query::parseQueries($queries);
// Get cursor document if there was a cursor query
$cursor = Query::getByType($queries, [Query::TYPE_CURSORAFTER, Query::TYPE_CURSORBEFORE]);
$cursor = reset($cursor);
if ($cursor) {
$messageId = $cursor->getValue();
2023-10-18 06:23:26 +13:00
$cursorDocument = Authorization::skip(fn () => $dbForProject->getDocument('messages', $messageId));
2023-10-07 02:53:46 +13:00
2023-10-18 06:23:26 +13:00
if ($cursorDocument->isEmpty()) {
2023-10-07 02:53:46 +13:00
throw new Exception(Exception::GENERAL_CURSOR_NOT_FOUND, "Message '{$messageId}' for the 'cursor' value not found.");
}
$cursor->setValue($cursorDocument);
}
$filterQueries = Query::groupByType($queries)['filters'];
$response->dynamic(new Document([
'total' => $dbForProject->count('messages', $filterQueries, APP_LIMIT_COUNT),
'messages' => $dbForProject->find('messages', $queries),
]), Response::MODEL_MESSAGE_LIST);
});
2023-10-06 11:00:47 +13:00
App::get('/v1/messaging/messages/:messageId')
2023-10-04 23:45:59 +13:00
->desc('Get Message')
->groups(['api', 'messaging'])
->label('scope', 'messages.read')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'getMessage')
->label('sdk.description', '/docs/references/messaging/get-message.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_MESSAGE)
2023-10-06 11:00:47 +13:00
->param('messageId', '', new UID(), 'Message ID.')
2023-10-04 23:45:59 +13:00
->inject('dbForProject')
->inject('response')
2023-10-06 11:00:47 +13:00
->action(function (string $messageId, Database $dbForProject, Response $response) {
$message = $dbForProject->getDocument('messages', $messageId);
2023-10-04 23:45:59 +13:00
if ($message->isEmpty()) {
throw new Exception(Exception::MESSAGE_NOT_FOUND);
}
$response->dynamic($message, Response::MODEL_MESSAGE);
});
2023-10-07 02:53:46 +13:00
2023-10-12 19:18:26 +13:00
App::patch('/v1/messaging/messages/email/:messageId')
2023-10-07 02:53:46 +13:00
->desc('Update an email.')
->groups(['api', 'messaging'])
->label('audits.event', 'messages.update')
->label('audits.resource', 'messages/{response.$id}')
->label('scope', 'messages.write')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN, APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'messaging')
->label('sdk.method', 'updateEmail')
->label('sdk.description', '/docs/references/messaging/update-email.md')
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_MESSAGE)
->param('messageId', '', new UID(), 'Message ID.')
->param('to', [], new ArrayList(new Text(Database::LENGTH_KEY)), 'List of Topic IDs or List of User IDs or List of Target IDs.', true)
->param('subject', '', new Text(998), 'Email Subject.', true)
->param('description', '', new Text(256), 'Description for Message.', true)
->param('content', '', new Text(64230), 'Email Content.', true)
->param('status', '', new WhiteList(['draft', 'processing']), 'Message Status.', true)
->param('html', false, new Boolean(), 'Is content of type HTML', true)
2023-10-14 01:56:54 +13:00
->param('deliveryTime', null, new DatetimeValidator(), 'Delivery time for message in ISO 8601 format.', true)
2023-10-07 02:53:46 +13:00
->inject('dbForProject')
->inject('project')
->inject('messaging')
->inject('response')
2023-10-14 01:56:54 +13:00
->action(function (string $messageId, array $to, string $subject, string $description, string $content, string $status, bool $html, ?string $deliveryTime, Database $dbForProject, Document $project, Messaging $messaging, Response $response) {
2023-10-07 02:53:46 +13:00
$message = $dbForProject->getDocument('messages', $messageId);
if ($message->isEmpty()) {
throw new Exception(Exception::MESSAGE_NOT_FOUND);
}
if ($message->getAttribute('status') === 'sent') {
throw new Exception(Exception::MESSAGE_ALREADY_SENT);
}
2023-10-07 02:53:46 +13:00
if (\count($to) > 0) {
$message->setAttribute('to', $to);
}
$data = $message->getAttribute('data');
if (!empty($subject)) {
$data['subject'] = $subject;
}
if (!empty($content)) {
$data['content'] = $content;
}
if (!empty($description)) {
$data['description'] = $description;
}
if (!empty($html)) {
$data['html'] = $html;
}
$message->setAttribute('data', $data);
$message->setAttribute('search', $message->getId() . ' ' . $data['description'] . ' ' . $data['subject']);
if (!empty($status)) {
$message->setAttribute('status', $status);
}
$message = $dbForProject->updateDocument('messages', $message->getId(), $message);
if ($status === 'processing') {
$messaging
->setMessageId($message->getId())
2023-10-14 01:56:54 +13:00
->setProject($project);
if (!empty($deliveryTime)) {
$messaging
2023-10-10 00:59:26 +13:00
->setDeliveryTime($deliveryTime)
2023-10-14 01:56:54 +13:00
->schedule();
} else {
$messaging->trigger();
}
2023-10-07 02:53:46 +13:00
}
$response
->dynamic($message, Response::MODEL_MESSAGE);
});