0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2025-01-06 20:40:08 -05:00
logto/packages/core/src/routes/admin-user.ts

273 lines
6.5 KiB
TypeScript
Raw Normal View History

import { passwordRegEx, usernameRegEx } from '@logto/core-kit';
import { arbitraryObjectGuard, userInfoSelectFields } from '@logto/schemas';
import { has } from '@silverhand/essentials';
import pick from 'lodash.pick';
import { literal, object, string } from 'zod';
2022-02-16 03:34:32 -05:00
import RequestError from '@/errors/RequestError';
import { encryptUserPassword, generateUserId, insertUser } from '@/lib/user';
import koaGuard from '@/middleware/koa-guard';
2022-02-16 02:55:08 -05:00
import koaPagination from '@/middleware/koa-pagination';
import { findRolesByRoleNames } from '@/queries/roles';
2022-02-16 03:34:32 -05:00
import {
deleteUserById,
deleteUserIdentity,
findUsers,
countUsers,
2022-02-16 03:34:32 -05:00
findUserById,
hasUser,
updateUserById,
} from '@/queries/user';
import assertThat from '@/utils/assert-that';
import { AuthedRouter } from './types';
export default function adminUserRoutes<T extends AuthedRouter>(router: T) {
router.get(
'/users',
koaPagination(),
koaGuard({
query: object({ search: string().optional(), hideAdminUser: literal('true').optional() }),
}),
async (ctx, next) => {
const { limit, offset } = ctx.pagination;
const {
query: { search, hideAdminUser: _hideAdminUser },
} = ctx.guard;
2022-02-16 02:55:08 -05:00
const hideAdminUser = _hideAdminUser === 'true';
const [{ count }, users] = await Promise.all([
countUsers(search, hideAdminUser),
findUsers(limit, offset, search, hideAdminUser),
]);
2022-02-16 02:55:08 -05:00
ctx.pagination.totalCount = count;
ctx.body = users.map((user) => pick(user, ...userInfoSelectFields));
return next();
}
);
2022-02-16 02:56:51 -05:00
router.get(
'/users/:userId',
koaGuard({
params: object({ userId: string() }),
2022-02-16 02:56:51 -05:00
}),
async (ctx, next) => {
const {
params: { userId },
} = ctx.guard;
const user = await findUserById(userId);
2022-02-16 02:56:51 -05:00
ctx.body = pick(user, ...userInfoSelectFields);
return next();
}
);
router.get(
'/users/:userId/custom-data',
koaGuard({
params: object({ userId: string() }),
response: arbitraryObjectGuard,
}),
async (ctx, next) => {
const {
params: { userId },
} = ctx.guard;
const { customData } = await findUserById(userId);
ctx.body = customData;
return next();
}
);
router.patch(
'/users/:userId/custom-data',
koaGuard({
params: object({ userId: string() }),
body: object({ customData: arbitraryObjectGuard }),
response: arbitraryObjectGuard,
}),
async (ctx, next) => {
const {
params: { userId },
body: { customData },
} = ctx.guard;
await findUserById(userId);
const user = await updateUserById(userId, {
customData,
});
ctx.body = user.customData;
return next();
}
);
2022-02-16 03:34:32 -05:00
router.post(
'/users',
koaGuard({
body: object({
username: string().regex(usernameRegEx),
password: string().regex(passwordRegEx),
2022-06-13 22:36:03 -05:00
name: string(),
2022-02-16 03:34:32 -05:00
}),
}),
async (ctx, next) => {
const { username, password, name } = ctx.guard.body;
assertThat(
!(await hasUser(username)),
new RequestError({
code: 'user.username_exists_register',
status: 422,
})
);
const id = await generateUserId();
const { passwordEncrypted, passwordEncryptionMethod } = await encryptUserPassword(password);
2022-02-16 03:34:32 -05:00
const user = await insertUser({
id,
username,
passwordEncrypted,
passwordEncryptionMethod,
name,
});
ctx.body = pick(user, ...userInfoSelectFields);
return next();
}
);
router.patch(
'/users/:userId',
koaGuard({
params: object({ userId: string() }),
body: object({
2022-06-13 22:36:03 -05:00
name: string().nullable().optional(),
2022-06-21 22:36:43 -05:00
avatar: string().url().or(literal('')).nullable().optional(),
customData: arbitraryObjectGuard.optional(),
2022-04-20 23:38:54 -05:00
roleNames: string().array().optional(),
}),
}),
async (ctx, next) => {
const {
params: { userId },
body,
} = ctx.guard;
await findUserById(userId);
2022-04-20 23:38:54 -05:00
// Temp solution to validate the existence of input roleNames
2022-05-20 04:59:55 -05:00
if (body.roleNames?.length) {
2022-04-20 23:38:54 -05:00
const { roleNames } = body;
const roles = await findRolesByRoleNames(roleNames);
if (roles.length !== roleNames.length) {
const resourcesNotFound = roleNames.filter(
(roleName) => !roles.some(({ name }) => roleName === name)
);
throw new RequestError({
status: 400,
code: 'user.invalid_role_names',
data: {
roleNames: resourcesNotFound.join(','),
},
});
2022-04-20 23:38:54 -05:00
}
}
const user = await updateUserById(
userId,
{
...body,
},
'replace'
);
ctx.body = pick(user, ...userInfoSelectFields);
return next();
}
);
router.patch(
'/users/:userId/password',
koaGuard({
params: object({ userId: string() }),
body: object({ password: string().regex(passwordRegEx) }),
}),
async (ctx, next) => {
const {
params: { userId },
body: { password },
} = ctx.guard;
await findUserById(userId);
const { passwordEncrypted, passwordEncryptionMethod } = await encryptUserPassword(password);
const user = await updateUserById(userId, {
passwordEncrypted,
passwordEncryptionMethod,
});
ctx.body = pick(user, ...userInfoSelectFields);
return next();
}
);
router.delete(
'/users/:userId',
koaGuard({
params: object({ userId: string() }),
}),
async (ctx, next) => {
const {
params: { userId },
} = ctx.guard;
if (userId === ctx.auth.id) {
throw new RequestError('user.cannot_delete_self');
}
await findUserById(userId);
await deleteUserById(userId);
ctx.status = 204;
return next();
}
);
router.delete(
'/users/:userId/identities/:target',
koaGuard({ params: object({ userId: string(), target: string() }) }),
async (ctx, next) => {
const {
params: { userId, target },
} = ctx.guard;
const { identities } = await findUserById(userId);
if (!has(identities, target)) {
throw new RequestError({ code: 'user.identity_not_exists', status: 404 });
}
const updatedUser = await deleteUserIdentity(userId, target);
ctx.body = pick(updatedUser, ...userInfoSelectFields);
return next();
}
);
}