1
0
Fork 0
mirror of synced 2024-07-01 20:50:49 +12:00
appwrite/app/controllers/api/users.php

866 lines
33 KiB
PHP
Raw Normal View History

2019-05-09 18:54:39 +12:00
<?php
2021-05-07 10:31:05 +12:00
use Appwrite\Auth\Auth;
use Appwrite\Auth\Validator\Password;
2022-01-19 00:05:04 +13:00
use Appwrite\Detector\Detector;
2022-05-26 01:36:25 +12:00
use Appwrite\Event\Delete;
use Appwrite\Event\Event;
use Appwrite\Event\Audit as EventAudit;
2022-01-19 00:05:04 +13:00
use Appwrite\Network\Validator\Email;
2022-05-26 01:36:25 +12:00
use Appwrite\Stats\Stats;
2022-01-19 00:05:04 +13:00
use Appwrite\Utopia\Database\Validator\CustomId;
2021-05-07 10:31:05 +12:00
use Appwrite\Utopia\Response;
2020-06-29 05:31:21 +12:00
use Utopia\App;
use Utopia\Audit\Audit;
2022-01-19 00:05:04 +13:00
use Utopia\Config\Config;
2022-05-26 01:36:25 +12:00
use Utopia\Locale\Locale;
use Appwrite\Extend\Exception;
2021-05-07 10:31:05 +12:00
use Utopia\Database\Document;
use Utopia\Database\Exception\Duplicate;
use Utopia\Database\Validator\UID;
use Utopia\Database\Database;
use Utopia\Database\Query;
use Utopia\Database\Validator\Authorization;
2022-01-19 00:05:04 +13:00
use Utopia\Validator\Assoc;
use Utopia\Validator\WhiteList;
use Utopia\Validator\Text;
use Utopia\Validator\Range;
use Utopia\Validator\Boolean;
2022-05-26 01:36:25 +12:00
use MaxMind\Db\Reader;
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::post('/v1/users')
2020-02-05 19:31:34 +13:00
->desc('Create User')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].create')
2020-02-05 19:31:34 +13:00
->label('scope', 'users.write')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2020-02-05 19:31:34 +13:00
->label('sdk.namespace', 'users')
->label('sdk.method', 'create')
->label('sdk.description', '/docs/references/users/create-user.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_CREATED)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USER)
->param('userId', '', new CustomId(), 'User ID. Choose your own unique ID or pass the string "unique()" to auto generate it. Valid chars are a-z, A-Z, 0-9, period, hyphen, and underscore. Can\'t start with a special char. Max length is 36 chars.')
2020-09-11 02:40:14 +12:00
->param('email', '', new Email(), 'User email.')
->param('password', '', new Password(), 'User password. Must be at least 8 chars.')
2020-09-11 02:40:14 +12:00
->param('name', '', new Text(128), 'User name. Max length: 128 chars.', true)
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-04-04 18:30:07 +12:00
->inject('events')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, string $email, string $password, string $name, Response $response, Database $dbForProject, Stats $usage, Event $events) {
2020-02-05 19:31:34 +13:00
$email = \strtolower($email);
2020-02-05 19:31:34 +13:00
2020-06-30 23:09:28 +12:00
try {
$userId = $userId == 'unique()' ? $dbForProject->getId() : $userId;
$user = $dbForProject->createDocument('users', new Document([
2021-05-07 10:31:05 +12:00
'$id' => $userId,
2021-06-12 06:23:16 +12:00
'$read' => ['role:all'],
2022-05-24 02:54:50 +12:00
'$write' => ['user:' . $userId],
2020-06-30 23:09:28 +12:00
'email' => $email,
'emailVerification' => false,
'status' => true,
2020-06-30 23:09:28 +12:00
'password' => Auth::passwordHash($password),
'passwordUpdate' => \time(),
2020-06-30 23:09:28 +12:00
'registration' => \time(),
'reset' => false,
'name' => $name,
2021-12-28 23:48:50 +13:00
'prefs' => new \stdClass(),
2022-04-26 22:36:49 +12:00
'sessions' => null,
2022-04-27 23:06:53 +12:00
'tokens' => null,
2022-04-28 00:44:47 +12:00
'memberships' => null,
2022-05-16 21:58:17 +12:00
'search' => implode(' ', [$userId, $email, $name])
2021-05-07 10:31:05 +12:00
]));
2020-06-30 23:09:28 +12:00
} catch (Duplicate $th) {
2022-02-07 01:49:01 +13:00
throw new Exception('Account already exists', 409, Exception::USER_ALREADY_EXISTS);
2020-06-30 23:09:28 +12:00
}
2020-02-05 19:31:34 +13:00
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.create', 1)
;
2022-04-04 18:30:07 +12:00
$events
->setParam('userId', $user->getId())
;
2021-05-07 10:31:05 +12:00
$response->setStatusCode(Response::STATUS_CODE_CREATED);
2021-07-26 02:47:18 +12:00
$response->dynamic($user, Response::MODEL_USER);
2020-12-27 05:54:42 +13:00
});
2020-02-05 19:31:34 +13:00
2020-06-29 05:31:21 +12:00
App::get('/v1/users')
2019-05-09 18:54:39 +12:00
->desc('List Users')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2019-05-09 18:54:39 +12:00
->label('scope', 'users.read')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'list')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/users/list-users.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USER_LIST)
2020-09-11 02:40:14 +12:00
->param('search', '', new Text(256), 'Search term to filter your list results. Max length: 256 chars.', true)
2021-12-15 00:21:44 +13:00
->param('limit', 25, new Range(0, 100), 'Maximum number of users to return in response. By default will return maximum 25 results. Maximum of 100 results allowed per request.', true)
->param('offset', 0, new Range(0, APP_LIMIT_COUNT), 'Offset value. The default value is 0. Use this param to manage pagination. [learn more about pagination](https://appwrite.io/docs/pagination)', true)
->param('cursor', '', new UID(), 'ID of the user used as the starting point for the query, excluding the user itself. Should be used for efficient pagination when working with large sets of data. [learn more about pagination](https://appwrite.io/docs/pagination)', true)
->param('cursorDirection', Database::CURSOR_AFTER, new WhiteList([Database::CURSOR_AFTER, Database::CURSOR_BEFORE]), 'Direction of the cursor.', true)
2020-09-11 02:40:14 +12:00
->param('orderType', 'ASC', new WhiteList(['ASC', 'DESC'], true), 'Order result by ASC or DESC order.', true)
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-05-26 01:36:25 +12:00
->action(function (string $search, int $limit, int $offset, string $cursor, string $cursorDirection, string $orderType, Response $response, Database $dbForProject, Stats $usage) {
2021-05-07 10:31:05 +12:00
if (!empty($cursor)) {
$cursorUser = $dbForProject->getDocument('users', $cursor);
2021-08-07 00:36:48 +12:00
if ($cursorUser->isEmpty()) {
2022-02-09 11:56:11 +13:00
throw new Exception("User '{$cursor}' for the 'cursor' value not found.", 400, Exception::GENERAL_CURSOR_NOT_FOUND);
2021-08-07 00:36:48 +12:00
}
}
2022-05-16 21:58:17 +12:00
$queries = [];
if (!empty($search)) {
$queries[] = new Query('search', Query::TYPE_SEARCH, [$search]);
}
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.read', 1)
;
2020-06-30 23:09:28 +12:00
2021-07-26 02:47:18 +12:00
$response->dynamic(new Document([
'users' => $dbForProject->find('users', $queries, $limit, $offset, [], [$orderType], $cursorUser ?? null, $cursorDirection),
2022-02-27 22:57:09 +13:00
'total' => $dbForProject->count('users', $queries, APP_LIMIT_COUNT),
2020-10-31 08:53:27 +13:00
]), Response::MODEL_USER_LIST);
2020-12-27 05:54:42 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::get('/v1/users/:userId')
2019-05-09 18:54:39 +12:00
->desc('Get User')
->groups(['api', 'users'])
2019-05-09 18:54:39 +12:00
->label('scope', 'users.read')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'get')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/users/get-user.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USER)
->param('userId', '', new UID(), 'User ID.')
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, Response $response, Database $dbForProject, Stats $usage) {
$user = $dbForProject->getDocument('users', $userId);
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2020-06-30 23:09:28 +12:00
}
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.read', 1)
;
2021-07-26 02:47:18 +12:00
$response->dynamic($user, Response::MODEL_USER);
2020-12-27 05:54:42 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::get('/v1/users/:userId/prefs')
2020-01-23 19:27:19 +13:00
->desc('Get User Preferences')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2019-05-09 18:54:39 +12:00
->label('scope', 'users.read')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'getPrefs')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/users/get-user-prefs.md')
2020-11-13 00:54:16 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
2021-04-22 01:37:51 +12:00
->label('sdk.response.model', Response::MODEL_PREFERENCES)
->param('userId', '', new UID(), 'User ID.')
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, Response $response, Database $dbForProject, Stats $usage) {
2019-05-09 18:54:39 +12:00
$user = $dbForProject->getDocument('users', $userId);
2019-05-09 18:54:39 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2020-06-30 23:09:28 +12:00
}
2019-05-09 18:54:39 +12:00
2021-01-11 00:55:59 +13:00
$prefs = $user->getAttribute('prefs', new \stdClass());
2019-05-09 18:54:39 +12:00
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.read', 1)
;
2021-07-26 02:47:18 +12:00
$response->dynamic(new Document($prefs), Response::MODEL_PREFERENCES);
2020-12-27 05:54:42 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::get('/v1/users/:userId/sessions')
2019-05-09 18:54:39 +12:00
->desc('Get User Sessions')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2019-05-09 18:54:39 +12:00
->label('scope', 'users.read')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'getSessions')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/users/get-user-sessions.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_SESSION_LIST)
->param('userId', '', new UID(), 'User ID.')
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2020-12-27 05:54:42 +13:00
->inject('locale')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, Response $response, Database $dbForProject, Locale $locale, Stats $usage) {
2019-05-09 18:54:39 +12:00
$user = $dbForProject->getDocument('users', $userId);
2020-06-30 23:09:28 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2020-06-30 23:09:28 +12:00
}
2019-05-09 18:54:39 +12:00
2021-02-20 01:12:47 +13:00
$sessions = $user->getAttribute('sessions', []);
2020-06-30 23:09:28 +12:00
2022-05-24 02:54:50 +12:00
foreach ($sessions as $key => $session) {
2021-02-20 01:12:47 +13:00
/** @var Document $session */
2019-05-09 18:54:39 +12:00
2022-05-24 02:54:50 +12:00
$countryName = $locale->getText('countries.' . strtolower($session->getAttribute('countryCode')), $locale->getText('locale.country.unknown'));
2021-07-23 08:15:01 +12:00
$session->setAttribute('countryName', $countryName);
2021-02-20 01:12:47 +13:00
$session->setAttribute('current', false);
2019-05-09 18:54:39 +12:00
2021-02-20 01:12:47 +13:00
$sessions[$key] = $session;
2019-05-09 18:54:39 +12:00
}
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.read', 1)
;
2021-07-26 02:47:18 +12:00
$response->dynamic(new Document([
2021-05-27 22:09:14 +12:00
'sessions' => $sessions,
2022-02-27 22:57:09 +13:00
'total' => count($sessions),
2020-10-31 08:53:27 +13:00
]), Response::MODEL_SESSION_LIST);
});
2019-05-09 18:54:39 +12:00
2022-05-13 01:20:06 +12:00
App::get('/v1/users/:userId/memberships')
->desc('Get User Memberships')
->groups(['api', 'users'])
->label('scope', 'users.read')
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'users')
->label('sdk.method', 'getMemberships')
->label('sdk.description', '/docs/references/users/get-user-memberships.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_MEMBERSHIP_LIST)
->param('userId', '', new UID(), 'User ID.')
->inject('response')
->inject('dbForProject')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, Response $response, Database $dbForProject) {
2022-05-13 01:20:06 +12:00
$user = $dbForProject->getDocument('users', $userId);
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-05-13 01:20:06 +12:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
}
2022-05-24 02:54:50 +12:00
$memberships = array_map(function ($membership) use ($dbForProject, $user) {
2022-05-13 01:20:06 +12:00
$team = $dbForProject->getDocument('teams', $membership->getAttribute('teamId'));
$membership
->setAttribute('teamName', $team->getAttribute('name'))
->setAttribute('userName', $user->getAttribute('name'))
2022-05-13 02:26:12 +12:00
->setAttribute('userEmail', $user->getAttribute('email'));
2022-05-13 01:20:06 +12:00
return $membership;
}, $user->getAttribute('memberships', []));
$response->dynamic(new Document([
'memberships' => $memberships,
'total' => count($memberships),
]), Response::MODEL_MEMBERSHIP_LIST);
});
2020-06-29 05:31:21 +12:00
App::get('/v1/users/:userId/logs')
2019-05-09 18:54:39 +12:00
->desc('Get User Logs')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2019-05-09 18:54:39 +12:00
->label('scope', 'users.read')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'getLogs')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/users/get-user-logs.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_LOG_LIST)
->param('userId', '', new UID(), 'User ID.')
2021-12-15 00:21:44 +13:00
->param('limit', 25, new Range(0, 100), 'Maximum number of logs to return in response. By default will return maximum 25 results. Maximum of 100 results allowed per request.', true)
->param('offset', 0, new Range(0, APP_LIMIT_COUNT), 'Offset value. The default value is 0. Use this value to manage pagination. [learn more about pagination](https://appwrite.io/docs/pagination)', true)
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2020-12-27 05:54:42 +13:00
->inject('locale')
->inject('geodb')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, int $limit, int $offset, Response $response, Database $dbForProject, Locale $locale, Reader $geodb, Stats $usage) {
2020-06-30 23:09:28 +12:00
$user = $dbForProject->getDocument('users', $userId);
2020-06-30 23:09:28 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2020-06-30 23:09:28 +12:00
}
2019-05-09 18:54:39 +12:00
$audit = new Audit($dbForProject);
2021-11-17 03:54:29 +13:00
2022-04-21 02:03:40 +12:00
$logs = $audit->getLogsByUser($user->getId(), $limit, $offset);
2020-06-30 23:09:28 +12:00
$output = [];
foreach ($logs as $i => &$log) {
$log['userAgent'] = (!empty($log['userAgent'])) ? $log['userAgent'] : 'UNKNOWN';
$detector = new Detector($log['userAgent']);
$detector->skipBotDetection(); // OPTIONAL: If called, bot detection will completely be skipped (bots will be detected as regular devices then)
2020-06-30 23:09:28 +12:00
$os = $detector->getOS();
$client = $detector->getClient();
$device = $detector->getDevice();
2020-10-31 08:53:27 +13:00
$output[$i] = new Document([
2020-06-30 23:09:28 +12:00
'event' => $log['event'],
'ip' => $log['ip'],
2021-06-13 06:39:59 +12:00
'time' => $log['time'],
'osCode' => $os['osCode'],
'osName' => $os['osName'],
'osVersion' => $os['osVersion'],
'clientType' => $client['clientType'],
'clientCode' => $client['clientCode'],
'clientName' => $client['clientName'],
'clientVersion' => $client['clientVersion'],
'clientEngine' => $client['clientEngine'],
'clientEngineVersion' => $client['clientEngineVersion'],
'deviceName' => $device['deviceName'],
'deviceBrand' => $device['deviceBrand'],
'deviceModel' => $device['deviceModel']
2020-10-31 08:53:27 +13:00
]);
$record = $geodb->get($log['ip']);
if ($record) {
2022-05-24 02:54:50 +12:00
$output[$i]['countryCode'] = $locale->getText('countries.' . strtolower($record['country']['iso_code']), false) ? \strtolower($record['country']['iso_code']) : '--';
$output[$i]['countryName'] = $locale->getText('countries.' . strtolower($record['country']['iso_code']), $locale->getText('locale.country.unknown'));
2020-10-31 08:53:27 +13:00
} else {
$output[$i]['countryCode'] = '--';
$output[$i]['countryName'] = $locale->getText('locale.country.unknown');
2019-05-09 18:54:39 +12:00
}
}
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.read', 1)
;
2021-11-17 03:54:29 +13:00
$response->dynamic(new Document([
2022-04-21 02:03:40 +12:00
'total' => $audit->countLogsByUser($user->getId()),
2021-11-17 03:54:29 +13:00
'logs' => $output,
]), Response::MODEL_LOG_LIST);
2020-12-27 05:54:42 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::patch('/v1/users/:userId/status')
2019-10-10 16:52:59 +13:00
->desc('Update User Status')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].update.status')
2019-05-09 18:54:39 +12:00
->label('scope', 'users.write')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'updateStatus')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/users/update-user-status.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USER)
->param('userId', '', new UID(), 'User ID.')
->param('status', null, new Boolean(true), 'User Status. To activate the user pass `true` and to block the user pass `false`.')
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-04-04 18:30:07 +12:00
->inject('events')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, bool $status, Response $response, Database $dbForProject, Stats $usage, Event $events) {
2019-05-09 18:54:39 +12:00
$user = $dbForProject->getDocument('users', $userId);
2019-05-09 18:54:39 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2020-06-30 23:09:28 +12:00
}
2019-05-09 18:54:39 +12:00
$user = $dbForProject->updateDocument('users', $user->getId(), $user->setAttribute('status', (bool) $status));
2019-10-21 19:01:07 +13:00
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.update', 1)
;
2022-04-04 18:30:07 +12:00
$events
->setParam('userId', $user->getId())
;
2021-07-26 02:47:18 +12:00
$response->dynamic($user, Response::MODEL_USER);
2020-12-27 05:54:42 +13:00
});
2019-05-09 18:54:39 +12:00
App::patch('/v1/users/:userId/verification')
->desc('Update Email Verification')
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].update.verification')
->label('scope', 'users.write')
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'users')
->label('sdk.method', 'updateVerification')
->label('sdk.description', '/docs/references/users/update-user-verification.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USER)
->param('userId', '', new UID(), 'User ID.')
->param('emailVerification', false, new Boolean(), 'User email verification status.')
->inject('response')
->inject('dbForProject')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-04-04 18:30:07 +12:00
->inject('events')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, bool $emailVerification, Response $response, Database $dbForProject, Stats $usage, Event $events) {
$user = $dbForProject->getDocument('users', $userId);
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
}
$user = $dbForProject->updateDocument('users', $user->getId(), $user->setAttribute('emailVerification', $emailVerification));
2019-05-09 18:54:39 +12:00
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.update', 1)
;
2022-04-04 18:30:07 +12:00
$events
->setParam('userId', $user->getId())
;
2021-07-26 02:47:18 +12:00
$response->dynamic($user, Response::MODEL_USER);
2020-12-27 05:54:42 +13:00
});
2019-05-09 18:54:39 +12:00
2021-08-29 23:13:58 +12:00
App::patch('/v1/users/:userId/name')
->desc('Update Name')
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].update.name')
2021-08-29 23:13:58 +12:00
->label('scope', 'users.write')
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'users')
->label('sdk.method', 'updateName')
->label('sdk.description', '/docs/references/users/update-user-name.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USER)
->param('userId', '', new UID(), 'User ID.')
2021-08-29 23:13:58 +12:00
->param('name', '', new Text(128), 'User name. Max length: 128 chars.')
->inject('response')
->inject('dbForProject')
2021-08-29 23:13:58 +12:00
->inject('audits')
2022-04-04 18:30:07 +12:00
->inject('events')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, string $name, Response $response, Database $dbForProject, EventAudit $audits, Event $events) {
2021-10-08 21:39:37 +13:00
$user = $dbForProject->getDocument('users', $userId);
2021-08-29 23:13:58 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2021-08-29 23:13:58 +12:00
}
2022-04-26 22:07:33 +12:00
$user
->setAttribute('name', $name)
->setAttribute('search', \implode(' ', [$user->getId(), $user->getAttribute('email'), $name]));
;
$user = $dbForProject->updateDocument('users', $user->getId(), $user);
2021-08-29 23:13:58 +12:00
$audits
2022-05-24 02:54:50 +12:00
->setResource('user/' . $user->getId())
2022-04-04 18:30:07 +12:00
;
$events
2021-08-29 23:13:58 +12:00
->setParam('userId', $user->getId())
;
$response->dynamic($user, Response::MODEL_USER);
});
App::patch('/v1/users/:userId/password')
->desc('Update Password')
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].update.password')
2021-08-29 23:13:58 +12:00
->label('scope', 'users.write')
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'users')
->label('sdk.method', 'updatePassword')
->label('sdk.description', '/docs/references/users/update-user-password.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USER)
->param('userId', '', new UID(), 'User ID.')
->param('password', '', new Password(), 'New user password. Must be at least 8 chars.')
2021-08-29 23:13:58 +12:00
->inject('response')
->inject('dbForProject')
2021-08-29 23:13:58 +12:00
->inject('audits')
2022-04-04 18:30:07 +12:00
->inject('events')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, string $password, Response $response, Database $dbForProject, EventAudit $audits, Event $events) {
2021-08-29 23:13:58 +12:00
$user = $dbForProject->getDocument('users', $userId);
2021-08-29 23:13:58 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2021-08-29 23:13:58 +12:00
}
2021-10-08 21:39:37 +13:00
$user
->setAttribute('password', Auth::passwordHash($password))
->setAttribute('passwordUpdate', \time());
$user = $dbForProject->updateDocument('users', $user->getId(), $user);
2021-08-29 23:13:58 +12:00
$audits
2022-05-24 02:54:50 +12:00
->setResource('user/' . $user->getId())
2022-04-04 18:30:07 +12:00
;
$events
2021-08-29 23:13:58 +12:00
->setParam('userId', $user->getId())
;
$response->dynamic($user, Response::MODEL_USER);
});
App::patch('/v1/users/:userId/email')
->desc('Update Email')
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].update.email')
2021-08-29 23:13:58 +12:00
->label('scope', 'users.write')
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'users')
->label('sdk.method', 'updateEmail')
->label('sdk.description', '/docs/references/users/update-user-email.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USER)
->param('userId', '', new UID(), 'User ID.')
2021-08-29 23:13:58 +12:00
->param('email', '', new Email(), 'User email.')
->inject('response')
->inject('dbForProject')
2021-08-29 23:13:58 +12:00
->inject('audits')
2022-04-04 18:30:07 +12:00
->inject('events')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, string $email, Response $response, Database $dbForProject, EventAudit $audits, Event $events) {
2021-08-29 23:13:58 +12:00
$user = $dbForProject->getDocument('users', $userId);
2021-08-29 23:13:58 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2021-08-29 23:13:58 +12:00
}
2021-10-08 06:57:23 +13:00
$isAnonymousUser = is_null($user->getAttribute('email')) && is_null($user->getAttribute('password')); // Check if request is from an anonymous account for converting
if (!$isAnonymousUser) {
2021-10-08 21:39:37 +13:00
//TODO: Remove previous unique ID.
2021-10-08 06:57:23 +13:00
}
$email = \strtolower($email);
2022-04-26 22:07:33 +12:00
$user
->setAttribute('email', $email)
->setAttribute('search', \implode(' ', [$user->getId(), $email, $user->getAttribute('name')]))
;
try {
2022-04-26 22:07:33 +12:00
$user = $dbForProject->updateDocument('users', $user->getId(), $user);
2022-05-24 02:54:50 +12:00
} catch (Duplicate $th) {
2022-02-07 01:49:01 +13:00
throw new Exception('Email already exists', 409, Exception::USER_EMAIL_ALREADY_EXISTS);
2021-08-29 23:13:58 +12:00
}
2022-04-04 18:30:07 +12:00
2021-08-29 23:13:58 +12:00
$audits
2022-05-24 02:54:50 +12:00
->setResource('user/' . $user->getId())
2022-04-04 18:30:07 +12:00
;
$events
2021-08-29 23:13:58 +12:00
->setParam('userId', $user->getId())
;
$response->dynamic($user, Response::MODEL_USER);
});
2020-06-29 05:31:21 +12:00
App::patch('/v1/users/:userId/prefs')
2020-01-23 19:27:19 +13:00
->desc('Update User Preferences')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].update.prefs')
->label('scope', 'users.write')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'updatePrefs')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/users/update-user-prefs.md')
2020-11-13 00:54:16 +13:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
2021-04-22 01:37:51 +12:00
->label('sdk.response.model', Response::MODEL_PREFERENCES)
->param('userId', '', new UID(), 'User ID.')
2020-09-11 02:40:14 +12:00
->param('prefs', '', new Assoc(), 'Prefs key-value JSON object.')
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-04-04 18:30:07 +12:00
->inject('events')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, array $prefs, Response $response, Database $dbForProject, Stats $usage, Event $events) {
$user = $dbForProject->getDocument('users', $userId);
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2020-06-30 23:09:28 +12:00
}
2020-01-20 09:38:00 +13:00
$user = $dbForProject->updateDocument('users', $user->getId(), $user->setAttribute('prefs', $prefs));
2019-10-21 19:01:07 +13:00
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.update', 1)
;
2022-04-04 18:30:07 +12:00
$events
->setParam('userId', $user->getId())
;
2021-07-26 02:47:18 +12:00
$response->dynamic(new Document($prefs), Response::MODEL_PREFERENCES);
2020-12-27 05:54:42 +13:00
});
2020-06-29 05:31:21 +12:00
App::delete('/v1/users/:userId/sessions/:sessionId')
2019-05-09 18:54:39 +12:00
->desc('Delete User Session')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].sessions.[sessionId].delete')
2019-05-09 18:54:39 +12:00
->label('scope', 'users.write')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'deleteSession')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/users/delete-user-session.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.model', Response::MODEL_NONE)
->param('userId', '', new UID(), 'User ID.')
->param('sessionId', null, new UID(), 'Session ID.')
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2020-12-27 05:54:42 +13:00
->inject('events')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, string $sessionId, Response $response, Database $dbForProject, Event $events, Stats $usage) {
2019-05-09 18:54:39 +12:00
$user = $dbForProject->getDocument('users', $userId);
2019-05-09 18:54:39 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2020-06-30 23:09:28 +12:00
}
2019-05-09 18:54:39 +12:00
2022-04-04 21:59:32 +12:00
$session = $dbForProject->getDocument('sessions', $sessionId);
2021-02-20 02:59:36 +13:00
2022-05-24 02:54:50 +12:00
if ($session->isEmpty()) {
2022-04-04 18:30:07 +12:00
throw new Exception('Session not found', 404, Exception::USER_SESSION_NOT_FOUND);
2019-05-09 18:54:39 +12:00
}
2022-04-04 21:59:32 +12:00
$dbForProject->deleteDocument('sessions', $session->getId());
2022-04-26 22:36:49 +12:00
$dbForProject->deleteCachedDocument('users', $user->getId());
2022-04-04 18:30:07 +12:00
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.update', 1)
->setParam('users.sessions.delete', 1)
;
2022-04-04 18:30:07 +12:00
$events
->setParam('userId', $user->getId())
->setParam('sessionId', $sessionId)
2022-04-04 18:30:07 +12:00
;
2020-10-31 08:53:27 +13:00
$response->noContent();
2020-12-27 05:54:42 +13:00
});
2019-05-09 18:54:39 +12:00
2020-06-29 05:31:21 +12:00
App::delete('/v1/users/:userId/sessions')
2019-05-09 18:54:39 +12:00
->desc('Delete User Sessions')
2020-06-26 06:32:12 +12:00
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].sessions.[sessionId].delete')
2019-05-09 18:54:39 +12:00
->label('scope', 'users.write')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'users')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'deleteSessions')
2019-10-09 21:31:51 +13:00
->label('sdk.description', '/docs/references/users/delete-user-sessions.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.model', Response::MODEL_NONE)
->param('userId', '', new UID(), 'User ID.')
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2020-12-27 05:54:42 +13:00
->inject('events')
2021-08-16 19:56:18 +12:00
->inject('usage')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, Response $response, Database $dbForProject, Event $events, Stats $usage) {
2019-05-09 18:54:39 +12:00
$user = $dbForProject->getDocument('users', $userId);
2019-05-09 18:54:39 +12:00
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
2020-06-30 23:09:28 +12:00
}
2019-05-09 18:54:39 +12:00
2021-07-17 22:04:43 +12:00
$sessions = $user->getAttribute('sessions', []);
foreach ($sessions as $key => $session) { /** @var Document $session */
$dbForProject->deleteDocument('sessions', $session->getId());
//TODO: fix this
2021-07-17 22:04:43 +12:00
}
2022-04-04 21:59:32 +12:00
$dbForProject->deleteCachedDocument('users', $user->getId());
2019-05-09 18:54:39 +12:00
2020-12-07 11:14:57 +13:00
$events
2022-04-04 18:30:07 +12:00
->setParam('userId', $user->getId())
2022-05-11 01:52:55 +12:00
->setPayload($response->output($user, Response::MODEL_USER))
2020-10-31 08:53:27 +13:00
;
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.update', 1)
->setParam('users.sessions.delete', 1)
;
2022-04-04 18:30:07 +12:00
2020-10-31 08:53:27 +13:00
$response->noContent();
2020-12-27 05:54:42 +13:00
});
App::delete('/v1/users/:userId')
->desc('Delete User')
->groups(['api', 'users'])
2022-04-04 18:30:07 +12:00
->label('event', 'users.[userId].delete')
->label('scope', 'users.write')
2021-04-16 19:22:17 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
->label('sdk.namespace', 'users')
2021-04-13 23:01:33 +12:00
->label('sdk.method', 'delete')
->label('sdk.description', '/docs/references/users/delete.md')
2020-11-12 10:02:24 +13:00
->label('sdk.response.code', Response::STATUS_CODE_NOCONTENT)
->label('sdk.response.model', Response::MODEL_NONE)
2022-06-01 02:24:01 +12:00
->param('userId', '', new UID(), 'User ID.')
2020-12-27 05:54:42 +13:00
->inject('response')
->inject('dbForProject')
2020-12-27 05:54:42 +13:00
->inject('events')
->inject('deletes')
2021-08-16 20:53:34 +12:00
->inject('usage')
2022-05-26 01:36:25 +12:00
->action(function (string $userId, Response $response, Database $dbForProject, Event $events, Delete $deletes, Stats $usage) {
2022-04-04 18:30:07 +12:00
$user = $dbForProject->getDocument('users', $userId);
2022-05-16 21:58:17 +12:00
if ($user->isEmpty()) {
2022-02-07 01:49:01 +13:00
throw new Exception('User not found', 404, Exception::USER_NOT_FOUND);
}
// clone user object to send to workers
$clone = clone $user;
2022-05-16 21:58:17 +12:00
$dbForProject->deleteDocument('users', $userId);
2020-10-31 08:53:27 +13:00
$deletes
2022-04-18 08:34:32 +12:00
->setType(DELETE_TYPE_DOCUMENT)
->setDocument($clone)
2020-10-31 08:53:27 +13:00
;
2020-12-07 11:14:57 +13:00
$events
2022-04-04 18:30:07 +12:00
->setParam('userId', $user->getId())
2022-05-11 01:52:55 +12:00
->setPayload($response->output($clone, Response::MODEL_USER))
2020-10-31 08:53:27 +13:00
;
2021-08-16 19:56:18 +12:00
$usage
->setParam('users.delete', 1)
;
$response->noContent();
2020-12-27 05:54:42 +13:00
});
App::get('/v1/users/usage')
2021-08-21 00:10:52 +12:00
->desc('Get usage stats for the users API')
->groups(['api', 'users'])
->label('scope', 'users.read')
->label('sdk.auth', [APP_AUTH_TYPE_ADMIN])
->label('sdk.namespace', 'users')
->label('sdk.method', 'getUsage')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_USAGE_USERS)
->param('range', '30d', new WhiteList(['24h', '7d', '30d', '90d'], true), 'Date range.', true)
2022-05-24 02:54:50 +12:00
->param('provider', '', new WhiteList(\array_merge(['email', 'anonymous'], \array_map(fn($value) => "oauth-" . $value, \array_keys(Config::getParam('providers', [])))), true), 'Provider Name.', true)
->inject('response')
->inject('dbForProject')
->inject('register')
2022-05-26 01:36:25 +12:00
->action(function (string $range, string $provider, Response $response, Database $dbForProject) {
$usage = [];
if (App::getEnv('_APP_USAGE_STATS', 'enabled') == 'enabled') {
2021-10-28 11:17:15 +13:00
$periods = [
'24h' => [
'period' => '30m',
'limit' => 48,
],
'7d' => [
'period' => '1d',
'limit' => 7,
],
'30d' => [
'period' => '1d',
'limit' => 30,
],
'90d' => [
'period' => '1d',
'limit' => 90,
],
];
$metrics = [
"users.count",
"users.create",
"users.read",
"users.update",
"users.delete",
"users.sessions.create",
"users.sessions.$provider.create",
"users.sessions.delete"
];
$stats = [];
2022-05-24 02:54:50 +12:00
Authorization::skip(function () use ($dbForProject, $periods, $range, $metrics, &$stats) {
foreach ($metrics as $metric) {
2021-10-28 11:17:15 +13:00
$limit = $periods[$range]['limit'];
$period = $periods[$range]['period'];
$requestDocs = $dbForProject->find('stats', [
new Query('period', Query::TYPE_EQUAL, [$period]),
new Query('metric', Query::TYPE_EQUAL, [$metric]),
], $limit, 0, ['time'], [Database::ORDER_DESC]);
2022-05-24 02:54:50 +12:00
$stats[$metric] = [];
foreach ($requestDocs as $requestDoc) {
$stats[$metric][] = [
'value' => $requestDoc->getAttribute('value'),
'date' => $requestDoc->getAttribute('time'),
];
}
// backfill metrics with empty values for graphs
$backfill = $limit - \count($requestDocs);
while ($backfill > 0) {
$last = $limit - $backfill - 1; // array index of last added metric
2022-05-24 02:54:50 +12:00
$diff = match ($period) { // convert period to seconds for unix timestamp math
'30m' => 1800,
'1d' => 86400,
};
$stats[$metric][] = [
'value' => 0,
2021-10-28 11:17:51 +13:00
'date' => ($stats[$metric][$last]['date'] ?? \time()) - $diff, // time of last metric minus period
];
$backfill--;
}
$stats[$metric] = array_reverse($stats[$metric]);
2022-05-24 02:54:50 +12:00
}
});
$usage = new Document([
'range' => $range,
2021-10-27 02:19:28 +13:00
'usersCount' => $stats["users.count"],
'usersCreate' => $stats["users.create"],
'usersRead' => $stats["users.read"],
'usersUpdate' => $stats["users.update"],
'usersDelete' => $stats["users.delete"],
'sessionsCreate' => $stats["users.sessions.create"],
'sessionsProviderCreate' => $stats["users.sessions.$provider.create"],
'sessionsDelete' => $stats["users.sessions.delete"]
]);
}
$response->dynamic($usage, Response::MODEL_USAGE_USERS);
2022-05-24 02:54:50 +12:00
});