1
0
Fork 0
mirror of synced 2024-06-02 19:04:49 +12:00
appwrite/app/controllers/api/teams.php

601 lines
24 KiB
PHP
Raw Normal View History

2019-05-09 18:54:39 +12:00
<?php
global $utopia, $register, $request, $response, $projectDB, $project, $user, $audit, $mode, $clients;
2019-05-09 18:54:39 +12:00
use Utopia\Exception;
use Utopia\Response;
2019-05-09 18:54:39 +12:00
use Utopia\Validator\Email;
use Utopia\Validator\Text;
use Utopia\Validator\Host;
use Utopia\Validator\Range;
use Utopia\Validator\ArrayList;
use Utopia\Validator\WhiteList;
use Utopia\Locale\Locale;
use Appwrite\Auth\Auth;
use Appwrite\Database\Database;
use Appwrite\Database\Document;
use Appwrite\Database\Validator\UID;
use Appwrite\Database\Validator\Authorization;
use Appwrite\Database\Exception\Duplicate;
use Appwrite\Template\Template;
2019-05-09 18:54:39 +12:00
2019-12-17 08:35:33 +13:00
include_once __DIR__ . '/../shared/api.php';
2019-11-30 07:26:06 +13:00
2019-05-09 18:54:39 +12:00
$utopia->post('/v1/teams')
->desc('Create Team')
->label('scope', 'teams.write')
2020-01-27 19:14:14 +13:00
->label('sdk.platform', [APP_PLATFORM_CLIENT, APP_PLATFORM_SERVER])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'teams')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'create')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/teams/create-team.md')
->param('name', null, function () { return new Text(100); }, 'Team name.')
2020-02-03 08:30:40 +13:00
->param('roles', ['owner'], function () { return new ArrayList(new Text(128)); }, 'Array of strings. Use this param to set the roles in the team for the user who created it. The default role is **owner**. A role can be any string. Learn more about [roles and permissions](/docs/permissions).', true)
2019-05-09 18:54:39 +12:00
->action(
function ($name, $roles) use ($response, $projectDB, $user, $mode) {
2019-05-09 18:54:39 +12:00
Authorization::disable();
$team = $projectDB->createDocument([
'$collection' => Database::SYSTEM_COLLECTION_TEAMS,
'$permissions' => [
'read' => ['team:{self}'],
'write' => ['team:{self}/owner'],
2019-05-09 18:54:39 +12:00
],
'name' => $name,
2020-02-17 20:16:11 +13:00
'sum' => ($mode !== APP_MODE_ADMIN && $user->getId()) ? 1 : 0,
2019-05-09 18:54:39 +12:00
'dateCreated' => time(),
]);
2020-02-11 19:17:40 +13:00
Authorization::reset();
2019-05-09 18:54:39 +12:00
if (false === $team) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed saving team to DB', 500);
}
2020-02-17 20:16:11 +13:00
if ($mode !== APP_MODE_ADMIN && $user->getId()) { // Don't add user on server mode
2019-05-09 18:54:39 +12:00
$membership = new Document([
'$collection' => Database::SYSTEM_COLLECTION_MEMBERSHIPS,
'$permissions' => [
2020-02-17 20:16:11 +13:00
'read' => ['user:'.$user->getId(), 'team:'.$team->getId()],
'write' => ['user:'.$user->getId(), 'team:'.$team->getId().'/owner'],
2019-05-09 18:54:39 +12:00
],
2020-02-17 20:16:11 +13:00
'userId' => $user->getId(),
'teamId' => $team->getId(),
2019-05-09 18:54:39 +12:00
'roles' => $roles,
'invited' => time(),
'joined' => time(),
'confirm' => true,
'secret' => '',
]);
// Attach user to team
$user->setAttribute('memberships', $membership, Document::SET_TYPE_APPEND);
$user = $projectDB->updateDocument($user->getArrayCopy());
if (false === $user) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed saving user to DB', 500);
}
}
$response
->setStatusCode(Response::STATUS_CODE_CREATED)
->json($team->getArrayCopy())
;
2019-05-09 18:54:39 +12:00
}
);
2020-02-01 11:34:07 +13:00
$utopia->get('/v1/teams')
->desc('List Teams')
->label('scope', 'teams.read')
->label('sdk.platform', [APP_PLATFORM_CLIENT, APP_PLATFORM_SERVER])
->label('sdk.namespace', 'teams')
->label('sdk.method', 'list')
->label('sdk.description', '/docs/references/teams/list-teams.md')
->param('search', '', function () { return new Text(256); }, 'Search term to filter your list results.', true)
->param('limit', 25, function () { return new Range(0, 100); }, 'Results limit value. By default will return maximum 25 results. Maximum of 100 results allowed per request.', true)
->param('offset', 0, function () { return new Range(0, 2000); }, 'Results offset. The default value is 0. Use this param to manage pagination.', true)
->param('orderType', 'ASC', function () { return new WhiteList(['ASC', 'DESC']); }, 'Order result by ASC or DESC order.', true)
->action(
function ($search, $limit, $offset, $orderType) use ($response, $projectDB) {
$results = $projectDB->getCollection([
'limit' => $limit,
'offset' => $offset,
'orderField' => 'dateCreated',
'orderType' => $orderType,
'orderCast' => 'int',
'search' => $search,
'filters' => [
'$collection='.Database::SYSTEM_COLLECTION_TEAMS,
],
]);
$response->json(['sum' => $projectDB->getSum(), 'teams' => $results]);
}
);
$utopia->get('/v1/teams/:teamId')
->desc('Get Team')
->label('scope', 'teams.read')
->label('sdk.platform', [APP_PLATFORM_CLIENT, APP_PLATFORM_SERVER])
->label('sdk.namespace', 'teams')
->label('sdk.method', 'get')
->label('sdk.description', '/docs/references/teams/get-team.md')
->param('teamId', '', function () { return new UID(); }, 'Team unique ID.')
->action(
function ($teamId) use ($response, $projectDB) {
$team = $projectDB->getDocument($teamId);
2020-02-17 20:16:11 +13:00
if (empty($team->getId()) || Database::SYSTEM_COLLECTION_TEAMS != $team->getCollection()) {
2020-02-01 11:34:07 +13:00
throw new Exception('Team not found', 404);
}
$response->json($team->getArrayCopy([]));
}
);
2019-05-09 18:54:39 +12:00
$utopia->put('/v1/teams/:teamId')
->desc('Update Team')
->label('scope', 'teams.write')
2020-01-27 19:14:14 +13:00
->label('sdk.platform', [APP_PLATFORM_CLIENT, APP_PLATFORM_SERVER])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'teams')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'update')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/teams/update-team.md')
->param('teamId', '', function () { return new UID(); }, 'Team unique ID.')
->param('name', null, function () { return new Text(100); }, 'Team name.')
2019-05-09 18:54:39 +12:00
->action(
function ($teamId, $name) use ($response, $projectDB) {
2019-05-09 18:54:39 +12:00
$team = $projectDB->getDocument($teamId);
2020-02-17 20:16:11 +13:00
if (empty($team->getId()) || Database::SYSTEM_COLLECTION_TEAMS != $team->getCollection()) {
2019-05-09 18:54:39 +12:00
throw new Exception('Team not found', 404);
}
$team = $projectDB->updateDocument(array_merge($team->getArrayCopy(), [
'name' => $name,
]));
if (false === $team) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed saving team to DB', 500);
}
$response->json($team->getArrayCopy());
}
);
$utopia->delete('/v1/teams/:teamId')
->desc('Delete Team')
->label('scope', 'teams.write')
2020-01-27 19:14:14 +13:00
->label('sdk.platform', [APP_PLATFORM_CLIENT, APP_PLATFORM_SERVER])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'teams')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'delete')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/teams/delete-team.md')
->param('teamId', '', function () { return new UID(); }, 'Team unique ID.')
2019-05-09 18:54:39 +12:00
->action(
function ($teamId) use ($response, $projectDB) {
2019-05-09 18:54:39 +12:00
$team = $projectDB->getDocument($teamId);
2020-02-17 20:16:11 +13:00
if (empty($team->getId()) || Database::SYSTEM_COLLECTION_TEAMS != $team->getCollection()) {
2019-05-09 18:54:39 +12:00
throw new Exception('Team not found', 404);
}
$memberships = $projectDB->getCollection([
'limit' => 2000, // TODO add members limit
'offset' => 0,
'filters' => [
'$collection='.Database::SYSTEM_COLLECTION_MEMBERSHIPS,
'teamId='.$teamId,
],
2019-05-09 18:54:39 +12:00
]);
foreach ($memberships as $member) {
2020-02-17 20:16:11 +13:00
if (!$projectDB->deleteDocument($member->getId())) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed to remove membership for team from DB', 500);
}
}
if (!$projectDB->deleteDocument($teamId)) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed to remove team from DB', 500);
}
$response->noContent();
}
);
$utopia->post('/v1/teams/:teamId/memberships')
->desc('Create Team Membership')
2020-02-10 05:53:33 +13:00
->label('scope', 'teams.write')
2020-01-27 19:14:14 +13:00
->label('sdk.platform', [APP_PLATFORM_CLIENT, APP_PLATFORM_SERVER])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'teams')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'createMembership')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/teams/create-team-membership.md')
->param('teamId', '', function () { return new UID(); }, 'Team unique ID.')
2020-02-10 18:58:29 +13:00
->param('email', '', function () { return new Email(); }, 'New team member email.')
->param('name', '', function () { return new Text(100); }, 'New team member name.', true)
2020-02-03 08:30:40 +13:00
->param('roles', [], function () { return new ArrayList(new Text(128)); }, 'Array of strings. Use this param to set the user roles in the team. A role can be any string. Learn more about [roles and permissions](/docs/permissions).')
2020-01-19 03:47:08 +13:00
->param('url', '', function () use ($clients) { return new Host($clients); }, 'URL to redirect the user back to your app from the invitation email.') // TODO add our own built-in confirm page
2019-05-09 18:54:39 +12:00
->action(
2020-01-19 03:47:08 +13:00
function ($teamId, $email, $name, $roles, $url) use ($response, $register, $project, $user, $audit, $projectDB) {
2019-05-09 18:54:39 +12:00
$name = (empty($name)) ? $email : $name;
$team = $projectDB->getDocument($teamId);
2020-02-17 20:16:11 +13:00
if (empty($team->getId()) || Database::SYSTEM_COLLECTION_TEAMS != $team->getCollection()) {
2019-05-09 18:54:39 +12:00
throw new Exception('Team not found', 404);
}
$memberships = $projectDB->getCollection([
'limit' => 50,
'offset' => 0,
'filters' => [
'$collection='.Database::SYSTEM_COLLECTION_MEMBERSHIPS,
2020-02-17 20:16:11 +13:00
'teamId='.$team->getId(),
],
2019-05-09 18:54:39 +12:00
]);
$invitee = $projectDB->getCollection([ // Get user by email address
'limit' => 1,
'first' => true,
'filters' => [
'$collection='.Database::SYSTEM_COLLECTION_USERS,
'email='.$email,
],
2019-05-09 18:54:39 +12:00
]);
if (empty($invitee)) { // Create new user if no user with same email found
2019-05-09 18:54:39 +12:00
Authorization::disable();
2020-03-22 10:10:06 +13:00
try {
$invitee = $projectDB->createDocument([
'$collection' => Database::SYSTEM_COLLECTION_USERS,
'$permissions' => [
'read' => ['user:{self}', '*'],
'write' => ['user:{self}'],
],
'email' => $email,
'emailVerification' => false,
'status' => Auth::USER_STATUS_UNACTIVATED,
'password' => Auth::passwordHash(Auth::passwordGenerator()),
'password-update' => time(),
'registration' => time(),
'reset' => false,
'name' => $name,
'tokens' => [],
], ['email' => $email]);
} catch (Duplicate $th) {
throw new Exception('Account already exists', 409);
}
2019-05-09 18:54:39 +12:00
2020-02-11 19:17:40 +13:00
Authorization::reset();
2019-05-09 18:54:39 +12:00
if (false === $invitee) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed saving user to DB', 500);
}
}
$isOwner = false;
foreach ($memberships as $member) {
2020-02-17 20:16:11 +13:00
if ($member->getAttribute('userId') == $invitee->getId()) {
2020-01-23 12:31:48 +13:00
throw new Exception('User has already been invited or is already a member of this team', 409);
2019-05-09 18:54:39 +12:00
}
2020-02-17 20:16:11 +13:00
if ($member->getAttribute('userId') == $user->getId() && in_array('owner', $member->getAttribute('roles', []))) {
2019-05-09 18:54:39 +12:00
$isOwner = true;
}
}
if (!$isOwner) {
2019-05-09 18:54:39 +12:00
throw new Exception('User is not allowed to send invitations for this team', 401);
}
$secret = Auth::tokenGenerator();
$membership = new Document([
'$collection' => Database::SYSTEM_COLLECTION_MEMBERSHIPS,
'$permissions' => [
'read' => ['*'],
2020-02-17 20:16:11 +13:00
'write' => ['user:'.$invitee->getId(), 'team:'.$team->getId().'/owner'],
2019-05-09 18:54:39 +12:00
],
2020-02-17 20:16:11 +13:00
'userId' => $invitee->getId(),
'teamId' => $team->getId(),
2019-05-09 18:54:39 +12:00
'roles' => $roles,
'invited' => time(),
'joined' => 0,
'confirm' => false,
'secret' => Auth::hash($secret),
]);
$membership = $projectDB->createDocument($membership->getArrayCopy());
if (false === $membership) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed saving membership to DB', 500);
}
2020-01-19 03:47:08 +13:00
$url = Template::parseURL($url);
2020-02-17 20:16:11 +13:00
$url['query'] = Template::mergeQuery(((isset($url['query'])) ? $url['query'] : ''), ['inviteId' => $membership->getId(), 'teamId' => $team->getId(), 'userId' => $invitee->getId(), 'secret' => $secret, 'teamId' => $teamId]);
2020-01-19 03:47:08 +13:00
$url = Template::unParseURL($url);
2019-05-09 18:54:39 +12:00
2020-02-10 20:34:02 +13:00
$body = new Template(__DIR__.'/../../config/locales/templates/'.Locale::getText('account.emails.invitation.body'));
2019-05-09 18:54:39 +12:00
$body
->setParam('{{direction}}', Locale::getText('settings.direction'))
->setParam('{{project}}', $project->getAttribute('name', ['[APP-NAME]']))
->setParam('{{team}}', $team->getAttribute('name', '[TEAM-NAME]'))
->setParam('{{owner}}', $user->getAttribute('name', ''))
2020-01-19 03:47:08 +13:00
->setParam('{{redirect}}', $url)
2019-05-09 18:54:39 +12:00
;
2019-08-31 18:02:21 +12:00
$mail = $register->get('smtp'); /* @var $mail \PHPMailer\PHPMailer\PHPMailer */
2019-05-09 18:54:39 +12:00
2019-08-09 09:49:46 +12:00
$mail->addAddress($email, $name);
2020-02-10 20:34:02 +13:00
$mail->Subject = sprintf(Locale::getText('account.emails.invitation.title'), $team->getAttribute('name', '[TEAM-NAME]'), $project->getAttribute('name', ['[APP-NAME]']));
$mail->Body = $body->render();
2019-08-09 09:49:46 +12:00
$mail->AltBody = strip_tags($body->render());
2019-05-09 18:54:39 +12:00
2019-08-09 09:49:46 +12:00
try {
$mail->send();
} catch (\Exception $error) {
2020-01-12 02:58:02 +13:00
throw new Exception('Error sending mail: ' . $error->getMessage(), 500);
2019-05-09 18:54:39 +12:00
}
$audit
2020-02-17 20:16:11 +13:00
->setParam('userId', $invitee->getId())
->setParam('event', 'teams.membership.create')
2020-02-12 11:24:30 +13:00
->setParam('resource', 'teams/'.$teamId)
2019-05-09 18:54:39 +12:00
;
$response
2020-01-19 03:47:08 +13:00
->setStatusCode(Response::STATUS_CODE_CREATED) // TODO change response of this endpoint
2020-01-20 09:02:50 +13:00
->json(array_merge($membership->getArrayCopy([
2020-02-17 20:16:11 +13:00
'$id',
2020-01-19 05:03:17 +13:00
'userId',
'teamId',
'roles',
'invited',
'joined',
'confirm',
2020-01-20 09:02:50 +13:00
]), [
'email' => $email,
'name' => $name,
2020-01-19 05:03:17 +13:00
]))
2020-01-19 03:47:08 +13:00
;
2019-05-09 18:54:39 +12:00
}
);
2020-02-01 11:34:07 +13:00
$utopia->get('/v1/teams/:teamId/memberships')
->desc('Get Team Memberships')
->label('scope', 'teams.read')
->label('sdk.platform', [APP_PLATFORM_CLIENT, APP_PLATFORM_SERVER])
->label('sdk.namespace', 'teams')
->label('sdk.method', 'getMemberships')
->label('sdk.description', '/docs/references/teams/get-team-members.md')
->param('teamId', '', function () { return new UID(); }, 'Team unique ID.')
->action(
function ($teamId) use ($response, $projectDB) {
$team = $projectDB->getDocument($teamId);
2020-02-17 20:16:11 +13:00
if (empty($team->getId()) || Database::SYSTEM_COLLECTION_TEAMS != $team->getCollection()) {
2020-02-01 11:34:07 +13:00
throw new Exception('Team not found', 404);
}
$memberships = $projectDB->getCollection([
'limit' => 50,
'offset' => 0,
'filters' => [
'$collection='.Database::SYSTEM_COLLECTION_MEMBERSHIPS,
'teamId='.$teamId,
],
]);
$users = [];
foreach ($memberships as $membership) {
if (empty($membership->getAttribute('userId', null))) {
continue;
}
$temp = $projectDB->getDocument($membership->getAttribute('userId', null))->getArrayCopy(['email', 'name']);
$users[] = array_merge($temp, $membership->getArrayCopy([
2020-02-17 20:16:11 +13:00
'$id',
2020-02-01 11:34:07 +13:00
'userId',
'teamId',
'roles',
'invited',
'joined',
'confirm',
]));
}
usort($users, function ($a, $b) {
if ($a['joined'] === 0 || $b['joined'] === 0) {
return $b['joined'] - $a['joined'];
}
return $a['joined'] - $b['joined'];
});
$response->json($users);
}
);
2019-05-09 18:54:39 +12:00
$utopia->patch('/v1/teams/:teamId/memberships/:inviteId/status')
->desc('Update Team Membership Status')
2020-01-20 01:22:54 +13:00
->label('scope', 'public')
2020-01-27 19:14:14 +13:00
->label('sdk.platform', [APP_PLATFORM_CLIENT])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'teams')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'updateMembershipStatus')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/teams/update-team-membership-status.md')
->param('teamId', '', function () { return new UID(); }, 'Team unique ID.')
2020-02-10 18:54:26 +13:00
->param('inviteId', '', function () { return new UID(); }, 'Invite unique ID.')
->param('userId', '', function () { return new UID(); }, 'User unique ID.')
->param('secret', '', function () { return new Text(256); }, 'Secret key.')
2019-05-09 18:54:39 +12:00
->action(
function ($teamId, $inviteId, $userId, $secret) use ($response, $request, $user, $audit, $projectDB, $protocol, $domainVerification) {
2020-01-19 09:02:47 +13:00
$membership = $projectDB->getDocument($inviteId);
2019-05-09 18:54:39 +12:00
2020-02-17 20:16:11 +13:00
if (empty($membership->getId()) || Database::SYSTEM_COLLECTION_MEMBERSHIPS != $membership->getCollection()) {
2019-05-09 18:54:39 +12:00
throw new Exception('Invite not found', 404);
}
2020-01-20 01:22:54 +13:00
if ($membership->getAttribute('teamId') !== $teamId) {
2019-05-09 18:54:39 +12:00
throw new Exception('Team IDs don\'t match', 404);
}
2020-01-20 01:22:54 +13:00
Authorization::disable();
2019-05-09 18:54:39 +12:00
$team = $projectDB->getDocument($teamId);
2020-01-20 01:22:54 +13:00
2020-02-11 19:17:40 +13:00
Authorization::reset();
2019-05-09 18:54:39 +12:00
2020-02-17 20:16:11 +13:00
if (empty($team->getId()) || Database::SYSTEM_COLLECTION_TEAMS != $team->getCollection()) {
2019-05-09 18:54:39 +12:00
throw new Exception('Team not found', 404);
}
2020-01-19 09:02:47 +13:00
if (Auth::hash($secret) !== $membership->getAttribute('secret')) {
2019-05-09 18:54:39 +12:00
throw new Exception('Secret key not valid', 401);
}
2020-01-19 09:02:47 +13:00
if ($userId != $membership->getAttribute('userId')) {
throw new Exception('Invite not belong to current user ('.$user->getAttribute('email').')', 401);
2019-05-09 18:54:39 +12:00
}
2020-02-17 20:16:11 +13:00
if (empty($user->getId())) {
2019-05-09 18:54:39 +12:00
$user = $projectDB->getCollection([ // Get user
'limit' => 1,
'first' => true,
'filters' => [
'$collection='.Database::SYSTEM_COLLECTION_USERS,
2020-02-17 20:16:11 +13:00
'$id='.$userId,
],
2019-05-09 18:54:39 +12:00
]);
}
2020-02-17 20:16:11 +13:00
if ($membership->getAttribute('userId') !== $user->getId()) {
throw new Exception('Invite not belong to current user ('.$user->getAttribute('email').')', 401);
2019-05-09 18:54:39 +12:00
}
2020-01-19 09:02:47 +13:00
$membership // Attach user to team
2019-05-09 18:54:39 +12:00
->setAttribute('joined', time())
->setAttribute('confirm', true)
;
$user
2020-02-10 10:37:28 +13:00
->setAttribute('emailVerification', true)
2020-01-19 09:02:47 +13:00
->setAttribute('memberships', $membership, Document::SET_TYPE_APPEND)
;
2019-05-09 18:54:39 +12:00
// Log user in
$expiry = time() + Auth::TOKEN_EXPIRATION_LOGIN_LONG;
$secret = Auth::tokenGenerator();
$user->setAttribute('tokens', new Document([
'$collection' => Database::SYSTEM_COLLECTION_TOKENS,
2020-02-17 20:16:11 +13:00
'$permissions' => ['read' => ['user:'.$user->getId()], 'write' => ['user:'.$user->getId()]],
2019-05-09 18:54:39 +12:00
'type' => Auth::TOKEN_TYPE_LOGIN,
'secret' => Auth::hash($secret), // On way hash encryption to protect DB leak
'expire' => $expiry,
'userAgent' => $request->getServer('HTTP_USER_AGENT', 'UNKNOWN'),
'ip' => $request->getIP(),
]), Document::SET_TYPE_APPEND);
2019-05-09 18:54:39 +12:00
Authorization::setRole('user:'.$userId);
2019-05-09 18:54:39 +12:00
$user = $projectDB->updateDocument($user->getArrayCopy());
if (false === $user) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed saving user to DB', 500);
}
2020-01-20 01:22:54 +13:00
Authorization::disable();
2019-05-09 18:54:39 +12:00
$team = $projectDB->updateDocument(array_merge($team->getArrayCopy(), [
'sum' => $team->getAttribute('sum', 0) + 1,
]));
2020-02-11 19:17:40 +13:00
Authorization::reset();
2020-01-20 09:02:50 +13:00
if (false === $team) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed saving team to DB', 500);
}
$audit
2020-02-17 20:16:11 +13:00
->setParam('userId', $user->getId())
->setParam('event', 'teams.membership.update')
2020-02-12 11:24:30 +13:00
->setParam('resource', 'teams/'.$teamId)
2019-05-09 18:54:39 +12:00
;
if(!$domainVerification) {
$response
->addHeader('X-Fallback-Cookies', json_encode([Auth::$cookieName => Auth::encodeSession($user->getId(), $secret)]))
;
}
$response
2020-02-27 22:17:09 +13:00
->addCookie(Auth::$cookieName.'_legacy', Auth::encodeSession($user->getId(), $secret), $expiry, '/', COOKIE_DOMAIN, ('https' == $protocol), true, null)
->addCookie(Auth::$cookieName, Auth::encodeSession($user->getId(), $secret), $expiry, '/', COOKIE_DOMAIN, ('https' == $protocol), true, COOKIE_SAMESITE)
2020-01-20 09:02:50 +13:00
->json(array_merge($membership->getArrayCopy([
2020-02-17 20:16:11 +13:00
'$id',
2020-01-19 09:02:47 +13:00
'userId',
'teamId',
'roles',
'invited',
'joined',
'confirm',
2020-01-20 09:02:50 +13:00
]), [
'email' => $user->getAttribute('email'),
'name' => $user->getAttribute('name'),
2020-01-19 09:02:47 +13:00
]))
;
2019-05-09 18:54:39 +12:00
}
);
$utopia->delete('/v1/teams/:teamId/memberships/:inviteId')
->desc('Delete Team Membership')
2020-02-10 05:53:33 +13:00
->label('scope', 'teams.write')
2020-01-27 19:14:14 +13:00
->label('sdk.platform', [APP_PLATFORM_CLIENT, APP_PLATFORM_SERVER])
2019-05-09 18:54:39 +12:00
->label('sdk.namespace', 'teams')
2020-01-31 05:18:46 +13:00
->label('sdk.method', 'deleteMembership')
2019-10-08 20:09:35 +13:00
->label('sdk.description', '/docs/references/teams/delete-team-membership.md')
->param('teamId', '', function () { return new UID(); }, 'Team unique ID.')
2020-02-10 18:54:26 +13:00
->param('inviteId', '', function () { return new UID(); }, 'Invite unique ID.')
2019-05-09 18:54:39 +12:00
->action(
function ($teamId, $inviteId) use ($response, $projectDB, $audit) {
2020-01-19 05:03:17 +13:00
$membership = $projectDB->getDocument($inviteId);
2019-05-09 18:54:39 +12:00
2020-02-17 20:16:11 +13:00
if (empty($membership->getId()) || Database::SYSTEM_COLLECTION_MEMBERSHIPS != $membership->getCollection()) {
2019-05-09 18:54:39 +12:00
throw new Exception('Invite not found', 404);
}
2020-01-19 05:03:17 +13:00
if ($membership->getAttribute('teamId') !== $teamId) {
2019-05-09 18:54:39 +12:00
throw new Exception('Team IDs don\'t match', 404);
}
$team = $projectDB->getDocument($teamId);
2020-02-17 20:16:11 +13:00
if (empty($team->getId()) || Database::SYSTEM_COLLECTION_TEAMS != $team->getCollection()) {
2019-05-09 18:54:39 +12:00
throw new Exception('Team not found', 404);
}
2020-02-17 20:16:11 +13:00
if (!$projectDB->deleteDocument($membership->getId())) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed to remove membership from DB', 500);
}
$team = $projectDB->updateDocument(array_merge($team->getArrayCopy(), [
'sum' => $team->getAttribute('sum', 0) - 1,
]));
if (false === $team) {
2019-05-09 18:54:39 +12:00
throw new Exception('Failed saving team to DB', 500);
}
$audit
2020-01-19 05:03:17 +13:00
->setParam('userId', $membership->getAttribute('userId'))
->setParam('event', 'teams.membership.delete')
2020-02-12 11:24:30 +13:00
->setParam('resource', 'teams/'.$teamId)
2019-05-09 18:54:39 +12:00
;
$response->noContent();
}
);