1
0
Fork 0
mirror of synced 2024-06-01 18:39:57 +12:00
appwrite/app/init.php

1656 lines
59 KiB
PHP
Raw Normal View History

2019-05-09 18:54:39 +12:00
<?php
2020-03-25 17:34:06 +13:00
/**
* Init
2022-05-24 02:54:50 +12:00
*
2020-09-25 10:32:39 +12:00
* Initializes both Appwrite API entry point, queue workers, and CLI tasks.
2020-12-27 00:56:29 +13:00
* Set configuration, framework resources & app constants
2022-05-24 02:54:50 +12:00
*
2020-03-25 17:34:06 +13:00
*/
2022-05-24 02:54:50 +12:00
if (\file_exists(__DIR__ . '/../vendor/autoload.php')) {
require_once __DIR__ . '/../vendor/autoload.php';
2019-08-01 08:35:42 +12:00
}
2019-05-09 18:54:39 +12:00
2022-05-24 02:54:50 +12:00
ini_set('memory_limit', '512M');
2021-03-10 19:53:49 +13:00
ini_set('display_errors', 1);
ini_set('display_startup_errors', 1);
ini_set('default_socket_timeout', -1);
error_reporting(E_ALL);
2024-03-07 06:34:21 +13:00
use Ahc\Jwt\JWT;
use Ahc\Jwt\JWTException;
2020-07-01 06:08:02 +12:00
use Appwrite\Auth\Auth;
use Appwrite\Event\Audit;
2024-03-07 06:34:21 +13:00
use Appwrite\Event\Build;
use Appwrite\Event\Certificate;
use Appwrite\Event\Database as EventDatabase;
2024-03-07 06:34:21 +13:00
use Appwrite\Event\Delete;
2020-06-28 07:42:38 +12:00
use Appwrite\Event\Event;
2024-03-07 06:34:21 +13:00
use Appwrite\Event\Func;
use Appwrite\Event\Mail;
use Appwrite\Event\Messaging;
2024-03-07 06:34:21 +13:00
use Appwrite\Event\Migration;
use Appwrite\Event\Usage;
use Appwrite\Extend\Exception;
use Appwrite\GraphQL\Promises\Adapter\Swoole;
2022-10-12 14:04:11 +13:00
use Appwrite\GraphQL\Schema;
2024-03-07 06:34:21 +13:00
use Appwrite\Hooks\Hooks;
use Appwrite\Network\Validator\Email;
use Appwrite\Network\Validator\Origin;
2020-05-13 10:00:00 +12:00
use Appwrite\OpenSSL\OpenSSL;
2022-10-20 01:41:38 +13:00
use Appwrite\URL\URL as AppwriteURL;
2024-03-07 06:34:21 +13:00
use MaxMind\Db\Reader;
use PHPMailer\PHPMailer\PHPMailer;
use Swoole\Database\PDOProxy;
2022-07-13 21:34:56 +12:00
use Utopia\App;
use Utopia\Cache\Adapter\Redis as RedisCache;
2024-03-07 06:34:21 +13:00
use Utopia\Cache\Adapter\Sharding;
2022-07-13 21:34:56 +12:00
use Utopia\Cache\Cache;
2024-03-07 06:34:21 +13:00
use Utopia\CLI\Console;
2022-07-13 21:34:56 +12:00
use Utopia\Config\Config;
2024-03-07 06:34:21 +13:00
use Utopia\Database\Adapter\MariaDB;
use Utopia\Database\Adapter\MySQL;
use Utopia\Database\Adapter\SQL;
2022-07-13 21:34:56 +12:00
use Utopia\Database\Database;
use Utopia\Database\Document;
2024-03-07 06:34:21 +13:00
use Utopia\Database\Helpers\ID;
2021-08-20 23:15:17 +12:00
use Utopia\Database\Query;
2022-07-13 21:34:56 +12:00
use Utopia\Database\Validator\Authorization;
2023-03-13 03:04:18 +13:00
use Utopia\Database\Validator\Datetime as DatetimeValidator;
2022-07-13 21:34:56 +12:00
use Utopia\Database\Validator\Structure;
2024-03-07 06:34:21 +13:00
use Utopia\Domains\Validator\PublicDomain;
use Utopia\DSN\DSN;
2024-03-07 06:34:21 +13:00
use Utopia\Locale\Locale;
use Utopia\Logger\Log;
use Utopia\Logger\Logger;
use Utopia\Pools\Group;
use Utopia\Pools\Pool;
use Utopia\Queue;
use Utopia\Queue\Connection;
2022-07-13 21:34:56 +12:00
use Utopia\Registry\Registry;
2022-02-17 22:13:39 +13:00
use Utopia\Storage\Device;
2022-05-10 23:15:56 +12:00
use Utopia\Storage\Device\Backblaze;
2022-02-09 04:40:56 +13:00
use Utopia\Storage\Device\DOSpaces;
2022-07-13 21:34:56 +12:00
use Utopia\Storage\Device\Linode;
2021-12-13 20:34:15 +13:00
use Utopia\Storage\Device\Local;
use Utopia\Storage\Device\S3;
2022-03-19 06:17:43 +13:00
use Utopia\Storage\Device\Wasabi;
2022-07-13 21:34:56 +12:00
use Utopia\Storage\Storage;
2024-04-02 00:08:46 +13:00
use Utopia\System\System;
use Utopia\Validator\Hostname;
use Utopia\Validator\IP;
2024-03-07 06:34:21 +13:00
use Utopia\Validator\Range;
use Utopia\Validator\URL;
2022-07-13 21:34:56 +12:00
use Utopia\Validator\WhiteList;
2024-03-07 06:34:21 +13:00
use Utopia\VCS\Adapter\Git\GitHub as VcsGitHub;
2019-05-09 18:54:39 +12:00
2019-10-01 17:57:41 +13:00
const APP_NAME = 'Appwrite';
const APP_DOMAIN = 'appwrite.io';
2020-03-02 11:10:52 +13:00
const APP_EMAIL_TEAM = 'team@localhost.test'; // Default email address
2021-02-19 05:48:11 +13:00
const APP_EMAIL_SECURITY = ''; // Default security email address
2022-05-24 02:54:50 +12:00
const APP_USERAGENT = APP_NAME . '-Server v%s. Please report abuse at %s';
2020-11-19 11:08:45 +13:00
const APP_MODE_DEFAULT = 'default';
2019-10-01 17:57:41 +13:00
const APP_MODE_ADMIN = 'admin';
2020-07-20 02:43:34 +12:00
const APP_PAGING_LIMIT = 12;
2021-05-10 06:37:47 +12:00
const APP_LIMIT_COUNT = 5000;
2024-01-11 15:54:44 +13:00
const APP_LIMIT_USERS = 10_000;
const APP_LIMIT_USER_PASSWORD_HISTORY = 20;
const APP_LIMIT_USER_SESSIONS_MAX = 100;
const APP_LIMIT_USER_SESSIONS_DEFAULT = 10;
2024-01-11 15:54:44 +13:00
const APP_LIMIT_ANTIVIRUS = 20_000_000; //20MB
const APP_LIMIT_ENCRYPTION = 20_000_000; //20MB
const APP_LIMIT_COMPRESSION = 20_000_000; //20MB
2022-05-01 19:54:58 +12:00
const APP_LIMIT_ARRAY_PARAMS_SIZE = 100; // Default maximum of how many elements can there be in API parameter that expects array value
2024-01-28 23:51:05 +13:00
const APP_LIMIT_ARRAY_LABELS_SIZE = 1000; // Default maximum of how many labels elements can there be in API parameter that expects array value
2022-06-22 21:08:11 +12:00
const APP_LIMIT_ARRAY_ELEMENT_SIZE = 4096; // Default maximum length of element in array parameter represented by maximum URL length.
2022-05-31 00:01:39 +12:00
const APP_LIMIT_SUBQUERY = 1000;
2024-01-11 15:54:44 +13:00
const APP_LIMIT_SUBSCRIBERS_SUBQUERY = 1_000_000;
const APP_LIMIT_WRITE_RATE_DEFAULT = 60; // Default maximum write rate per rate period
const APP_LIMIT_WRITE_RATE_PERIOD_DEFAULT = 60; // Default maximum write rate period in seconds
2022-09-23 15:53:25 +12:00
const APP_LIMIT_LIST_DEFAULT = 25; // Default maximum number of items to return in list API calls
2022-08-09 17:56:53 +12:00
const APP_KEY_ACCCESS = 24 * 60 * 60; // 24 hours
const APP_USER_ACCCESS = 24 * 60 * 60; // 24 hours
2022-09-01 01:11:23 +12:00
const APP_CACHE_UPDATE = 24 * 60 * 60; // 24 hours
2024-04-25 08:00:20 +12:00
const APP_CACHE_BUSTER = 406;
2024-04-25 09:28:26 +12:00
const APP_VERSION_STABLE = '1.5.5';
2021-08-17 10:05:52 +12:00
const APP_DATABASE_ATTRIBUTE_EMAIL = 'email';
const APP_DATABASE_ATTRIBUTE_ENUM = 'enum';
2021-08-17 10:05:52 +12:00
const APP_DATABASE_ATTRIBUTE_IP = 'ip';
2022-07-25 20:53:41 +12:00
const APP_DATABASE_ATTRIBUTE_DATETIME = 'datetime';
2021-08-17 10:05:52 +12:00
const APP_DATABASE_ATTRIBUTE_URL = 'url';
const APP_DATABASE_ATTRIBUTE_INT_RANGE = 'intRange';
const APP_DATABASE_ATTRIBUTE_FLOAT_RANGE = 'floatRange';
2024-01-11 15:54:44 +13:00
const APP_DATABASE_ATTRIBUTE_STRING_MAX_LENGTH = 1_073_741_824; // 2^32 bits / 4 bits per char
const APP_DATABASE_TIMEOUT_MILLISECONDS = 15_000;
2020-02-20 01:41:23 +13:00
const APP_STORAGE_UPLOADS = '/storage/uploads';
2020-07-15 09:20:46 +12:00
const APP_STORAGE_FUNCTIONS = '/storage/functions';
const APP_STORAGE_BUILDS = '/storage/builds';
2020-02-20 01:41:23 +13:00
const APP_STORAGE_CACHE = '/storage/cache';
const APP_STORAGE_CERTIFICATES = '/storage/certificates';
2020-02-25 23:04:12 +13:00
const APP_STORAGE_CONFIG = '/storage/config';
2022-01-31 21:37:45 +13:00
const APP_STORAGE_READ_BUFFER = 20 * (1000 * 1000); //20MB other names `APP_STORAGE_MEMORY_LIMIT`, `APP_STORAGE_MEMORY_BUFFER`, `APP_STORAGE_READ_LIMIT`, `APP_STORAGE_BUFFER_LIMIT`
2021-12-12 09:17:33 +13:00
const APP_SOCIAL_TWITTER = 'https://twitter.com/appwrite';
const APP_SOCIAL_TWITTER_HANDLE = 'appwrite';
2020-02-21 09:43:06 +13:00
const APP_SOCIAL_FACEBOOK = 'https://www.facebook.com/appwrite.io';
const APP_SOCIAL_LINKEDIN = 'https://www.linkedin.com/company/appwrite';
const APP_SOCIAL_INSTAGRAM = 'https://www.instagram.com/appwrite.io';
const APP_SOCIAL_GITHUB = 'https://github.com/appwrite';
2020-07-20 15:59:04 +12:00
const APP_SOCIAL_DISCORD = 'https://appwrite.io/discord';
2021-04-11 04:14:15 +12:00
const APP_SOCIAL_DISCORD_CHANNEL = '564160730845151244';
2020-05-18 03:57:42 +12:00
const APP_SOCIAL_DEV = 'https://dev.to/appwrite';
2022-05-24 02:54:50 +12:00
const APP_SOCIAL_STACKSHARE = 'https://stackshare.io/appwrite';
2021-12-11 13:17:28 +13:00
const APP_SOCIAL_YOUTUBE = 'https://www.youtube.com/c/appwrite?sub_confirmation=1';
2023-08-17 06:08:24 +12:00
const APP_HOSTNAME_INTERNAL = 'appwrite';
2024-02-21 01:06:35 +13:00
// Database Reconnect
const DATABASE_RECONNECT_SLEEP = 2;
const DATABASE_RECONNECT_MAX_ATTEMPTS = 10;
2024-02-21 01:06:35 +13:00
2021-08-09 11:42:08 +12:00
// Database Worker Types
const DATABASE_TYPE_CREATE_ATTRIBUTE = 'createAttribute';
const DATABASE_TYPE_CREATE_INDEX = 'createIndex';
const DATABASE_TYPE_DELETE_ATTRIBUTE = 'deleteAttribute';
const DATABASE_TYPE_DELETE_INDEX = 'deleteIndex';
2023-10-17 12:18:51 +13:00
const DATABASE_TYPE_DELETE_COLLECTION = 'deleteCollection';
const DATABASE_TYPE_DELETE_DATABASE = 'deleteDatabase';
2024-02-21 01:06:35 +13:00
// Build Worker Types
const BUILD_TYPE_DEPLOYMENT = 'deployment';
const BUILD_TYPE_RETRY = 'retry';
2024-02-21 01:06:35 +13:00
2020-12-19 03:05:15 +13:00
// Deletion Types
Database layer (#3338) * database response model * database collection config * new database scopes * database service update * database execption codes * remove read write permission from database model * updating tests and fixing some bugs * server side tests are now passing * databases api * tests for database endpoint * composer update * fix error * formatting * formatting fixes * get database test * more updates to events and usage * more usage updates * fix delete type * fix test * delete database * more fixes * databaseId in attributes and indexes * more fixes * fix issues * fix index subquery * fix console scope and index query * updating tests as required * fix phpcs errors and warnings * updates to review suggestions * UI progress * ui updates and cleaning up * fix type * rework database events * update tests * update types * event generation fixed * events config updated * updating context to support multiple * realtime updates * fix ids * update context * validator updates * fix naming conflict * fix tests * fix lint errors * fix wprler and realtime tests * fix webhooks test * fix event validator and other tests * formatting fixes * removing leftover var_dumps * remove leftover comment * update usage params * usage metrics updates * update database usage * fix usage * specs update * updates to usage * fix UI and usage * fix lints * internal id fixes * fixes for internal Id * renaming services and related files * rename tests * rename doc link * rename readme * fix test name * tests: fixes for 0.15.x sync Co-authored-by: Torsten Dittmann <torsten.dittmann@googlemail.com>
2022-06-22 22:51:49 +12:00
const DELETE_TYPE_DATABASES = 'databases';
2020-12-19 03:05:15 +13:00
const DELETE_TYPE_DOCUMENT = 'document';
2021-10-26 13:14:55 +13:00
const DELETE_TYPE_COLLECTIONS = 'collections';
const DELETE_TYPE_PROJECTS = 'projects';
const DELETE_TYPE_FUNCTIONS = 'functions';
const DELETE_TYPE_DEPLOYMENTS = 'deployments';
2021-10-26 13:14:55 +13:00
const DELETE_TYPE_USERS = 'users';
2022-05-24 02:54:50 +12:00
const DELETE_TYPE_TEAMS = 'teams';
2020-12-28 06:57:35 +13:00
const DELETE_TYPE_EXECUTIONS = 'executions';
2020-12-19 03:05:15 +13:00
const DELETE_TYPE_AUDIT = 'audit';
const DELETE_TYPE_ABUSE = 'abuse';
const DELETE_TYPE_USAGE = 'usage';
const DELETE_TYPE_REALTIME = 'realtime';
2021-11-07 18:54:28 +13:00
const DELETE_TYPE_BUCKETS = 'buckets';
2023-07-31 07:10:25 +12:00
const DELETE_TYPE_INSTALLATIONS = 'installations';
2023-03-10 20:42:52 +13:00
const DELETE_TYPE_RULES = 'rules';
const DELETE_TYPE_SESSIONS = 'sessions';
2022-08-15 21:05:41 +12:00
const DELETE_TYPE_CACHE_BY_TIMESTAMP = 'cacheByTimeStamp';
const DELETE_TYPE_CACHE_BY_RESOURCE = 'cacheByResource';
2022-11-17 01:51:43 +13:00
const DELETE_TYPE_SCHEDULES = 'schedules';
2023-10-26 06:33:23 +13:00
const DELETE_TYPE_TOPIC = 'topic';
const DELETE_TYPE_TARGET = 'target';
2024-01-19 17:23:44 +13:00
const DELETE_TYPE_EXPIRED_TARGETS = 'invalid_targets';
const DELETE_TYPE_SESSION_TARGETS = 'session_targets';
2024-02-21 01:06:35 +13:00
// Message types
const MESSAGE_SEND_TYPE_INTERNAL = 'internal';
const MESSAGE_SEND_TYPE_EXTERNAL = 'external';
// Mail Types
const MAIL_TYPE_VERIFICATION = 'verification';
2021-08-30 22:44:52 +12:00
const MAIL_TYPE_MAGIC_SESSION = 'magicSession';
const MAIL_TYPE_RECOVERY = 'recovery';
const MAIL_TYPE_INVITATION = 'invitation';
2022-04-14 00:26:07 +12:00
const MAIL_TYPE_CERTIFICATE = 'certificate';
2021-03-29 10:22:12 +13:00
// Auth Types
const APP_AUTH_TYPE_SESSION = 'Session';
const APP_AUTH_TYPE_JWT = 'JWT';
const APP_AUTH_TYPE_KEY = 'Key';
const APP_AUTH_TYPE_ADMIN = 'Admin';
2021-09-30 21:41:29 +13:00
// Response related
2022-05-24 02:54:50 +12:00
const MAX_OUTPUT_CHUNK_SIZE = 2 * 1024 * 1024; // 2MB
// Function headers
2023-08-20 23:43:06 +12:00
const FUNCTION_ALLOWLIST_HEADERS_REQUEST = ['content-type', 'agent', 'content-length', 'host'];
2023-08-12 01:34:57 +12:00
const FUNCTION_ALLOWLIST_HEADERS_RESPONSE = ['content-type', 'content-length'];
2023-11-29 17:05:37 +13:00
// Message types
const MESSAGE_TYPE_EMAIL = 'email';
const MESSAGE_TYPE_SMS = 'sms';
const MESSAGE_TYPE_PUSH = 'push';
2023-02-03 07:16:01 +13:00
// Usage metrics
2023-02-03 05:36:32 +13:00
const METRIC_TEAMS = 'teams';
const METRIC_USERS = 'users';
2024-02-12 14:18:19 +13:00
const METRIC_MESSAGES = 'messages';
const METRIC_SESSIONS = 'sessions';
2023-02-03 05:36:32 +13:00
const METRIC_DATABASES = 'databases';
const METRIC_COLLECTIONS = 'collections';
2023-02-07 03:03:45 +13:00
const METRIC_DATABASE_ID_COLLECTIONS = '{databaseInternalId}.collections';
2023-02-03 05:36:32 +13:00
const METRIC_DOCUMENTS = 'documents';
2023-02-07 03:03:45 +13:00
const METRIC_DATABASE_ID_DOCUMENTS = '{databaseInternalId}.documents';
const METRIC_DATABASE_ID_COLLECTION_ID_DOCUMENTS = '{databaseInternalId}.{collectionInternalId}.documents';
2023-02-03 05:36:32 +13:00
const METRIC_BUCKETS = 'buckets';
const METRIC_FILES = 'files';
const METRIC_FILES_STORAGE = 'files.storage';
2023-02-07 03:03:45 +13:00
const METRIC_BUCKET_ID_FILES = '{bucketInternalId}.files';
const METRIC_BUCKET_ID_FILES_STORAGE = '{bucketInternalId}.files.storage';
const METRIC_FUNCTIONS = 'functions';
const METRIC_DEPLOYMENTS = 'deployments';
const METRIC_DEPLOYMENTS_STORAGE = 'deployments.storage';
const METRIC_BUILDS = 'builds';
const METRIC_BUILDS_STORAGE = 'builds.storage';
const METRIC_BUILDS_COMPUTE = 'builds.compute';
const METRIC_FUNCTION_ID_BUILDS = '{functionInternalId}.builds';
2023-02-03 07:16:01 +13:00
const METRIC_FUNCTION_ID_BUILDS_STORAGE = '{functionInternalId}.builds.storage';
const METRIC_FUNCTION_ID_BUILDS_COMPUTE = '{functionInternalId}.builds.compute';
const METRIC_FUNCTION_ID_DEPLOYMENTS = '{resourceType}.{resourceInternalId}.deployments';
const METRIC_FUNCTION_ID_DEPLOYMENTS_STORAGE = '{resourceType}.{resourceInternalId}.deployments.storage';
const METRIC_EXECUTIONS = 'executions';
const METRIC_EXECUTIONS_COMPUTE = 'executions.compute';
const METRIC_FUNCTION_ID_EXECUTIONS = '{functionInternalId}.executions';
const METRIC_FUNCTION_ID_EXECUTIONS_COMPUTE = '{functionInternalId}.executions.compute';
const METRIC_NETWORK_REQUESTS = 'network.requests';
const METRIC_NETWORK_INBOUND = 'network.inbound';
const METRIC_NETWORK_OUTBOUND = 'network.outbound';
2019-05-09 18:54:39 +12:00
2019-10-01 17:57:41 +13:00
$register = new Registry();
2020-06-19 12:04:09 +12:00
2024-04-02 00:02:47 +13:00
App::setMode(System::getEnv('_APP_ENV', App::MODE_TYPE_PRODUCTION));
2019-05-09 18:54:39 +12:00
if (!App::isProduction()) {
2024-01-09 09:11:48 +13:00
// Allow specific domains to skip public domain validation in dev environment
// Useful for existing tests involving webhooks
PublicDomain::allow(['request-catcher']);
}
2019-10-01 17:57:41 +13:00
/*
2019-05-09 18:54:39 +12:00
* ENV vars
*/
2022-05-24 02:54:50 +12:00
Config::load('events', __DIR__ . '/config/events.php');
Config::load('auth', __DIR__ . '/config/auth.php');
2024-03-05 11:12:54 +13:00
Config::load('apis', __DIR__ . '/config/apis.php'); // List of APIs
2022-05-24 02:54:50 +12:00
Config::load('errors', __DIR__ . '/config/errors.php');
2023-10-26 06:33:23 +13:00
Config::load('oAuthProviders', __DIR__ . '/config/oAuthProviders.php');
2022-05-24 02:54:50 +12:00
Config::load('platforms', __DIR__ . '/config/platforms.php');
Config::load('collections', __DIR__ . '/config/collections.php');
Config::load('runtimes', __DIR__ . '/config/runtimes.php');
2023-09-05 05:53:25 +12:00
Config::load('runtimes-v2', __DIR__ . '/config/runtimes-v2.php');
2023-10-25 20:39:59 +13:00
Config::load('usage', __DIR__ . '/config/usage.php');
2022-05-24 02:54:50 +12:00
Config::load('roles', __DIR__ . '/config/roles.php'); // User roles and scopes
Config::load('scopes', __DIR__ . '/config/scopes.php'); // User roles and scopes
Config::load('services', __DIR__ . '/config/services.php'); // List of services
Config::load('variables', __DIR__ . '/config/variables.php'); // List of env variables
Config::load('regions', __DIR__ . '/config/regions.php'); // List of available regions
2022-05-24 02:54:50 +12:00
Config::load('avatar-browsers', __DIR__ . '/config/avatars/browsers.php');
Config::load('avatar-credit-cards', __DIR__ . '/config/avatars/credit-cards.php');
Config::load('avatar-flags', __DIR__ . '/config/avatars/flags.php');
Config::load('locale-codes', __DIR__ . '/config/locale/codes.php');
Config::load('locale-currencies', __DIR__ . '/config/locale/currencies.php');
Config::load('locale-eu', __DIR__ . '/config/locale/eu.php');
Config::load('locale-languages', __DIR__ . '/config/locale/languages.php');
Config::load('locale-phones', __DIR__ . '/config/locale/phones.php');
Config::load('locale-countries', __DIR__ . '/config/locale/countries.php');
Config::load('locale-continents', __DIR__ . '/config/locale/continents.php');
Config::load('locale-templates', __DIR__ . '/config/locale/templates.php');
2022-05-24 02:54:50 +12:00
Config::load('storage-logos', __DIR__ . '/config/storage/logos.php');
Config::load('storage-mimes', __DIR__ . '/config/storage/mimes.php');
Config::load('storage-inputs', __DIR__ . '/config/storage/inputs.php');
Config::load('storage-outputs', __DIR__ . '/config/storage/outputs.php');
/**
* New DB Filters
*/
2022-05-24 02:54:50 +12:00
Database::addFilter(
'casting',
function (mixed $value) {
2022-04-22 01:14:01 +12:00
return json_encode(['value' => $value], JSON_PRESERVE_ZERO_FRACTION);
},
function (mixed $value) {
2021-08-28 11:42:53 +12:00
if (is_null($value)) {
2024-03-07 06:34:21 +13:00
return;
2021-08-28 11:42:53 +12:00
}
2022-05-31 00:01:39 +12:00
return json_decode($value, true)['value'];
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'enum',
function (mixed $value, Document $attribute) {
if ($attribute->isSet('elements')) {
$attribute->removeAttribute('elements');
}
2022-05-31 00:01:39 +12:00
return $value;
},
function (mixed $value, Document $attribute) {
2023-08-22 15:25:55 +12:00
$formatOptions = \json_decode($attribute->getAttribute('formatOptions', '[]'), true);
if (isset($formatOptions['elements'])) {
$attribute->setAttribute('elements', $formatOptions['elements']);
}
2022-05-31 00:01:39 +12:00
return $value;
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'range',
function (mixed $value, Document $attribute) {
if ($attribute->isSet('min')) {
$attribute->removeAttribute('min');
}
if ($attribute->isSet('max')) {
$attribute->removeAttribute('max');
}
2022-05-31 00:01:39 +12:00
return $value;
},
function (mixed $value, Document $attribute) {
2021-12-09 06:49:44 +13:00
$formatOptions = json_decode($attribute->getAttribute('formatOptions', '[]'), true);
if (isset($formatOptions['min']) || isset($formatOptions['max'])) {
$attribute
->setAttribute('min', $formatOptions['min'])
->setAttribute('max', $formatOptions['max'])
;
}
2022-05-31 00:01:39 +12:00
return $value;
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'subQueryAttributes',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2021-08-20 23:15:17 +12:00
},
function (mixed $value, Document $document, Database $database) {
$attributes = $database->find('attributes', [
Query::equal('collectionInternalId', [$document->getInternalId()]),
Query::equal('databaseInternalId', [$document->getAttribute('databaseInternalId')]),
Query::limit($database->getLimitForAttributes()),
]);
foreach ($attributes as $attribute) {
if ($attribute->getAttribute('type') === Database::VAR_RELATIONSHIP) {
$options = $attribute->getAttribute('options');
foreach ($options as $key => $value) {
$attribute->setAttribute($key, $value);
}
$attribute->removeAttribute('options');
}
}
return $attributes;
2021-08-20 23:15:17 +12:00
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'subQueryIndexes',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2021-08-23 03:00:00 +12:00
},
function (mixed $value, Document $document, Database $database) {
2021-08-23 03:00:00 +12:00
return $database
->find('indexes', [
2022-08-12 11:53:52 +12:00
Query::equal('collectionInternalId', [$document->getInternalId()]),
Query::equal('databaseInternalId', [$document->getAttribute('databaseInternalId')]),
2023-08-22 15:25:55 +12:00
Query::limit($database->getLimitForIndexes()),
2022-08-12 11:53:52 +12:00
]);
2021-08-23 03:00:00 +12:00
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'subQueryPlatforms',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
},
function (mixed $value, Document $document, Database $database) {
return $database
->find('platforms', [
2022-08-12 11:53:52 +12:00
Query::equal('projectInternalId', [$document->getInternalId()]),
Query::limit(APP_LIMIT_SUBQUERY),
]);
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'subQueryKeys',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
},
function (mixed $value, Document $document, Database $database) {
return $database
->find('keys', [
2022-08-12 11:53:52 +12:00
Query::equal('projectInternalId', [$document->getInternalId()]),
Query::limit(APP_LIMIT_SUBQUERY),
]);
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'subQueryWebhooks',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
},
function (mixed $value, Document $document, Database $database) {
return $database
->find('webhooks', [
2022-08-12 11:53:52 +12:00
Query::equal('projectInternalId', [$document->getInternalId()]),
Query::limit(APP_LIMIT_SUBQUERY),
]);
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'subQuerySessions',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2022-04-04 21:59:32 +12:00
},
function (mixed $value, Document $document, Database $database) {
2022-06-02 05:13:15 +12:00
return Authorization::skip(fn () => $database->find('sessions', [
2022-08-12 11:53:52 +12:00
Query::equal('userInternalId', [$document->getInternalId()]),
Query::limit(APP_LIMIT_SUBQUERY),
]));
2022-04-04 21:59:32 +12:00
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'subQueryTokens',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2022-04-27 23:06:53 +12:00
},
function (mixed $value, Document $document, Database $database) {
2024-03-07 06:34:21 +13:00
return Authorization::skip(fn () => $database
2022-04-27 23:06:53 +12:00
->find('tokens', [
2022-08-12 11:53:52 +12:00
Query::equal('userInternalId', [$document->getInternalId()]),
Query::limit(APP_LIMIT_SUBQUERY),
]));
}
);
2022-05-31 00:01:39 +12:00
2023-06-23 01:35:49 +12:00
Database::addFilter(
'subQueryChallenges',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2023-06-23 01:35:49 +12:00
},
function (mixed $value, Document $document, Database $database) {
2024-03-07 06:34:21 +13:00
return Authorization::skip(fn () => $database
2023-06-23 01:35:49 +12:00
->find('challenges', [
Query::equal('userInternalId', [$document->getInternalId()]),
Query::limit(APP_LIMIT_SUBQUERY),
]));
}
);
2024-03-01 09:59:49 +13:00
Database::addFilter(
'subQueryAuthenticators',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2024-03-01 09:59:49 +13:00
},
function (mixed $value, Document $document, Database $database) {
2024-03-07 06:34:21 +13:00
return Authorization::skip(fn () => $database
2024-03-01 09:59:49 +13:00
->find('authenticators', [
Query::equal('userInternalId', [$document->getInternalId()]),
Query::limit(APP_LIMIT_SUBQUERY),
]));
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'subQueryMemberships',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2022-04-28 00:44:47 +12:00
},
function (mixed $value, Document $document, Database $database) {
2024-03-07 06:34:21 +13:00
return Authorization::skip(fn () => $database
2022-04-28 00:44:47 +12:00
->find('memberships', [
2022-08-12 11:53:52 +12:00
Query::equal('userInternalId', [$document->getInternalId()]),
Query::limit(APP_LIMIT_SUBQUERY),
]));
2021-08-23 03:00:00 +12:00
}
);
2022-07-20 19:18:49 +12:00
Database::addFilter(
'subQueryVariables',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2022-07-20 19:18:49 +12:00
},
function (mixed $value, Document $document, Database $database) {
2022-08-10 00:32:33 +12:00
return $database
2022-07-20 19:18:49 +12:00
->find('variables', [
2023-03-12 05:06:02 +13:00
Query::equal('resourceInternalId', [$document->getInternalId()]),
Query::equal('resourceType', ['function']),
2022-09-03 02:19:36 +12:00
Query::limit(APP_LIMIT_SUBQUERY),
]);
2021-08-23 03:00:00 +12:00
}
);
2022-05-24 02:54:50 +12:00
Database::addFilter(
'encrypt',
function (mixed $value) {
2024-04-02 00:02:47 +13:00
$key = System::getEnv('_APP_OPENSSL_KEY_V1');
$iv = OpenSSL::randomPseudoBytes(OpenSSL::cipherIVLength(OpenSSL::CIPHER_AES_128_GCM));
$tag = null;
2022-05-31 00:01:39 +12:00
return json_encode([
'data' => OpenSSL::encrypt($value, OpenSSL::CIPHER_AES_128_GCM, $key, 0, $iv, $tag),
'method' => OpenSSL::CIPHER_AES_128_GCM,
2021-12-15 01:10:38 +13:00
'iv' => \bin2hex($iv),
'tag' => \bin2hex($tag ?? ''),
'version' => '1',
]);
},
function (mixed $value) {
2022-05-24 02:54:50 +12:00
if (is_null($value)) {
2024-03-07 06:34:21 +13:00
return;
2021-12-03 20:38:21 +13:00
}
$value = json_decode($value, true);
2024-04-02 00:02:47 +13:00
$key = System::getEnv('_APP_OPENSSL_KEY_V' . $value['version']);
return OpenSSL::decrypt($value['data'], $value['method'], $key, 0, hex2bin($value['iv']), hex2bin($value['tag']));
}
);
2019-05-09 18:54:39 +12:00
2023-07-28 19:56:07 +12:00
Database::addFilter(
'subQueryProjectVariables',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
2023-07-28 19:56:07 +12:00
},
function (mixed $value, Document $document, Database $database) {
return $database
->find('variables', [
Query::equal('resourceType', ['project']),
Query::limit(APP_LIMIT_SUBQUERY)
]);
}
);
Database::addFilter(
'userSearch',
function (mixed $value, Document $user) {
$searchValues = [
$user->getId(),
$user->getAttribute('email', ''),
$user->getAttribute('name', ''),
$user->getAttribute('phone', '')
];
foreach ($user->getAttribute('labels', []) as $label) {
$searchValues[] = 'label:' . $label;
}
$search = implode(' ', \array_filter($searchValues));
return $search;
},
function (mixed $value) {
return $value;
}
);
Database::addFilter(
'subQueryTargets',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
},
function (mixed $value, Document $document, Database $database) {
2024-03-07 06:34:21 +13:00
return Authorization::skip(fn () => $database
->find('targets', [
Query::equal('userInternalId', [$document->getInternalId()]),
2023-10-26 06:33:23 +13:00
Query::limit(APP_LIMIT_SUBQUERY)
]));
}
);
Database::addFilter(
2023-08-25 22:25:04 +12:00
'subQueryTopicTargets',
function (mixed $value) {
2024-03-07 06:34:21 +13:00
return;
},
function (mixed $value, Document $document, Database $database) {
$targetIds = Authorization::skip(fn () => \array_map(
2024-01-15 18:25:34 +13:00
fn ($document) => $document->getAttribute('targetInternalId'),
$database->find('subscribers', [
Query::equal('topicInternalId', [$document->getInternalId()]),
2023-10-26 21:16:45 +13:00
Query::limit(APP_LIMIT_SUBSCRIBERS_SUBQUERY)
2023-08-25 11:16:15 +12:00
])
));
if (\count($targetIds) > 0) {
2024-01-15 18:25:34 +13:00
return $database->find('targets', [
Query::equal('$internalId', $targetIds)
]);
}
return [];
}
);
2023-11-14 21:37:52 +13:00
Database::addFilter(
'providerSearch',
function (mixed $value, Document $provider) {
$searchValues = [
$provider->getId(),
$provider->getAttribute('name', ''),
$provider->getAttribute('provider', ''),
$provider->getAttribute('type', '')
];
$search = \implode(' ', \array_filter($searchValues));
2023-11-14 21:37:52 +13:00
return $search;
},
function (mixed $value) {
return $value;
}
);
Database::addFilter(
'topicSearch',
function (mixed $value, Document $topic) {
$searchValues = [
$topic->getId(),
$topic->getAttribute('name', ''),
$topic->getAttribute('description', ''),
];
$search = \implode(' ', \array_filter($searchValues));
2023-11-14 21:37:52 +13:00
return $search;
},
function (mixed $value) {
return $value;
}
);
Database::addFilter(
'messageSearch',
function (mixed $value, Document $message) {
$searchValues = [
$message->getId(),
$message->getAttribute('description', ''),
$message->getAttribute('status', ''),
];
$data = \json_decode($message->getAttribute('data', []), true);
$providerType = $message->getAttribute('providerType', '');
if ($providerType === MESSAGE_TYPE_EMAIL) {
2023-11-29 17:05:37 +13:00
$searchValues = \array_merge($searchValues, [$data['subject'], MESSAGE_TYPE_EMAIL]);
} elseif ($providerType === MESSAGE_TYPE_SMS) {
2023-11-29 17:05:37 +13:00
$searchValues = \array_merge($searchValues, [$data['content'], MESSAGE_TYPE_SMS]);
2023-11-14 21:37:52 +13:00
} else {
2023-11-29 17:05:37 +13:00
$searchValues = \array_merge($searchValues, [$data['title'], MESSAGE_TYPE_PUSH]);
2023-11-14 21:37:52 +13:00
}
$search = \implode(' ', \array_filter($searchValues));
2023-11-14 21:37:52 +13:00
return $search;
},
function (mixed $value) {
return $value;
}
);
2021-08-20 23:15:17 +12:00
/**
* DB Formats
*/
2022-05-24 02:54:50 +12:00
Structure::addFormat(APP_DATABASE_ATTRIBUTE_EMAIL, function () {
return new Email();
2021-08-22 03:09:08 +12:00
}, Database::VAR_STRING);
2022-07-29 00:38:54 +12:00
Structure::addFormat(APP_DATABASE_ATTRIBUTE_DATETIME, function () {
return new DatetimeValidator();
}, Database::VAR_DATETIME);
2022-05-24 02:54:50 +12:00
Structure::addFormat(APP_DATABASE_ATTRIBUTE_ENUM, function ($attribute) {
$elements = $attribute['formatOptions']['elements'];
2021-12-16 23:15:55 +13:00
return new WhiteList($elements, true);
}, Database::VAR_STRING);
2022-05-24 02:54:50 +12:00
Structure::addFormat(APP_DATABASE_ATTRIBUTE_IP, function () {
return new IP();
2021-08-22 03:09:08 +12:00
}, Database::VAR_STRING);
2022-05-24 02:54:50 +12:00
Structure::addFormat(APP_DATABASE_ATTRIBUTE_URL, function () {
return new URL();
2021-08-22 03:09:08 +12:00
}, Database::VAR_STRING);
2022-05-24 02:54:50 +12:00
Structure::addFormat(APP_DATABASE_ATTRIBUTE_INT_RANGE, function ($attribute) {
2021-08-22 03:09:08 +12:00
$min = $attribute['formatOptions']['min'] ?? -INF;
$max = $attribute['formatOptions']['max'] ?? INF;
return new Range($min, $max, Range::TYPE_INTEGER);
}, Database::VAR_INTEGER);
2021-07-28 06:19:37 +12:00
2022-05-24 02:54:50 +12:00
Structure::addFormat(APP_DATABASE_ATTRIBUTE_FLOAT_RANGE, function ($attribute) {
2021-08-22 03:09:08 +12:00
$min = $attribute['formatOptions']['min'] ?? -INF;
$max = $attribute['formatOptions']['max'] ?? INF;
return new Range($min, $max, Range::TYPE_FLOAT);
}, Database::VAR_FLOAT);
2019-10-01 17:57:41 +13:00
/*
2019-05-09 18:54:39 +12:00
* Registry
*/
2022-05-24 02:54:50 +12:00
$register->set('logger', function () {
2022-06-24 21:48:55 +12:00
// Register error logger
2024-04-02 00:02:47 +13:00
$providerName = System::getEnv('_APP_LOGGING_PROVIDER', '');
$providerConfig = System::getEnv('_APP_LOGGING_CONFIG', '');
2022-05-24 02:54:50 +12:00
if (empty($providerName) || empty($providerConfig)) {
2024-03-07 06:34:21 +13:00
return;
}
2022-05-24 02:54:50 +12:00
if (!Logger::hasProvider($providerName)) {
2022-08-14 17:35:25 +12:00
throw new Exception(Exception::GENERAL_SERVER_ERROR, "Logging provider not supported. Logging is disabled");
}
2022-05-24 02:54:50 +12:00
$classname = '\\Utopia\\Logger\\Adapter\\' . \ucfirst($providerName);
$adapter = new $classname($providerConfig);
return new Logger($adapter);
});
$register->set('pools', function () {
2022-10-19 21:35:30 +13:00
$group = new Group();
2022-06-23 20:50:00 +12:00
2023-08-23 23:52:17 +12:00
$fallbackForDB = 'db_main=' . AppwriteURL::unparse([
'scheme' => 'mariadb',
2024-04-02 00:02:47 +13:00
'host' => System::getEnv('_APP_DB_HOST', 'mariadb'),
'port' => System::getEnv('_APP_DB_PORT', '3306'),
'user' => System::getEnv('_APP_DB_USER', ''),
'pass' => System::getEnv('_APP_DB_PASS', ''),
'path' => System::getEnv('_APP_DB_SCHEMA', ''),
]);
2023-08-23 23:52:17 +12:00
$fallbackForRedis = 'redis_main=' . AppwriteURL::unparse([
'scheme' => 'redis',
2024-04-02 00:02:47 +13:00
'host' => System::getEnv('_APP_REDIS_HOST', 'redis'),
'port' => System::getEnv('_APP_REDIS_PORT', '6379'),
'user' => System::getEnv('_APP_REDIS_USER', ''),
'pass' => System::getEnv('_APP_REDIS_PASS', ''),
]);
2021-03-25 00:37:57 +13:00
$connections = [
'console' => [
'type' => 'database',
2024-04-02 00:02:47 +13:00
'dsns' => System::getEnv('_APP_CONNECTIONS_DB_CONSOLE', $fallbackForDB),
'multiple' => false,
'schemes' => ['mariadb', 'mysql'],
],
'database' => [
'type' => 'database',
2024-04-02 00:02:47 +13:00
'dsns' => System::getEnv('_APP_CONNECTIONS_DB_PROJECT', $fallbackForDB),
'multiple' => true,
'schemes' => ['mariadb', 'mysql'],
],
'queue' => [
'type' => 'queue',
2024-04-02 00:02:47 +13:00
'dsns' => System::getEnv('_APP_CONNECTIONS_QUEUE', $fallbackForRedis),
'multiple' => false,
'schemes' => ['redis'],
],
'pubsub' => [
'type' => 'pubsub',
2024-04-02 00:02:47 +13:00
'dsns' => System::getEnv('_APP_CONNECTIONS_PUBSUB', $fallbackForRedis),
'multiple' => false,
'schemes' => ['redis'],
],
'cache' => [
'type' => 'cache',
2024-04-02 00:02:47 +13:00
'dsns' => System::getEnv('_APP_CONNECTIONS_CACHE', $fallbackForRedis),
'multiple' => true,
'schemes' => ['redis'],
],
];
2019-05-09 18:54:39 +12:00
2024-04-02 00:02:47 +13:00
$maxConnections = System::getEnv('_APP_CONNECTIONS_MAX', 151);
$instanceConnections = $maxConnections / System::getEnv('_APP_POOL_CLIENTS', 14);
2019-05-09 18:54:39 +12:00
2024-04-02 00:02:47 +13:00
$multiprocessing = System::getEnv('_APP_SERVER_MULTIPROCESS', 'disabled') === 'enabled';
2022-11-22 20:57:44 +13:00
if ($multiprocessing) {
2024-04-02 00:02:47 +13:00
$workerCount = swoole_cpu_num() * intval(System::getEnv('_APP_WORKER_PER_CORE', 6));
2022-11-22 20:17:56 +13:00
} else {
$workerCount = 1;
2021-03-25 00:37:57 +13:00
}
2019-05-09 18:54:39 +12:00
2022-11-16 20:18:59 +13:00
if ($workerCount > $instanceConnections) {
throw new \Exception('Pool size is too small. Increase the number of allowed database connections or decrease the number of workers.', 500);
}
2019-05-09 18:54:39 +12:00
$poolSize = (int)($instanceConnections / $workerCount);
foreach ($connections as $key => $connection) {
$type = $connection['type'] ?? '';
$dsns = $connection['dsns'] ?? '';
$multipe = $connection['multiple'] ?? false;
$schemes = $connection['schemes'] ?? [];
2022-10-16 03:14:17 +13:00
$config = [];
$dsns = explode(',', $connection['dsns'] ?? '');
foreach ($dsns as &$dsn) {
$dsn = explode('=', $dsn);
$name = ($multipe) ? $key . '_' . $dsn[0] : $key;
2022-10-16 03:14:17 +13:00
$dsn = $dsn[1] ?? '';
$config[] = $name;
2022-10-19 21:35:30 +13:00
if (empty($dsn)) {
2022-10-16 07:17:03 +13:00
//throw new Exception(Exception::GENERAL_SERVER_ERROR, "Missing value for DSN connection in {$key}");
continue;
2022-10-16 03:14:17 +13:00
}
$dsn = new DSN($dsn);
$dsnHost = $dsn->getHost();
$dsnPort = $dsn->getPort();
$dsnUser = $dsn->getUser();
$dsnPass = $dsn->getPassword();
2022-10-20 02:55:58 +13:00
$dsnScheme = $dsn->getScheme();
$dsnDatabase = $dsn->getPath();
2019-05-09 18:54:39 +12:00
if (!in_array($dsnScheme, $schemes)) {
throw new Exception(Exception::GENERAL_SERVER_ERROR, "Invalid console database scheme");
}
2019-05-09 18:54:39 +12:00
/**
* Get Resource
2022-10-19 21:35:30 +13:00
*
* Creation could be reused across connection types like database, cache, queue, etc.
2022-10-19 21:35:30 +13:00
*
* Resource assignment to an adapter will happen below.
*/
2022-10-20 02:55:58 +13:00
switch ($dsnScheme) {
case 'mysql':
case 'mariadb':
2022-10-20 02:55:58 +13:00
$resource = function () use ($dsnHost, $dsnPort, $dsnUser, $dsnPass, $dsnDatabase) {
return new PDOProxy(function () use ($dsnHost, $dsnPort, $dsnUser, $dsnPass, $dsnDatabase) {
return new PDO("mysql:host={$dsnHost};port={$dsnPort};dbname={$dsnDatabase};charset=utf8mb4", $dsnUser, $dsnPass, array(
// No need to set PDO::ATTR_ERRMODE it is overwritten in PDOProxy
2022-10-17 11:49:53 +13:00
PDO::ATTR_TIMEOUT => 3, // Seconds
PDO::ATTR_PERSISTENT => true,
PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
PDO::ATTR_EMULATE_PREPARES => true,
PDO::ATTR_STRINGIFY_FETCHES => true
));
2022-10-17 11:49:53 +13:00
});
};
break;
case 'redis':
2022-10-19 21:35:30 +13:00
$resource = function () use ($dsnHost, $dsnPort, $dsnPass) {
2022-10-17 11:49:53 +13:00
$redis = new Redis();
@$redis->pconnect($dsnHost, (int)$dsnPort);
2022-10-19 21:35:30 +13:00
if ($dsnPass) {
2022-10-17 11:49:53 +13:00
$redis->auth($dsnPass);
}
$redis->setOption(Redis::OPT_READ_TIMEOUT, -1);
2022-10-19 21:35:30 +13:00
2022-10-17 11:49:53 +13:00
return $redis;
};
break;
2022-10-19 21:35:30 +13:00
default:
throw new Exception(Exception::GENERAL_SERVER_ERROR, "Invalid scheme");
}
2022-11-16 20:18:59 +13:00
$pool = new Pool($name, $poolSize, function () use ($type, $resource, $dsn) {
2022-10-17 11:49:53 +13:00
// Get Adapter
switch ($type) {
case 'database':
$adapter = match ($dsn->getScheme()) {
'mariadb' => new MariaDB($resource()),
'mysql' => new MySQL($resource()),
default => null
};
2022-10-19 21:35:30 +13:00
2023-12-15 02:32:06 +13:00
$adapter->setDatabase($dsn->getPath());
2022-10-17 11:49:53 +13:00
break;
case 'pubsub':
$adapter = $resource();
2022-11-16 05:03:42 +13:00
break;
2022-11-10 06:01:43 +13:00
case 'queue':
$adapter = match ($dsn->getScheme()) {
'redis' => new Queue\Connection\Redis($dsn->getHost(), $dsn->getPort()),
2022-11-16 05:03:42 +13:00
default => null
2022-11-10 06:01:43 +13:00
};
2022-10-17 11:49:53 +13:00
break;
case 'cache':
$adapter = match ($dsn->getScheme()) {
'redis' => new RedisCache($resource()),
default => null
};
break;
2022-10-19 21:35:30 +13:00
2022-10-17 11:49:53 +13:00
default:
throw new Exception(Exception::GENERAL_SERVER_ERROR, "Server error: Missing adapter implementation.");
2022-10-17 11:49:53 +13:00
}
2022-10-19 21:35:30 +13:00
2022-10-16 03:14:17 +13:00
return $adapter;
});
$group->add($pool);
}
2022-10-16 03:14:17 +13:00
Config::setParam('pools-' . $key, $config);
}
2019-05-09 18:54:39 +12:00
return $group;
2019-05-09 18:54:39 +12:00
});
2022-11-16 05:03:42 +13:00
2023-11-21 05:31:20 +13:00
$register->set('db', function () {
// This is usually for our workers or CLI commands scope
2024-04-02 00:02:47 +13:00
$dbHost = System::getEnv('_APP_DB_HOST', '');
$dbPort = System::getEnv('_APP_DB_PORT', '');
$dbUser = System::getEnv('_APP_DB_USER', '');
$dbPass = System::getEnv('_APP_DB_PASS', '');
$dbScheme = System::getEnv('_APP_DB_SCHEMA', '');
2023-12-24 22:55:51 +13:00
return new PDO(
"mysql:host={$dbHost};port={$dbPort};dbname={$dbScheme};charset=utf8mb4",
$dbUser,
$dbPass,
SQL::getPDOAttributes()
);
2019-05-09 18:54:39 +12:00
});
2023-08-21 00:29:43 +12:00
2020-06-29 08:45:36 +12:00
$register->set('smtp', function () {
2019-08-09 09:49:46 +12:00
$mail = new PHPMailer(true);
2019-05-09 18:54:39 +12:00
2019-08-09 09:49:46 +12:00
$mail->isSMTP();
2019-05-09 18:54:39 +12:00
2024-04-02 00:02:47 +13:00
$username = System::getEnv('_APP_SMTP_USERNAME');
$password = System::getEnv('_APP_SMTP_PASSWORD');
2019-08-09 09:49:46 +12:00
2019-10-01 17:57:41 +13:00
$mail->XMailer = 'Appwrite Mailer';
2024-04-02 00:02:47 +13:00
$mail->Host = System::getEnv('_APP_SMTP_HOST', 'smtp');
$mail->Port = System::getEnv('_APP_SMTP_PORT', 25);
2023-08-30 16:56:08 +12:00
$mail->SMTPAuth = !empty($username) && !empty($password);
2019-10-01 17:57:41 +13:00
$mail->Username = $username;
$mail->Password = $password;
2024-04-02 00:02:47 +13:00
$mail->SMTPSecure = System::getEnv('_APP_SMTP_SECURE', '');
2020-01-12 02:58:02 +13:00
$mail->SMTPAutoTLS = false;
2020-06-13 04:49:56 +12:00
$mail->CharSet = 'UTF-8';
2019-08-09 09:49:46 +12:00
2024-04-02 00:02:47 +13:00
$from = \urldecode(System::getEnv('_APP_SYSTEM_EMAIL_NAME', APP_NAME . ' Server'));
$email = System::getEnv('_APP_SYSTEM_EMAIL_ADDRESS', APP_EMAIL_TEAM);
$mail->setFrom($email, $from);
$mail->addReplyTo($email, $from);
2019-05-09 18:54:39 +12:00
2019-08-09 09:49:46 +12:00
$mail->isHTML(true);
2019-09-28 13:48:50 +12:00
2019-08-09 09:49:46 +12:00
return $mail;
2019-05-09 18:54:39 +12:00
});
2020-10-30 03:08:09 +13:00
$register->set('geodb', function () {
2024-02-13 17:55:32 +13:00
return new Reader(__DIR__ . '/assets/dbip/dbip-country-lite-2024-02.mmdb');
2020-10-30 03:08:09 +13:00
});
2022-12-26 22:24:51 +13:00
$register->set('passwordsDictionary', function () {
2022-12-26 23:22:38 +13:00
$content = \file_get_contents(__DIR__ . '/assets/security/10k-common-passwords');
2022-12-27 19:16:19 +13:00
$content = explode("\n", $content);
2022-12-26 23:22:38 +13:00
$content = array_flip($content);
return $content;
2022-12-23 20:19:11 +13:00
});
2022-04-07 18:38:38 +12:00
$register->set('promiseAdapter', function () {
2022-10-12 14:04:11 +13:00
return new Swoole();
2022-04-07 18:38:38 +12:00
});
2023-12-16 11:19:43 +13:00
$register->set('hooks', function () {
return new Hooks();
});
2019-10-01 17:57:41 +13:00
/*
2019-05-09 18:54:39 +12:00
* Localization
*/
Locale::$exceptions = false;
2023-04-17 14:10:17 +12:00
$locales = Config::getParam('locale-codes', []);
2023-04-17 14:15:46 +12:00
foreach ($locales as $locale) {
2023-04-17 14:10:17 +12:00
$code = $locale['code'];
2023-04-18 19:14:03 +12:00
$path = __DIR__ . '/config/locale/translations/' . $code . '.json';
2023-04-17 14:10:17 +12:00
if (!\file_exists($path)) {
$path = __DIR__ . '/config/locale/translations/' . \substr($code, 0, 2) . '.json'; // if `ar-ae` doesn't exist, look for `ar`
if (!\file_exists($path)) {
$path = __DIR__ . '/config/locale/translations/en.json'; // if none translation exists, use default from `en.json`
2023-04-17 21:10:41 +12:00
}
2023-04-17 14:10:17 +12:00
}
2023-04-17 21:10:41 +12:00
Locale::setLanguageFromJSON($code, $path);
2023-04-17 14:10:17 +12:00
}
2019-10-09 17:33:33 +13:00
2020-06-20 23:20:49 +12:00
\stream_context_set_default([ // Set global user agent and http settings
2019-05-09 18:54:39 +12:00
'http' => [
'method' => 'GET',
2022-05-24 02:54:50 +12:00
'user_agent' => \sprintf(
APP_USERAGENT,
2024-04-02 00:02:47 +13:00
System::getEnv('_APP_VERSION', 'UNKNOWN'),
System::getEnv('_APP_SYSTEM_SECURITY_EMAIL_ADDRESS', APP_EMAIL_SECURITY)
2022-05-24 02:54:50 +12:00
),
2019-10-01 17:57:41 +13:00
'timeout' => 2,
],
2019-05-09 18:54:39 +12:00
]);
2020-06-30 16:34:13 +12:00
// Runtime Execution
2024-03-07 06:34:21 +13:00
App::setResource('log', fn () => new Log());
2022-05-24 02:54:50 +12:00
App::setResource('logger', function ($register) {
return $register->get('logger');
}, ['register']);
2020-06-30 16:34:13 +12:00
2023-12-16 11:19:43 +13:00
App::setResource('hooks', function ($register) {
return $register->get('hooks');
}, ['register']);
2024-03-07 06:34:21 +13:00
App::setResource('register', fn () => $register);
2024-04-02 00:02:47 +13:00
App::setResource('locale', fn () => new Locale(System::getEnv('_APP_LOCALE', 'en')));
2020-06-30 16:34:13 +12:00
2023-04-17 14:15:46 +12:00
App::setResource('localeCodes', function () {
2024-03-07 06:34:21 +13:00
return array_map(fn ($locale) => $locale['code'], Config::getParam('locale-codes', []));
2023-04-17 14:10:17 +12:00
});
2020-06-30 16:34:13 +12:00
// Queues
2022-11-19 03:20:05 +13:00
App::setResource('queue', function (Group $pools) {
2022-11-17 08:39:35 +13:00
return $pools->get('queue')->pop()->getResource();
2022-11-16 07:13:17 +13:00
}, ['pools']);
2022-12-21 05:11:30 +13:00
App::setResource('queueForMessaging', function (Connection $queue) {
return new Messaging($queue);
}, ['queue']);
2023-06-12 02:08:48 +12:00
App::setResource('queueForMails', function (Connection $queue) {
return new Mail($queue);
}, ['queue']);
2022-12-21 05:11:30 +13:00
App::setResource('queueForBuilds', function (Connection $queue) {
2022-12-21 01:22:58 +13:00
return new Build($queue);
}, ['queue']);
2022-12-21 05:11:30 +13:00
App::setResource('queueForDatabase', function (Connection $queue) {
return new EventDatabase($queue);
2022-12-21 01:22:58 +13:00
}, ['queue']);
2022-12-21 05:11:30 +13:00
App::setResource('queueForDeletes', function (Connection $queue) {
return new Delete($queue);
}, ['queue']);
2022-12-21 05:11:30 +13:00
App::setResource('queueForEvents', function (Connection $queue) {
2023-06-02 15:54:34 +12:00
return new Event($queue);
2022-12-21 01:22:58 +13:00
}, ['queue']);
2022-12-21 05:11:30 +13:00
App::setResource('queueForAudits', function (Connection $queue) {
2022-12-21 00:11:36 +13:00
return new Audit($queue);
2022-12-21 01:22:58 +13:00
}, ['queue']);
2022-11-19 03:20:05 +13:00
App::setResource('queueForFunctions', function (Connection $queue) {
return new Func($queue);
}, ['queue']);
2023-10-25 20:39:59 +13:00
App::setResource('queueForUsage', function (Connection $queue) {
return new Usage($queue);
}, ['queue']);
2022-12-21 05:11:30 +13:00
App::setResource('queueForCertificates', function (Connection $queue) {
2022-12-14 00:16:12 +13:00
return new Certificate($queue);
}, ['queue']);
2023-10-02 06:39:26 +13:00
App::setResource('queueForMigrations', function (Connection $queue) {
return new Migration($queue);
}, ['queue']);
App::setResource('clients', function ($request, $console, $project) {
2021-08-20 23:15:17 +12:00
$console->setAttribute('platforms', [ // Always allow current host
2022-08-14 22:33:36 +12:00
'$collection' => ID::custom('platforms'),
2021-01-13 05:36:21 +13:00
'name' => 'Current Host',
'type' => Origin::CLIENT_TYPE_WEB,
2021-01-13 05:36:21 +13:00
'hostname' => $request->getHostname(),
], Document::SET_TYPE_APPEND);
2021-12-21 23:48:10 +13:00
2024-04-02 00:02:47 +13:00
$hostnames = explode(',', System::getEnv('_APP_CONSOLE_HOSTNAMES', ''));
$validator = new Hostname();
foreach ($hostnames as $hostname) {
$hostname = trim($hostname);
if (!$validator->isValid($hostname)) {
continue;
2023-12-30 09:52:50 +13:00
}
$console->setAttribute('platforms', [
'$collection' => ID::custom('platforms'),
'type' => Origin::CLIENT_TYPE_WEB,
'name' => $hostname,
'hostname' => $hostname,
], Document::SET_TYPE_APPEND);
2023-12-30 09:52:50 +13:00
}
2020-07-01 06:08:02 +12:00
/**
* Get All verified client URLs for both console and current projects
* + Filter for duplicated entries
*/
$clientsConsole = \array_map(
fn ($node) => $node['hostname'],
\array_filter(
$console->getAttribute('platforms', []),
fn ($node) => (isset($node['type']) && ($node['type'] === Origin::CLIENT_TYPE_WEB) && isset($node['hostname']) && !empty($node['hostname']))
)
);
$clients = \array_unique(
\array_merge(
$clientsConsole,
\array_map(
fn ($node) => $node['hostname'],
\array_filter(
$project->getAttribute('platforms', []),
fn ($node) => (isset($node['type']) && ($node['type'] === Origin::CLIENT_TYPE_WEB || $node['type'] === Origin::CLIENT_TYPE_FLUTTER_WEB) && isset($node['hostname']) && !empty($node['hostname']))
)
)
)
);
2020-07-01 06:08:02 +12:00
return $clients;
2021-01-13 05:36:21 +13:00
}, ['request', 'console', 'project']);
2020-07-01 06:08:02 +12:00
2022-05-24 02:54:50 +12:00
App::setResource('user', function ($mode, $project, $console, $request, $response, $dbForProject, $dbForConsole) {
/** @var Appwrite\Utopia\Request $request */
2020-10-30 02:50:49 +13:00
/** @var Appwrite\Utopia\Response $response */
2021-05-07 10:31:05 +12:00
/** @var Utopia\Database\Document $project */
/** @var Utopia\Database\Database $dbForProject */
2021-05-07 10:31:05 +12:00
/** @var Utopia\Database\Database $dbForConsole */
2021-04-01 23:59:11 +13:00
/** @var string $mode */
2020-07-01 06:08:02 +12:00
Authorization::setDefaultStatus(true);
2022-05-24 02:54:50 +12:00
Auth::setCookieName('a_session_' . $project->getId());
2020-07-01 06:08:02 +12:00
if (APP_MODE_ADMIN === $mode) {
2022-05-24 02:54:50 +12:00
Auth::setCookieName('a_session_' . $console->getId());
2020-07-01 06:08:02 +12:00
}
$session = Auth::decodeSession(
2022-05-24 02:54:50 +12:00
$request->getCookie(
Auth::$cookieName, // Get sessions
$request->getCookie(Auth::$cookieName . '_legacy', '')
)
2023-12-13 04:39:24 +13:00
);
// Get session from header for SSR clients
if (empty($session['id']) && empty($session['secret'])) {
$sessionHeader = $request->getHeader('x-appwrite-session', '');
2020-07-01 06:08:02 +12:00
2023-12-13 04:39:24 +13:00
if (!empty($sessionHeader)) {
$session = Auth::decodeSession($sessionHeader);
}
}
2020-07-01 06:08:02 +12:00
2023-12-13 04:39:24 +13:00
// Get fallback session from old clients (no SameSite support) or clients who block 3rd-party cookies
2022-05-24 02:54:50 +12:00
if ($response) {
$response->addHeader('X-Debug-Fallback', 'false');
}
2020-07-01 06:08:02 +12:00
2022-05-24 02:54:50 +12:00
if (empty($session['id']) && empty($session['secret'])) {
if ($response) {
$response->addHeader('X-Debug-Fallback', 'true');
}
2020-07-05 10:22:22 +12:00
$fallback = $request->getHeader('x-fallback-cookies', '');
2020-07-01 06:08:02 +12:00
$fallback = \json_decode($fallback, true);
$session = Auth::decodeSession(((isset($fallback[Auth::$cookieName])) ? $fallback[Auth::$cookieName] : ''));
}
2021-05-07 10:31:05 +12:00
Auth::$unique = $session['id'] ?? '';
Auth::$secret = $session['secret'] ?? '';
2020-07-01 06:08:02 +12:00
if (APP_MODE_ADMIN !== $mode) {
2021-05-17 21:37:33 +12:00
if ($project->isEmpty()) {
2023-08-11 22:49:05 +12:00
$user = new Document([]);
2022-05-24 02:54:50 +12:00
} else {
2023-06-19 02:08:53 +12:00
if ($project->getId() === 'console') {
2023-06-18 23:38:37 +12:00
$user = $dbForConsole->getDocument('users', Auth::$unique);
} else {
$user = $dbForProject->getDocument('users', Auth::$unique);
}
2021-05-17 21:37:33 +12:00
}
2022-05-24 02:54:50 +12:00
} else {
2021-05-07 10:31:05 +12:00
$user = $dbForConsole->getDocument('users', Auth::$unique);
2020-07-01 06:08:02 +12:00
}
2020-06-30 16:34:13 +12:00
2022-05-24 02:54:50 +12:00
if (
$user->isEmpty() // Check a document has been found in the DB
2024-01-16 03:37:47 +13:00
|| !Auth::sessionVerify($user->getAttribute('sessions', []), Auth::$secret)
2022-05-24 02:54:50 +12:00
) { // Validate user has valid login token
2023-08-11 22:49:05 +12:00
$user = new Document([]);
2020-07-01 06:08:02 +12:00
}
if (APP_MODE_ADMIN === $mode) {
2021-05-16 22:55:12 +12:00
if ($user->find('teamId', $project->getAttribute('teamId'), 'memberships')) {
Authorization::setDefaultStatus(false); // Cancel security segmentation for admin users.
2020-07-01 06:08:02 +12:00
} else {
2023-08-11 22:49:05 +12:00
$user = new Document([]);
2020-07-01 06:08:02 +12:00
}
}
2020-12-29 09:31:42 +13:00
$authJWT = $request->getHeader('x-appwrite-jwt', '');
2021-05-17 21:37:33 +12:00
if (!empty($authJWT) && !$project->isEmpty()) { // JWT authentication
2024-04-02 00:02:47 +13:00
$jwt = new JWT(System::getEnv('_APP_OPENSSL_KEY_V1'), 'HS256', 900, 10); // Instantiate with key, algo, maxAge and leeway.
2020-12-29 10:23:09 +13:00
2020-12-29 09:31:42 +13:00
try {
$payload = $jwt->decode($authJWT);
} catch (JWTException $error) {
2022-08-14 17:35:25 +12:00
throw new Exception(Exception::USER_JWT_INVALID, 'Failed to verify JWT. ' . $error->getMessage());
2020-12-29 09:31:42 +13:00
}
2021-08-17 20:58:33 +12:00
2020-12-29 09:31:42 +13:00
$jwtUserId = $payload['userId'] ?? '';
$jwtSessionId = $payload['sessionId'] ?? '';
2022-05-24 02:54:50 +12:00
if ($jwtUserId && $jwtSessionId) {
$user = $dbForProject->getDocument('users', $jwtUserId);
2020-12-29 09:31:42 +13:00
}
2021-05-16 10:41:42 +12:00
if (empty($user->find('$id', $jwtSessionId, 'sessions'))) { // Match JWT to active token
2023-08-11 22:49:05 +12:00
$user = new Document([]);
2020-12-29 09:31:42 +13:00
}
}
$dbForProject->setMetadata('user', $user->getId());
$dbForConsole->setMetadata('user', $user->getId());
2020-07-01 06:08:02 +12:00
return $user;
}, ['mode', 'project', 'console', 'request', 'response', 'dbForProject', 'dbForConsole']);
2020-07-01 06:08:02 +12:00
2022-05-24 02:54:50 +12:00
App::setResource('project', function ($dbForConsole, $request, $console) {
/** @var Appwrite\Utopia\Request $request */
2021-05-17 21:37:33 +12:00
/** @var Utopia\Database\Database $dbForConsole */
/** @var Utopia\Database\Document $console */
2020-07-01 20:55:14 +12:00
$projectId = $request->getParam('project', $request->getHeader('x-appwrite-project', ''));
2020-06-30 16:34:13 +12:00
2023-08-08 03:17:34 +12:00
if (empty($projectId) || $projectId === 'console') {
2021-05-16 21:18:34 +12:00
return $console;
}
2020-06-30 16:34:13 +12:00
2024-03-07 06:34:21 +13:00
$project = Authorization::skip(fn () => $dbForConsole->getDocument('projects', $projectId));
2020-07-01 06:08:02 +12:00
return $project;
2021-05-16 21:18:34 +12:00
}, ['dbForConsole', 'request', 'console']);
App::setResource('session', function (Document $user) {
2023-06-23 01:35:49 +12:00
if ($user->isEmpty()) {
2024-03-07 06:34:21 +13:00
return;
2023-06-23 01:35:49 +12:00
}
$sessions = $user->getAttribute('sessions', []);
$sessionId = Auth::sessionVerify($user->getAttribute('sessions'), Auth::$secret);
2023-06-23 01:35:49 +12:00
if (!$sessionId) {
2024-03-07 06:34:21 +13:00
return;
2023-06-23 01:35:49 +12:00
}
foreach ($sessions as $session) {/** @var Document $session */
if ($sessionId === $session->getId()) {
return $session;
}
}
2024-03-07 06:34:21 +13:00
return;
}, ['user']);
2023-06-23 01:35:49 +12:00
2022-05-24 02:54:50 +12:00
App::setResource('console', function () {
2021-08-20 23:15:17 +12:00
return new Document([
2022-08-14 22:33:36 +12:00
'$id' => ID::custom('console'),
'$internalId' => ID::custom('console'),
2021-05-16 21:18:34 +12:00
'name' => 'Appwrite',
2022-08-14 22:33:36 +12:00
'$collection' => ID::custom('projects'),
2021-05-16 21:18:34 +12:00
'description' => 'Appwrite core engine',
'logo' => '',
'teamId' => -1,
'webhooks' => [],
'keys' => [],
'platforms' => [
[
2022-08-14 22:33:36 +12:00
'$collection' => ID::custom('platforms'),
2021-05-16 21:18:34 +12:00
'name' => 'Localhost',
'type' => Origin::CLIENT_TYPE_WEB,
2021-05-16 21:18:34 +12:00
'hostname' => 'localhost',
], // Current host is added on app init
],
'legalName' => '',
'legalCountry' => '',
'legalState' => '',
'legalCity' => '',
'legalAddress' => '',
'legalTaxId' => '',
2021-08-06 20:34:17 +12:00
'auths' => [
2024-04-02 00:02:47 +13:00
'invites' => System::getEnv('_APP_CONSOLE_INVITES', 'enabled') === 'enabled',
'limit' => (System::getEnv('_APP_CONSOLE_WHITELIST_ROOT', 'enabled') === 'enabled') ? 1 : 0, // limit signup to 1 user
'duration' => Auth::TOKEN_EXPIRATION_LOGIN_LONG, // 1 Year in seconds
2021-08-06 20:34:17 +12:00
],
2024-04-02 00:02:47 +13:00
'authWhitelistEmails' => (!empty(System::getEnv('_APP_CONSOLE_WHITELIST_EMAILS', null))) ? \explode(',', System::getEnv('_APP_CONSOLE_WHITELIST_EMAILS', null)) : [],
'authWhitelistIPs' => (!empty(System::getEnv('_APP_CONSOLE_WHITELIST_IPS', null))) ? \explode(',', System::getEnv('_APP_CONSOLE_WHITELIST_IPS', null)) : [],
2023-10-26 06:33:23 +13:00
'oAuthProviders' => [
2023-04-18 00:22:47 +12:00
'githubEnabled' => true,
2024-04-02 00:02:47 +13:00
'githubSecret' => System::getEnv('_APP_CONSOLE_GITHUB_SECRET', ''),
'githubAppid' => System::getEnv('_APP_CONSOLE_GITHUB_APP_ID', '')
2023-04-18 00:22:47 +12:00
],
2021-05-16 21:18:34 +12:00
]);
}, []);
2020-06-30 16:34:13 +12:00
2022-10-16 03:14:17 +13:00
App::setResource('dbForProject', function (Group $pools, Database $dbForConsole, Cache $cache, Document $project) {
2022-10-19 21:35:30 +13:00
if ($project->isEmpty() || $project->getId() === 'console') {
2022-10-16 03:14:17 +13:00
return $dbForConsole;
}
2021-05-03 20:28:31 +12:00
$dbAdapter = $pools
2022-10-16 03:14:17 +13:00
->get($project->getAttribute('database'))
->pop()
2023-10-19 15:32:45 +13:00
->getResource();
2021-05-03 20:28:31 +12:00
$database = new Database($dbAdapter, $cache);
2021-05-03 20:28:31 +12:00
2023-10-19 15:32:45 +13:00
$database
->setNamespace('_' . $project->getInternalId())
->setMetadata('host', \gethostname())
2023-10-26 22:28:23 +13:00
->setMetadata('project', $project->getId())
2023-10-19 15:45:32 +13:00
->setTimeout(APP_DATABASE_TIMEOUT_MILLISECONDS);
2021-05-03 20:28:31 +12:00
return $database;
2022-10-16 03:14:17 +13:00
}, ['pools', 'dbForConsole', 'cache', 'project']);
2020-07-01 06:08:02 +12:00
App::setResource('dbForConsole', function (Group $pools, Cache $cache) {
$dbAdapter = $pools
->get('console')
->pop()
->getResource()
;
2020-06-30 16:34:13 +12:00
$database = new Database($dbAdapter, $cache);
2020-06-30 16:34:13 +12:00
2023-10-19 15:32:45 +13:00
$database
->setNamespace('_console')
->setMetadata('host', \gethostname())
2023-10-26 22:28:23 +13:00
->setMetadata('project', 'console')
2023-10-19 15:45:32 +13:00
->setTimeout(APP_DATABASE_TIMEOUT_MILLISECONDS);
2020-07-01 06:08:02 +12:00
2021-05-04 21:31:26 +12:00
return $database;
}, ['pools', 'cache']);
2022-07-28 04:59:27 +12:00
2023-05-23 16:37:25 +12:00
App::setResource('getProjectDB', function (Group $pools, Database $dbForConsole, $cache) {
$databases = []; // TODO: @Meldiron This should probably be responsibility of utopia-php/pools
$getProjectDB = function (Document $project) use ($pools, $dbForConsole, $cache, &$databases) {
if ($project->isEmpty() || $project->getId() === 'console') {
return $dbForConsole;
}
$databaseName = $project->getAttribute('database');
if (isset($databases[$databaseName])) {
$database = $databases[$databaseName];
2023-10-19 17:21:11 +13:00
$database
->setNamespace('_' . $project->getInternalId())
->setMetadata('host', \gethostname())
2023-10-26 22:28:23 +13:00
->setMetadata('project', $project->getId())
2023-10-19 17:21:11 +13:00
->setTimeout(APP_DATABASE_TIMEOUT_MILLISECONDS);
2023-05-23 16:37:25 +12:00
return $database;
}
$dbAdapter = $pools
->get($databaseName)
->pop()
->getResource();
$database = new Database($dbAdapter, $cache);
$databases[$databaseName] = $database;
2020-06-30 16:34:13 +12:00
2023-10-19 17:21:11 +13:00
$database
->setNamespace('_' . $project->getInternalId())
->setMetadata('host', \gethostname())
2023-10-26 22:28:23 +13:00
->setMetadata('project', $project->getId())
2023-10-19 17:21:11 +13:00
->setTimeout(APP_DATABASE_TIMEOUT_MILLISECONDS);
2023-05-23 16:37:25 +12:00
return $database;
};
return $getProjectDB;
}, ['pools', 'dbForConsole', 'cache']);
2022-10-16 03:14:17 +13:00
App::setResource('cache', function (Group $pools) {
$list = Config::getParam('pools-cache', []);
$adapters = [];
2022-10-19 21:35:30 +13:00
foreach ($list as $value) {
$adapters[] = $pools
->get($value)
->pop()
->getResource()
;
}
2020-06-30 16:34:13 +12:00
return new Cache(new Sharding($adapters));
2022-10-16 03:14:17 +13:00
}, ['pools']);
2020-07-01 06:08:02 +12:00
2024-02-21 03:10:51 +13:00
App::setResource('deviceForLocal', function () {
2021-12-13 20:34:15 +13:00
return new Local();
});
2024-02-21 03:10:51 +13:00
App::setResource('deviceForFiles', function ($project) {
return getDevice(APP_STORAGE_UPLOADS . '/app-' . $project->getId());
2021-12-13 20:34:15 +13:00
}, ['project']);
2024-02-21 03:10:51 +13:00
App::setResource('deviceForFunctions', function ($project) {
return getDevice(APP_STORAGE_FUNCTIONS . '/app-' . $project->getId());
}, ['project']);
2024-02-21 03:10:51 +13:00
App::setResource('deviceForBuilds', function ($project) {
return getDevice(APP_STORAGE_BUILDS . '/app-' . $project->getId());
}, ['project']);
2022-05-24 02:54:50 +12:00
function getDevice($root): Device
{
2024-04-02 00:02:47 +13:00
$connection = System::getEnv('_APP_CONNECTIONS_STORAGE', '');
2023-08-23 22:08:48 +12:00
if (!empty($connection)) {
$acl = 'private';
$device = Storage::DEVICE_LOCAL;
$accessKey = '';
$accessSecret = '';
$bucket = '';
$region = '';
2023-08-23 22:08:48 +12:00
try {
$dsn = new DSN($connection);
$device = $dsn->getScheme();
2023-08-23 23:52:17 +12:00
$accessKey = $dsn->getUser() ?? '';
$accessSecret = $dsn->getPassword() ?? '';
$bucket = $dsn->getPath() ?? '';
2023-08-23 22:08:48 +12:00
$region = $dsn->getParam('region');
} catch (\Throwable $e) {
2023-08-23 23:52:17 +12:00
Console::warning($e->getMessage() . 'Invalid DSN. Defaulting to Local device.');
2023-08-23 22:08:48 +12:00
}
2023-08-23 22:08:48 +12:00
switch ($device) {
case Storage::DEVICE_S3:
return new S3($root, $accessKey, $accessSecret, $bucket, $region, $acl);
case STORAGE::DEVICE_DO_SPACES:
return new DOSpaces($root, $accessKey, $accessSecret, $bucket, $region, $acl);
case Storage::DEVICE_BACKBLAZE:
return new Backblaze($root, $accessKey, $accessSecret, $bucket, $region, $acl);
case Storage::DEVICE_LINODE:
return new Linode($root, $accessKey, $accessSecret, $bucket, $region, $acl);
case Storage::DEVICE_WASABI:
return new Wasabi($root, $accessKey, $accessSecret, $bucket, $region, $acl);
case Storage::DEVICE_LOCAL:
default:
return new Local($root);
}
} else {
2024-04-02 00:02:47 +13:00
switch (strtolower(System::getEnv('_APP_STORAGE_DEVICE', Storage::DEVICE_LOCAL) ?? '')) {
2023-08-23 22:08:48 +12:00
case Storage::DEVICE_LOCAL:
default:
return new Local($root);
case Storage::DEVICE_S3:
2024-04-02 00:02:47 +13:00
$s3AccessKey = System::getEnv('_APP_STORAGE_S3_ACCESS_KEY', '');
$s3SecretKey = System::getEnv('_APP_STORAGE_S3_SECRET', '');
$s3Region = System::getEnv('_APP_STORAGE_S3_REGION', '');
$s3Bucket = System::getEnv('_APP_STORAGE_S3_BUCKET', '');
2023-08-23 22:08:48 +12:00
$s3Acl = 'private';
return new S3($root, $s3AccessKey, $s3SecretKey, $s3Bucket, $s3Region, $s3Acl);
case Storage::DEVICE_DO_SPACES:
2024-04-02 00:02:47 +13:00
$doSpacesAccessKey = System::getEnv('_APP_STORAGE_DO_SPACES_ACCESS_KEY', '');
$doSpacesSecretKey = System::getEnv('_APP_STORAGE_DO_SPACES_SECRET', '');
$doSpacesRegion = System::getEnv('_APP_STORAGE_DO_SPACES_REGION', '');
$doSpacesBucket = System::getEnv('_APP_STORAGE_DO_SPACES_BUCKET', '');
2023-08-23 22:08:48 +12:00
$doSpacesAcl = 'private';
return new DOSpaces($root, $doSpacesAccessKey, $doSpacesSecretKey, $doSpacesBucket, $doSpacesRegion, $doSpacesAcl);
case Storage::DEVICE_BACKBLAZE:
2024-04-02 00:02:47 +13:00
$backblazeAccessKey = System::getEnv('_APP_STORAGE_BACKBLAZE_ACCESS_KEY', '');
$backblazeSecretKey = System::getEnv('_APP_STORAGE_BACKBLAZE_SECRET', '');
$backblazeRegion = System::getEnv('_APP_STORAGE_BACKBLAZE_REGION', '');
$backblazeBucket = System::getEnv('_APP_STORAGE_BACKBLAZE_BUCKET', '');
2023-08-23 22:08:48 +12:00
$backblazeAcl = 'private';
return new Backblaze($root, $backblazeAccessKey, $backblazeSecretKey, $backblazeBucket, $backblazeRegion, $backblazeAcl);
case Storage::DEVICE_LINODE:
2024-04-02 00:02:47 +13:00
$linodeAccessKey = System::getEnv('_APP_STORAGE_LINODE_ACCESS_KEY', '');
$linodeSecretKey = System::getEnv('_APP_STORAGE_LINODE_SECRET', '');
$linodeRegion = System::getEnv('_APP_STORAGE_LINODE_REGION', '');
$linodeBucket = System::getEnv('_APP_STORAGE_LINODE_BUCKET', '');
2023-08-23 22:08:48 +12:00
$linodeAcl = 'private';
return new Linode($root, $linodeAccessKey, $linodeSecretKey, $linodeBucket, $linodeRegion, $linodeAcl);
case Storage::DEVICE_WASABI:
2024-04-02 00:02:47 +13:00
$wasabiAccessKey = System::getEnv('_APP_STORAGE_WASABI_ACCESS_KEY', '');
$wasabiSecretKey = System::getEnv('_APP_STORAGE_WASABI_SECRET', '');
$wasabiRegion = System::getEnv('_APP_STORAGE_WASABI_REGION', '');
$wasabiBucket = System::getEnv('_APP_STORAGE_WASABI_BUCKET', '');
2023-08-23 22:08:48 +12:00
$wasabiAcl = 'private';
return new Wasabi($root, $wasabiAccessKey, $wasabiSecretKey, $wasabiBucket, $wasabiRegion, $wasabiAcl);
}
2021-12-13 20:34:15 +13:00
}
}
2020-06-27 00:27:58 +12:00
2022-05-24 02:54:50 +12:00
App::setResource('mode', function ($request) {
/** @var Appwrite\Utopia\Request $request */
2021-12-15 00:15:00 +13:00
/**
* Defines the mode for the request:
* - 'default' => Requests for Client and Server Side
* - 'admin' => Request from the Console on non-console projects
*/
2020-11-19 11:08:45 +13:00
return $request->getParam('mode', $request->getHeader('x-appwrite-mode', APP_MODE_DEFAULT));
2020-07-01 06:08:02 +12:00
}, ['request']);
2020-07-03 08:24:14 +12:00
2022-05-24 02:54:50 +12:00
App::setResource('geodb', function ($register) {
2020-10-30 04:54:36 +13:00
/** @var Utopia\Registry\Registry $register */
2020-10-30 03:08:09 +13:00
return $register->get('geodb');
2020-10-30 04:54:36 +13:00
}, ['register']);
2021-03-10 20:42:45 +13:00
App::setResource('passwordsDictionary', function ($register) {
2022-12-23 20:19:11 +13:00
/** @var Utopia\Registry\Registry $register */
2022-12-26 22:24:51 +13:00
return $register->get('passwordsDictionary');
2022-12-23 20:19:11 +13:00
}, ['register']);
2022-08-11 01:45:04 +12:00
App::setResource('servers', function () {
$platforms = Config::getParam('platforms');
$server = $platforms[APP_PLATFORM_SERVER];
$languages = array_map(function ($language) {
return strtolower($language['name']);
2023-01-17 00:09:13 +13:00
}, $server['sdks']);
2022-08-11 01:45:04 +12:00
return $languages;
2022-08-11 01:49:56 +12:00
});
2022-04-07 18:38:38 +12:00
App::setResource('promiseAdapter', function ($register) {
return $register->get('promiseAdapter');
}, ['register']);
App::setResource('schema', function ($utopia, $dbForProject) {
$complexity = function (int $complexity, array $args) {
$queries = Query::parseQueries($args['queries'] ?? []);
2023-05-01 21:18:50 +12:00
$query = Query::getByType($queries, [Query::TYPE_LIMIT])[0] ?? null;
$limit = $query ? $query->getValue() : APP_LIMIT_LIST_DEFAULT;
return $complexity * $limit;
};
$attributes = function (int $limit, int $offset) use ($dbForProject) {
2024-03-07 06:34:21 +13:00
$attrs = Authorization::skip(fn () => $dbForProject->find('attributes', [
Query::limit($limit),
Query::offset($offset),
]));
return \array_map(function ($attr) {
return $attr->getArrayCopy();
}, $attrs);
};
$urls = [
2022-10-17 13:27:15 +13:00
'list' => function (string $databaseId, string $collectionId, array $args) {
return "/v1/databases/$databaseId/collections/$collectionId/documents";
},
2022-10-17 13:27:15 +13:00
'create' => function (string $databaseId, string $collectionId, array $args) {
return "/v1/databases/$databaseId/collections/$collectionId/documents";
},
2022-10-17 13:27:15 +13:00
'read' => function (string $databaseId, string $collectionId, array $args) {
return "/v1/databases/$databaseId/collections/$collectionId/documents/{$args['documentId']}";
},
2022-10-17 13:27:15 +13:00
'update' => function (string $databaseId, string $collectionId, array $args) {
return "/v1/databases/$databaseId/collections/$collectionId/documents/{$args['documentId']}";
},
2022-10-17 13:27:15 +13:00
'delete' => function (string $databaseId, string $collectionId, array $args) {
return "/v1/databases/$databaseId/collections/$collectionId/documents/{$args['documentId']}";
},
];
2022-10-17 13:27:15 +13:00
$params = [
'list' => function (string $databaseId, string $collectionId, array $args) {
return [ 'queries' => $args['queries']];
},
'create' => function (string $databaseId, string $collectionId, array $args) {
$id = $args['id'] ?? 'unique()';
$permissions = $args['permissions'] ?? null;
unset($args['id']);
unset($args['permissions']);
// Order must be the same as the route params
return [
'databaseId' => $databaseId,
'documentId' => $id,
'collectionId' => $collectionId,
'data' => $args,
'permissions' => $permissions,
];
},
'update' => function (string $databaseId, string $collectionId, array $args) {
$documentId = $args['id'];
$permissions = $args['permissions'] ?? null;
unset($args['id']);
unset($args['permissions']);
// Order must be the same as the route params
return [
'databaseId' => $databaseId,
'collectionId' => $collectionId,
'documentId' => $documentId,
'data' => $args,
'permissions' => $permissions,
];
},
];
return Schema::build(
$utopia,
$complexity,
$attributes,
2022-10-17 13:27:15 +13:00
$urls,
$params,
);
}, ['utopia', 'dbForProject']);
2023-04-26 20:21:10 +12:00
App::setResource('contributors', function () {
2023-08-30 06:25:00 +12:00
$path = 'app/config/contributors.json';
2023-04-26 20:21:10 +12:00
$list = (file_exists($path)) ? json_decode(file_get_contents($path), true) : [];
return $list;
2023-08-30 06:25:00 +12:00
});
2023-04-26 20:21:10 +12:00
App::setResource('employees', function () {
2023-08-30 06:25:00 +12:00
$path = 'app/config/employees.json';
2023-04-26 20:21:10 +12:00
$list = (file_exists($path)) ? json_decode(file_get_contents($path), true) : [];
return $list;
2023-08-30 06:25:00 +12:00
});
2023-04-26 20:21:10 +12:00
App::setResource('heroes', function () {
2023-08-30 06:25:00 +12:00
$path = 'app/config/heroes.json';
2023-04-26 20:21:10 +12:00
$list = (file_exists($path)) ? json_decode(file_get_contents($path), true) : [];
return $list;
2023-08-30 06:25:00 +12:00
});
2023-05-26 20:44:08 +12:00
App::setResource('gitHub', function (Cache $cache) {
return new VcsGitHub($cache);
}, ['cache']);
2023-08-05 07:15:32 +12:00
App::setResource('requestTimestamp', function ($request) {
2023-04-12 02:59:45 +12:00
//TODO: Move this to the Request class itself
$timestampHeader = $request->getHeader('x-appwrite-timestamp');
$requestTimestamp = null;
if (!empty($timestampHeader)) {
try {
$requestTimestamp = new \DateTime($timestampHeader);
} catch (\Throwable $e) {
throw new Exception(Exception::GENERAL_ARGUMENT_INVALID, 'Invalid X-Appwrite-Timestamp header value');
}
}
return $requestTimestamp;
}, ['request']);