mirror of
https://github.com/appwrite/appwrite.git
synced 2026-05-26 13:51:13 +00:00
147af0579d
Follows utopia-php/http#249 migration guide to drop usage of the shims kept for one-minor-release compat. Affected call sites switch to: - Handler / hook context: inject 'route' (Route|null from the per-request DI container) instead of $utopia->getRoute(). - Resource factories (init/resources/request.php): Router::matchRoute() returning ?RouteMatch instead of $utopia->match($request). - Error path in app/http.php: $app->getResource('routeMatch')?->route, guarded with try/catch for the pre-dispatch error case. - GraphQL resolver inner-request pattern (Resolvers.php): capture getResource('routeMatch') as the original, Router::matchRoute() for the inner URL, rebind 'route' / 'routeMatch' on the resolver container in finally instead of setRoute($original). Http::execute() shim is retained — the guide explicitly keeps it for this hand-built-route case. Redundant re-match in shared/api.php storage-cache branch removed; the injected $route is already the current one. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
108 lines
4.4 KiB
PHP
108 lines
4.4 KiB
PHP
<?php
|
|
|
|
use Appwrite\Extend\Exception;
|
|
use Appwrite\Utopia\Database\Documents\User;
|
|
use Appwrite\Utopia\Request;
|
|
use MaxMind\Db\Reader;
|
|
use Utopia\Config\Config;
|
|
use Utopia\Database\DateTime;
|
|
use Utopia\Database\Document;
|
|
use Utopia\Database\Validator\Authorization;
|
|
use Utopia\Http\Http;
|
|
use Utopia\Http\Route;
|
|
use Utopia\System\System;
|
|
|
|
Http::init()
|
|
->groups(['mfaProtected'])
|
|
->inject('session')
|
|
->action(function (Document $session) {
|
|
$isSessionFresh = false;
|
|
|
|
$lastUpdate = $session->getAttribute('mfaUpdatedAt');
|
|
if (!empty($lastUpdate)) {
|
|
$now = DateTime::now();
|
|
$maxAllowedDate = DateTime::addSeconds(new \DateTime($lastUpdate), MFA_RECENT_DURATION); // Maximum date until session is considered safe before asking for another challenge
|
|
|
|
$isSessionFresh = DateTime::formatTz($maxAllowedDate) >= DateTime::formatTz($now);
|
|
}
|
|
|
|
if (!$isSessionFresh) {
|
|
throw new Exception(Exception::USER_CHALLENGE_REQUIRED);
|
|
}
|
|
});
|
|
|
|
Http::init()
|
|
->groups(['auth'])
|
|
->inject('request')
|
|
->inject('project')
|
|
->inject('geodb')
|
|
->inject('user')
|
|
->inject('authorization')
|
|
->inject('route')
|
|
->action(function (Request $request, Document $project, Reader $geodb, User $user, Authorization $authorization, ?Route $route) {
|
|
$denylist = System::getEnv('_APP_CONSOLE_COUNTRIES_DENYLIST', '');
|
|
if (!empty($denylist && $project->getId() === 'console')) {
|
|
$countries = explode(',', $denylist);
|
|
$record = $geodb->get($request->getIP()) ?? [];
|
|
$country = $record['country']['iso_code'] ?? '';
|
|
if (in_array($country, $countries)) {
|
|
throw new Exception(Exception::GENERAL_REGION_ACCESS_DENIED);
|
|
}
|
|
}
|
|
|
|
$isPrivilegedUser = $user->isPrivileged($authorization->getRoles());
|
|
$isAppUser = $user->isApp($authorization->getRoles());
|
|
|
|
if ($isAppUser || $isPrivilegedUser) { // Skip limits for app and console devs
|
|
return;
|
|
}
|
|
|
|
$auths = $project->getAttribute('auths', []);
|
|
switch ($route?->getLabel('auth.type', '')) {
|
|
case 'email-password':
|
|
if (($auths[Config::getParam('auth')['email-password']['key']] ?? true) === false) {
|
|
throw new Exception(Exception::USER_AUTH_METHOD_UNSUPPORTED, 'Email / Password authentication is disabled for this project');
|
|
}
|
|
break;
|
|
|
|
case 'magic-url':
|
|
if (($auths[Config::getParam('auth')['magic-url']['key']] ?? true) === false) {
|
|
throw new Exception(Exception::USER_AUTH_METHOD_UNSUPPORTED, 'Magic URL authentication is disabled for this project');
|
|
}
|
|
break;
|
|
|
|
case 'anonymous':
|
|
if (($auths[Config::getParam('auth')['anonymous']['key']] ?? true) === false) {
|
|
throw new Exception(Exception::USER_AUTH_METHOD_UNSUPPORTED, 'Anonymous authentication is disabled for this project');
|
|
}
|
|
break;
|
|
|
|
case 'phone':
|
|
if (($auths[Config::getParam('auth')['phone']['key']] ?? true) === false) {
|
|
throw new Exception(Exception::USER_AUTH_METHOD_UNSUPPORTED, 'Phone authentication is disabled for this project');
|
|
}
|
|
break;
|
|
|
|
case 'invites':
|
|
if (($auths[Config::getParam('auth')['invites']['key']] ?? true) === false) {
|
|
throw new Exception(Exception::USER_AUTH_METHOD_UNSUPPORTED, 'Invites authentication is disabled for this project');
|
|
}
|
|
break;
|
|
|
|
case 'jwt':
|
|
if (($auths[Config::getParam('auth')['jwt']['key']] ?? true) === false) {
|
|
throw new Exception(Exception::USER_AUTH_METHOD_UNSUPPORTED, 'JWT authentication is disabled for this project');
|
|
}
|
|
break;
|
|
|
|
case 'email-otp':
|
|
if (($auths[Config::getParam('auth')['email-otp']['key']] ?? true) === false) {
|
|
throw new Exception(Exception::USER_AUTH_METHOD_UNSUPPORTED, 'Email OTP authentication is disabled for this project');
|
|
}
|
|
break;
|
|
|
|
default:
|
|
throw new Exception(Exception::USER_AUTH_METHOD_UNSUPPORTED, 'Unsupported authentication route');
|
|
}
|
|
});
|