$value], JSON_PRESERVE_ZERO_FRACTION); }, function (mixed $value) { if (is_null($value)) { return null; } return json_decode($value, true)['value']; } ); Database::addFilter( 'enum', function (mixed $value, Document $attribute) { if ($attribute->isSet('elements')) { $attribute->removeAttribute('elements'); } return $value; }, function (mixed $value, Document $attribute) { $formatOptions = json_decode($attribute->getAttribute('formatOptions', '[]'), true); if (isset($formatOptions['elements'])) { $attribute->setAttribute('elements', $formatOptions['elements']); } return $value; } ); Database::addFilter( 'range', function (mixed $value, Document $attribute) { if ($attribute->isSet('min')) { $attribute->removeAttribute('min'); } if ($attribute->isSet('max')) { $attribute->removeAttribute('max'); } return $value; }, function (mixed $value, Document $attribute) { $formatOptions = json_decode($attribute->getAttribute('formatOptions', '[]'), true); if (isset($formatOptions['min']) || isset($formatOptions['max'])) { $attribute ->setAttribute('min', $formatOptions['min']) ->setAttribute('max', $formatOptions['max']) ; } return $value; } ); Database::addFilter( 'subQueryAttributes', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return $database ->find('attributes', [ Query::equal('collectionInternalId', [$document->getInternalId()]), Query::equal('databaseInternalId', [$document->getAttribute('databaseInternalId')]), Query::limit($database->getLimitForAttributes()), ]); } ); Database::addFilter( 'subQueryIndexes', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return $database ->find('indexes', [ Query::equal('collectionInternalId', [$document->getInternalId()]), Query::equal('databaseInternalId', [$document->getAttribute('databaseInternalId')]), Query::limit(64), ]); } ); Database::addFilter( 'subQueryPlatforms', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return $database ->find('platforms', [ Query::equal('projectInternalId', [$document->getInternalId()]), Query::limit(APP_LIMIT_SUBQUERY), ]); } ); Database::addFilter( 'subQueryDomains', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return $database ->find('domains', [ Query::equal('projectInternalId', [$document->getInternalId()]), Query::limit(APP_LIMIT_SUBQUERY), ]); } ); Database::addFilter( 'subQueryKeys', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return $database ->find('keys', [ Query::equal('projectInternalId', [$document->getInternalId()]), Query::limit(APP_LIMIT_SUBQUERY), ]); } ); Database::addFilter( 'subQueryWebhooks', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return $database ->find('webhooks', [ Query::equal('projectInternalId', [$document->getInternalId()]), Query::limit(APP_LIMIT_SUBQUERY), ]); } ); Database::addFilter( 'subQuerySessions', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return Authorization::skip(fn () => $database->find('sessions', [ Query::equal('userInternalId', [$document->getInternalId()]), Query::limit(APP_LIMIT_SUBQUERY), ])); } ); Database::addFilter( 'subQueryTokens', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return Authorization::skip(fn() => $database ->find('tokens', [ Query::equal('userInternalId', [$document->getInternalId()]), Query::limit(APP_LIMIT_SUBQUERY), ])); } ); Database::addFilter( 'subQueryMemberships', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return Authorization::skip(fn() => $database ->find('memberships', [ Query::equal('userInternalId', [$document->getInternalId()]), Query::limit(APP_LIMIT_SUBQUERY), ])); } ); Database::addFilter( 'subQueryVariables', function (mixed $value) { return null; }, function (mixed $value, Document $document, Database $database) { return $database ->find('variables', [ Query::equal('functionInternalId', [$document->getInternalId()]), Query::limit(APP_LIMIT_SUBQUERY), ]); } ); Database::addFilter( 'encrypt', function (mixed $value) { $key = App::getEnv('_APP_OPENSSL_KEY_V1'); $iv = OpenSSL::randomPseudoBytes(OpenSSL::cipherIVLength(OpenSSL::CIPHER_AES_128_GCM)); $tag = null; return json_encode([ 'data' => OpenSSL::encrypt($value, OpenSSL::CIPHER_AES_128_GCM, $key, 0, $iv, $tag), 'method' => OpenSSL::CIPHER_AES_128_GCM, 'iv' => \bin2hex($iv), 'tag' => \bin2hex($tag ?? ''), 'version' => '1', ]); }, function (mixed $value) { if (is_null($value)) { return null; } $value = json_decode($value, true); $key = App::getEnv('_APP_OPENSSL_KEY_V' . $value['version']); return OpenSSL::decrypt($value['data'], $value['method'], $key, 0, hex2bin($value['iv']), hex2bin($value['tag'])); } ); /** * DB Formats */ Structure::addFormat(APP_DATABASE_ATTRIBUTE_EMAIL, function () { return new Email(); }, Database::VAR_STRING); Structure::addFormat(APP_DATABASE_ATTRIBUTE_DATETIME, function () { return new DatetimeValidator(); }, Database::VAR_DATETIME); Structure::addFormat(APP_DATABASE_ATTRIBUTE_ENUM, function ($attribute) { $elements = $attribute['formatOptions']['elements']; return new WhiteList($elements, true); }, Database::VAR_STRING); Structure::addFormat(APP_DATABASE_ATTRIBUTE_IP, function () { return new IP(); }, Database::VAR_STRING); Structure::addFormat(APP_DATABASE_ATTRIBUTE_URL, function () { return new URL(); }, Database::VAR_STRING); Structure::addFormat(APP_DATABASE_ATTRIBUTE_INT_RANGE, function ($attribute) { $min = $attribute['formatOptions']['min'] ?? -INF; $max = $attribute['formatOptions']['max'] ?? INF; return new Range($min, $max, Range::TYPE_INTEGER); }, Database::VAR_INTEGER); Structure::addFormat(APP_DATABASE_ATTRIBUTE_FLOAT_RANGE, function ($attribute) { $min = $attribute['formatOptions']['min'] ?? -INF; $max = $attribute['formatOptions']['max'] ?? INF; return new Range($min, $max, Range::TYPE_FLOAT); }, Database::VAR_FLOAT); /* * Registry */ $register->set('logger', function () { // Register error logger $providerName = App::getEnv('_APP_LOGGING_PROVIDER', ''); $providerConfig = App::getEnv('_APP_LOGGING_CONFIG', ''); if (empty($providerName) || empty($providerConfig)) { return null; } if (!Logger::hasProvider($providerName)) { throw new Exception(Exception::GENERAL_SERVER_ERROR, "Logging provider not supported. Logging is disabled"); } $classname = '\\Utopia\\Logger\\Adapter\\' . \ucfirst($providerName); $adapter = new $classname($providerConfig); return new Logger($adapter); }); $register->set('dbPool', function () { // Register DB connection $dbHost = App::getEnv('_APP_DB_HOST', ''); $dbPort = App::getEnv('_APP_DB_PORT', ''); $dbUser = App::getEnv('_APP_DB_USER', ''); $dbPass = App::getEnv('_APP_DB_PASS', ''); $dbScheme = App::getEnv('_APP_DB_SCHEMA', ''); $pool = new PDOPool( (new PDOConfig()) ->withHost($dbHost) ->withPort($dbPort) ->withDbName($dbScheme) ->withCharset('utf8mb4') ->withUsername($dbUser) ->withPassword($dbPass) ->withOptions([ PDO::ATTR_ERRMODE => App::isDevelopment() ? PDO::ERRMODE_WARNING : PDO::ERRMODE_SILENT, // If in production mode, warnings are not displayed 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, ]), 64 ); return $pool; }); $register->set('redisPool', function () { $redisHost = App::getEnv('_APP_REDIS_HOST', ''); $redisPort = App::getEnv('_APP_REDIS_PORT', ''); $redisUser = App::getEnv('_APP_REDIS_USER', ''); $redisPass = App::getEnv('_APP_REDIS_PASS', ''); $redisAuth = ''; if ($redisUser && $redisPass) { $redisAuth = $redisUser . ':' . $redisPass; } $pool = new RedisPool( (new RedisConfig()) ->withHost($redisHost) ->withPort($redisPort) ->withAuth($redisAuth) ->withDbIndex(0), 64 ); return $pool; }); $register->set('influxdb', function () { // Register DB connection $host = App::getEnv('_APP_INFLUXDB_HOST', ''); $port = App::getEnv('_APP_INFLUXDB_PORT', ''); if (empty($host) || empty($port)) { return; } $driver = new InfluxDB\Driver\Curl("http://{$host}:{$port}"); $client = new InfluxDB\Client($host, $port, '', '', false, false, 5); $client->setDriver($driver); return $client; }); $register->set('statsd', function () { // Register DB connection $host = App::getEnv('_APP_STATSD_HOST', 'telegraf'); $port = App::getEnv('_APP_STATSD_PORT', 8125); $connection = new \Domnikl\Statsd\Connection\UdpSocket($host, $port); $statsd = new \Domnikl\Statsd\Client($connection); return $statsd; }); $register->set('smtp', function () { $mail = new PHPMailer(true); $mail->isSMTP(); $username = App::getEnv('_APP_SMTP_USERNAME', null); $password = App::getEnv('_APP_SMTP_PASSWORD', null); $mail->XMailer = 'Appwrite Mailer'; $mail->Host = App::getEnv('_APP_SMTP_HOST', 'smtp'); $mail->Port = App::getEnv('_APP_SMTP_PORT', 25); $mail->SMTPAuth = (!empty($username) && !empty($password)); $mail->Username = $username; $mail->Password = $password; $mail->SMTPSecure = App::getEnv('_APP_SMTP_SECURE', false); $mail->SMTPAutoTLS = false; $mail->CharSet = 'UTF-8'; $from = \urldecode(App::getEnv('_APP_SYSTEM_EMAIL_NAME', APP_NAME . ' Server')); $email = App::getEnv('_APP_SYSTEM_EMAIL_ADDRESS', APP_EMAIL_TEAM); $mail->setFrom($email, $from); $mail->addReplyTo($email, $from); $mail->isHTML(true); return $mail; }); $register->set('geodb', function () { return new Reader(__DIR__ . '/assets/dbip/dbip-country-lite-2022-06.mmdb'); }); $register->set('db', function () { // This is usually for our workers or CLI commands scope $dbHost = App::getEnv('_APP_DB_HOST', ''); $dbPort = App::getEnv('_APP_DB_PORT', ''); $dbUser = App::getEnv('_APP_DB_USER', ''); $dbPass = App::getEnv('_APP_DB_PASS', ''); $dbScheme = App::getEnv('_APP_DB_SCHEMA', ''); $pdo = new PDO("mysql:host={$dbHost};port={$dbPort};dbname={$dbScheme};charset=utf8mb4", $dbUser, $dbPass, array( PDO::ATTR_TIMEOUT => 3, // Seconds PDO::ATTR_PERSISTENT => true, PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC, PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION, PDO::ATTR_EMULATE_PREPARES => true, PDO::ATTR_STRINGIFY_FETCHES => true, )); return $pdo; }); $register->set('cache', function () { // This is usually for our workers or CLI commands scope $redis = new Redis(); $redis->pconnect(App::getEnv('_APP_REDIS_HOST', ''), App::getEnv('_APP_REDIS_PORT', '')); $redis->setOption(Redis::OPT_READ_TIMEOUT, -1); return $redis; }); /* * Localization */ Locale::$exceptions = false; Locale::setLanguageFromJSON('af', __DIR__ . '/config/locale/translations/af.json'); Locale::setLanguageFromJSON('ar', __DIR__ . '/config/locale/translations/ar.json'); Locale::setLanguageFromJSON('as', __DIR__ . '/config/locale/translations/as.json'); Locale::setLanguageFromJSON('az', __DIR__ . '/config/locale/translations/az.json'); Locale::setLanguageFromJSON('be', __DIR__ . '/config/locale/translations/be.json'); Locale::setLanguageFromJSON('bg', __DIR__ . '/config/locale/translations/bg.json'); Locale::setLanguageFromJSON('bh', __DIR__ . '/config/locale/translations/bh.json'); Locale::setLanguageFromJSON('bn', __DIR__ . '/config/locale/translations/bn.json'); Locale::setLanguageFromJSON('bs', __DIR__ . '/config/locale/translations/bs.json'); Locale::setLanguageFromJSON('ca', __DIR__ . '/config/locale/translations/ca.json'); Locale::setLanguageFromJSON('cs', __DIR__ . '/config/locale/translations/cs.json'); Locale::setLanguageFromJSON('da', __DIR__ . '/config/locale/translations/da.json'); Locale::setLanguageFromJSON('de', __DIR__ . '/config/locale/translations/de.json'); Locale::setLanguageFromJSON('el', __DIR__ . '/config/locale/translations/el.json'); Locale::setLanguageFromJSON('en', __DIR__ . '/config/locale/translations/en.json'); Locale::setLanguageFromJSON('eo', __DIR__ . '/config/locale/translations/eo.json'); Locale::setLanguageFromJSON('es', __DIR__ . '/config/locale/translations/es.json'); Locale::setLanguageFromJSON('fa', __DIR__ . '/config/locale/translations/fa.json'); Locale::setLanguageFromJSON('fi', __DIR__ . '/config/locale/translations/fi.json'); Locale::setLanguageFromJSON('fo', __DIR__ . '/config/locale/translations/fo.json'); Locale::setLanguageFromJSON('fr', __DIR__ . '/config/locale/translations/fr.json'); Locale::setLanguageFromJSON('ga', __DIR__ . '/config/locale/translations/ga.json'); Locale::setLanguageFromJSON('gu', __DIR__ . '/config/locale/translations/gu.json'); Locale::setLanguageFromJSON('he', __DIR__ . '/config/locale/translations/he.json'); Locale::setLanguageFromJSON('hi', __DIR__ . '/config/locale/translations/hi.json'); Locale::setLanguageFromJSON('hr', __DIR__ . '/config/locale/translations/hr.json'); Locale::setLanguageFromJSON('hu', __DIR__ . '/config/locale/translations/hu.json'); Locale::setLanguageFromJSON('hy', __DIR__ . '/config/locale/translations/hy.json'); Locale::setLanguageFromJSON('id', __DIR__ . '/config/locale/translations/id.json'); Locale::setLanguageFromJSON('is', __DIR__ . '/config/locale/translations/is.json'); Locale::setLanguageFromJSON('it', __DIR__ . '/config/locale/translations/it.json'); Locale::setLanguageFromJSON('ja', __DIR__ . '/config/locale/translations/ja.json'); Locale::setLanguageFromJSON('jv', __DIR__ . '/config/locale/translations/jv.json'); Locale::setLanguageFromJSON('kn', __DIR__ . '/config/locale/translations/kn.json'); Locale::setLanguageFromJSON('km', __DIR__ . '/config/locale/translations/km.json'); Locale::setLanguageFromJSON('ko', __DIR__ . '/config/locale/translations/ko.json'); Locale::setLanguageFromJSON('la', __DIR__ . '/config/locale/translations/la.json'); Locale::setLanguageFromJSON('lb', __DIR__ . '/config/locale/translations/lb.json'); Locale::setLanguageFromJSON('lt', __DIR__ . '/config/locale/translations/lt.json'); Locale::setLanguageFromJSON('lv', __DIR__ . '/config/locale/translations/lv.json'); Locale::setLanguageFromJSON('ml', __DIR__ . '/config/locale/translations/ml.json'); Locale::setLanguageFromJSON('mr', __DIR__ . '/config/locale/translations/mr.json'); Locale::setLanguageFromJSON('ms', __DIR__ . '/config/locale/translations/ms.json'); Locale::setLanguageFromJSON('nb', __DIR__ . '/config/locale/translations/nb.json'); Locale::setLanguageFromJSON('ne', __DIR__ . '/config/locale/translations/ne.json'); Locale::setLanguageFromJSON('nl', __DIR__ . '/config/locale/translations/nl.json'); Locale::setLanguageFromJSON('nn', __DIR__ . '/config/locale/translations/nn.json'); Locale::setLanguageFromJSON('or', __DIR__ . '/config/locale/translations/or.json'); Locale::setLanguageFromJSON('pa', __DIR__ . '/config/locale/translations/pa.json'); Locale::setLanguageFromJSON('pl', __DIR__ . '/config/locale/translations/pl.json'); Locale::setLanguageFromJSON('pt-br', __DIR__ . '/config/locale/translations/pt-br.json'); Locale::setLanguageFromJSON('pt-pt', __DIR__ . '/config/locale/translations/pt-pt.json'); Locale::setLanguageFromJSON('ro', __DIR__ . '/config/locale/translations/ro.json'); Locale::setLanguageFromJSON('ru', __DIR__ . '/config/locale/translations/ru.json'); Locale::setLanguageFromJSON('sa', __DIR__ . '/config/locale/translations/sa.json'); Locale::setLanguageFromJSON('sd', __DIR__ . '/config/locale/translations/sd.json'); Locale::setLanguageFromJSON('si', __DIR__ . '/config/locale/translations/si.json'); Locale::setLanguageFromJSON('sk', __DIR__ . '/config/locale/translations/sk.json'); Locale::setLanguageFromJSON('sl', __DIR__ . '/config/locale/translations/sl.json'); Locale::setLanguageFromJSON('sn', __DIR__ . '/config/locale/translations/sn.json'); Locale::setLanguageFromJSON('sq', __DIR__ . '/config/locale/translations/sq.json'); Locale::setLanguageFromJSON('sv', __DIR__ . '/config/locale/translations/sv.json'); Locale::setLanguageFromJSON('ta', __DIR__ . '/config/locale/translations/ta.json'); Locale::setLanguageFromJSON('te', __DIR__ . '/config/locale/translations/te.json'); Locale::setLanguageFromJSON('th', __DIR__ . '/config/locale/translations/th.json'); Locale::setLanguageFromJSON('tl', __DIR__ . '/config/locale/translations/tl.json'); Locale::setLanguageFromJSON('tr', __DIR__ . '/config/locale/translations/tr.json'); Locale::setLanguageFromJSON('uk', __DIR__ . '/config/locale/translations/uk.json'); Locale::setLanguageFromJSON('ur', __DIR__ . '/config/locale/translations/ur.json'); Locale::setLanguageFromJSON('vi', __DIR__ . '/config/locale/translations/vi.json'); Locale::setLanguageFromJSON('zh-cn', __DIR__ . '/config/locale/translations/zh-cn.json'); Locale::setLanguageFromJSON('zh-tw', __DIR__ . '/config/locale/translations/zh-tw.json'); \stream_context_set_default([ // Set global user agent and http settings 'http' => [ 'method' => 'GET', 'user_agent' => \sprintf( APP_USERAGENT, App::getEnv('_APP_VERSION', 'UNKNOWN'), App::getEnv('_APP_SYSTEM_SECURITY_EMAIL_ADDRESS', APP_EMAIL_SECURITY) ), 'timeout' => 2, ], ]); // Runtime Execution App::setResource('logger', function ($register) { return $register->get('logger'); }, ['register']); App::setResource('loggerBreadcrumbs', function () { return []; }); App::setResource('register', fn() => $register); App::setResource('locale', fn() => new Locale(App::getEnv('_APP_LOCALE', 'en'))); // Queues App::setResource('events', fn() => new Event('', '')); App::setResource('audits', fn() => new Audit()); App::setResource('mails', fn() => new Mail()); App::setResource('deletes', fn() => new Delete()); App::setResource('database', fn() => new EventDatabase()); App::setResource('messaging', fn() => new Phone()); App::setResource('usage', function ($register) { return new Stats($register->get('statsd')); }, ['register']); App::setResource('clients', function ($request, $console, $project) { $console->setAttribute('platforms', [ // Always allow current host '$collection' => ID::custom('platforms'), 'name' => 'Current Host', 'type' => 'web', 'hostname' => $request->getHostname(), ], Document::SET_TYPE_APPEND); /** * 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'] === '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'] === 'web' && isset($node['hostname']) && !empty($node['hostname'])) ) ) ) ); return $clients; }, ['request', 'console', 'project']); App::setResource('user', function ($mode, $project, $console, $request, $response, $dbForProject, $dbForConsole) { /** @var Appwrite\Utopia\Request $request */ /** @var Appwrite\Utopia\Response $response */ /** @var Utopia\Database\Document $project */ /** @var Utopia\Database\Database $dbForProject */ /** @var Utopia\Database\Database $dbForConsole */ /** @var string $mode */ Authorization::setDefaultStatus(true); Auth::setCookieName('a_session_' . $project->getId()); $authDuration = $project->getAttribute('auths', [])['duration'] ?? Auth::TOKEN_EXPIRATION_LOGIN_LONG; if (APP_MODE_ADMIN === $mode) { Auth::setCookieName('a_session_' . $console->getId()); $authDuration = Auth::TOKEN_EXPIRATION_LOGIN_LONG; } $session = Auth::decodeSession( $request->getCookie( Auth::$cookieName, // Get sessions $request->getCookie(Auth::$cookieName . '_legacy', '') ) );// Get fallback session from old clients (no SameSite support) // Get fallback session from clients who block 3rd-party cookies if ($response) { $response->addHeader('X-Debug-Fallback', 'false'); } if (empty($session['id']) && empty($session['secret'])) { if ($response) { $response->addHeader('X-Debug-Fallback', 'true'); } $fallback = $request->getHeader('x-fallback-cookies', ''); $fallback = \json_decode($fallback, true); $session = Auth::decodeSession(((isset($fallback[Auth::$cookieName])) ? $fallback[Auth::$cookieName] : '')); } Auth::$unique = $session['id'] ?? ''; Auth::$secret = $session['secret'] ?? ''; if (APP_MODE_ADMIN !== $mode) { if ($project->isEmpty()) { $user = new Document(['$id' => ID::custom(''), '$collection' => 'users']); } else { $user = $dbForProject->getDocument('users', Auth::$unique); } } else { $user = $dbForConsole->getDocument('users', Auth::$unique); } if ( $user->isEmpty() // Check a document has been found in the DB || !Auth::sessionVerify($user->getAttribute('sessions', []), Auth::$secret, $authDuration) ) { // Validate user has valid login token $user = new Document(['$id' => ID::custom(''), '$collection' => 'users']); } if (APP_MODE_ADMIN === $mode) { if ($user->find('teamId', $project->getAttribute('teamId'), 'memberships')) { Authorization::setDefaultStatus(false); // Cancel security segmentation for admin users. } else { $user = new Document(['$id' => ID::custom(''), '$collection' => 'users']); } } $authJWT = $request->getHeader('x-appwrite-jwt', ''); if (!empty($authJWT) && !$project->isEmpty()) { // JWT authentication $jwt = new JWT(App::getEnv('_APP_OPENSSL_KEY_V1'), 'HS256', 900, 10); // Instantiate with key, algo, maxAge and leeway. try { $payload = $jwt->decode($authJWT); } catch (JWTException $error) { throw new Exception(Exception::USER_JWT_INVALID, 'Failed to verify JWT. ' . $error->getMessage()); } $jwtUserId = $payload['userId'] ?? ''; $jwtSessionId = $payload['sessionId'] ?? ''; if ($jwtUserId && $jwtSessionId) { $user = $dbForProject->getDocument('users', $jwtUserId); } if (empty($user->find('$id', $jwtSessionId, 'sessions'))) { // Match JWT to active token $user = new Document(['$id' => ID::custom(''), '$collection' => 'users']); } } return $user; }, ['mode', 'project', 'console', 'request', 'response', 'dbForProject', 'dbForConsole']); App::setResource('project', function ($dbForConsole, $request, $console) { /** @var Appwrite\Utopia\Request $request */ /** @var Utopia\Database\Database $dbForConsole */ /** @var Utopia\Database\Document $console */ $projectId = $request->getParam('project', $request->getHeader('x-appwrite-project', 'console')); if ($projectId === 'console') { return $console; } $project = Authorization::skip(fn() => $dbForConsole->getDocument('projects', $projectId)); return $project; }, ['dbForConsole', 'request', 'console']); App::setResource('console', function () { return new Document([ '$id' => ID::custom('console'), '$internalId' => ID::custom('console'), 'name' => 'Appwrite', '$collection' => ID::custom('projects'), 'description' => 'Appwrite core engine', 'logo' => '', 'teamId' => -1, 'webhooks' => [], 'keys' => [], 'platforms' => [ [ '$collection' => ID::custom('platforms'), 'name' => 'Localhost', 'type' => 'web', 'hostname' => 'localhost', ], // Current host is added on app init ], 'legalName' => '', 'legalCountry' => '', 'legalState' => '', 'legalCity' => '', 'legalAddress' => '', 'legalTaxId' => '', 'auths' => [ 'limit' => (App::getEnv('_APP_CONSOLE_WHITELIST_ROOT', 'enabled') === 'enabled') ? 1 : 0, // limit signup to 1 user 'duration' => Auth::TOKEN_EXPIRATION_LOGIN_LONG, // 1 Year in seconds ], 'authWhitelistEmails' => (!empty(App::getEnv('_APP_CONSOLE_WHITELIST_EMAILS', null))) ? \explode(',', App::getEnv('_APP_CONSOLE_WHITELIST_EMAILS', null)) : [], 'authWhitelistIPs' => (!empty(App::getEnv('_APP_CONSOLE_WHITELIST_IPS', null))) ? \explode(',', App::getEnv('_APP_CONSOLE_WHITELIST_IPS', null)) : [], ]); }, []); App::setResource('dbForProject', function ($db, $cache, Document $project) { $cache = new Cache(new RedisCache($cache)); $database = new Database(new MariaDB($db), $cache); $database->setDefaultDatabase(App::getEnv('_APP_DB_SCHEMA', 'appwrite')); $database->setNamespace("_{$project->getInternalId()}"); return $database; }, ['db', 'cache', 'project']); App::setResource('dbForConsole', function ($db, $cache) { $cache = new Cache(new RedisCache($cache)); $database = new Database(new MariaDB($db), $cache); $database->setDefaultDatabase(App::getEnv('_APP_DB_SCHEMA', 'appwrite')); $database->setNamespace('_console'); return $database; }, ['db', 'cache']); App::setResource('deviceLocal', function () { return new Local(); }); App::setResource('deviceFiles', function ($project) { return getDevice(APP_STORAGE_UPLOADS . '/app-' . $project->getId()); }, ['project']); App::setResource('deviceFunctions', function ($project) { return getDevice(APP_STORAGE_FUNCTIONS . '/app-' . $project->getId()); }, ['project']); App::setResource('deviceBuilds', function ($project) { return getDevice(APP_STORAGE_BUILDS . '/app-' . $project->getId()); }, ['project']); function getDevice($root): Device { switch (App::getEnv('_APP_STORAGE_DEVICE', Storage::DEVICE_LOCAL)) { case Storage::DEVICE_LOCAL: default: return new Local($root); case Storage::DEVICE_S3: $s3AccessKey = App::getEnv('_APP_STORAGE_S3_ACCESS_KEY', ''); $s3SecretKey = App::getEnv('_APP_STORAGE_S3_SECRET', ''); $s3Region = App::getEnv('_APP_STORAGE_S3_REGION', ''); $s3Bucket = App::getEnv('_APP_STORAGE_S3_BUCKET', ''); $s3Acl = 'private'; return new S3($root, $s3AccessKey, $s3SecretKey, $s3Bucket, $s3Region, $s3Acl); case Storage::DEVICE_DO_SPACES: $doSpacesAccessKey = App::getEnv('_APP_STORAGE_DO_SPACES_ACCESS_KEY', ''); $doSpacesSecretKey = App::getEnv('_APP_STORAGE_DO_SPACES_SECRET', ''); $doSpacesRegion = App::getEnv('_APP_STORAGE_DO_SPACES_REGION', ''); $doSpacesBucket = App::getEnv('_APP_STORAGE_DO_SPACES_BUCKET', ''); $doSpacesAcl = 'private'; return new DOSpaces($root, $doSpacesAccessKey, $doSpacesSecretKey, $doSpacesBucket, $doSpacesRegion, $doSpacesAcl); case Storage::DEVICE_BACKBLAZE: $backblazeAccessKey = App::getEnv('_APP_STORAGE_BACKBLAZE_ACCESS_KEY', ''); $backblazeSecretKey = App::getEnv('_APP_STORAGE_BACKBLAZE_SECRET', ''); $backblazeRegion = App::getEnv('_APP_STORAGE_BACKBLAZE_REGION', ''); $backblazeBucket = App::getEnv('_APP_STORAGE_BACKBLAZE_BUCKET', ''); $backblazeAcl = 'private'; return new Backblaze($root, $backblazeAccessKey, $backblazeSecretKey, $backblazeBucket, $backblazeRegion, $backblazeAcl); case Storage::DEVICE_LINODE: $linodeAccessKey = App::getEnv('_APP_STORAGE_LINODE_ACCESS_KEY', ''); $linodeSecretKey = App::getEnv('_APP_STORAGE_LINODE_SECRET', ''); $linodeRegion = App::getEnv('_APP_STORAGE_LINODE_REGION', ''); $linodeBucket = App::getEnv('_APP_STORAGE_LINODE_BUCKET', ''); $linodeAcl = 'private'; return new Linode($root, $linodeAccessKey, $linodeSecretKey, $linodeBucket, $linodeRegion, $linodeAcl); case Storage::DEVICE_WASABI: $wasabiAccessKey = App::getEnv('_APP_STORAGE_WASABI_ACCESS_KEY', ''); $wasabiSecretKey = App::getEnv('_APP_STORAGE_WASABI_SECRET', ''); $wasabiRegion = App::getEnv('_APP_STORAGE_WASABI_REGION', ''); $wasabiBucket = App::getEnv('_APP_STORAGE_WASABI_BUCKET', ''); $wasabiAcl = 'private'; return new Wasabi($root, $wasabiAccessKey, $wasabiSecretKey, $wasabiBucket, $wasabiRegion, $wasabiAcl); } } App::setResource('mode', function ($request) { /** @var Appwrite\Utopia\Request $request */ /** * Defines the mode for the request: * - 'default' => Requests for Client and Server Side * - 'admin' => Request from the Console on non-console projects */ return $request->getParam('mode', $request->getHeader('x-appwrite-mode', APP_MODE_DEFAULT)); }, ['request']); App::setResource('geodb', function ($register) { /** @var Utopia\Registry\Registry $register */ return $register->get('geodb'); }, ['register']); App::setResource('sms', function () { $dsn = new DSN(App::getEnv('_APP_SMS_PROVIDER')); $user = $dsn->getUser(); $secret = $dsn->getPassword(); return match ($dsn->getHost()) { 'mock' => new Mock($user, $secret), // used for tests 'twilio' => new Twilio($user, $secret), 'text-magic' => new TextMagic($user, $secret), 'telesign' => new Telesign($user, $secret), 'msg91' => new Msg91($user, $secret), 'vonage' => new Vonage($user, $secret), default => null }; }); App::setResource('servers', function () { $platforms = Config::getParam('platforms'); $server = $platforms[APP_PLATFORM_SERVER]; $languages = array_map(function ($language) { return strtolower($language['name']); }, $server['languages']); return $languages; });