2022-11-16 14:35:38 +08:00
|
|
|
import { emailRegEx, passwordRegEx, phoneRegEx, usernameRegEx } from '@logto/core-kit';
|
2022-11-16 14:57:41 +08:00
|
|
|
import { arbitraryObjectGuard, userInfoSelectFields } from '@logto/schemas';
|
2022-11-18 12:25:31 +08:00
|
|
|
import { has } from '@silverhand/essentials';
|
2022-11-11 14:29:37 +08:00
|
|
|
import { argon2Verify } from 'hash-wasm';
|
2022-11-11 10:50:20 +08:00
|
|
|
import pick from 'lodash.pick';
|
|
|
|
import type { Provider } from 'oidc-provider';
|
2022-11-18 12:25:31 +08:00
|
|
|
import { object, string, unknown } from 'zod';
|
2022-11-11 10:50:20 +08:00
|
|
|
|
2022-11-21 16:38:24 +08:00
|
|
|
import { getLogtoConnectorById } from '#src/connectors/index.js';
|
|
|
|
import RequestError from '#src/errors/RequestError/index.js';
|
|
|
|
import { checkSessionHealth } from '#src/lib/session.js';
|
2022-11-22 01:26:17 +08:00
|
|
|
import { getUserInfoByAuthCode } from '#src/lib/social.js';
|
2022-12-05 14:14:21 +08:00
|
|
|
import { checkIdentifierCollision, encryptUserPassword } from '#src/lib/user.js';
|
2022-11-21 16:38:24 +08:00
|
|
|
import koaGuard from '#src/middleware/koa-guard.js';
|
2022-11-22 01:26:17 +08:00
|
|
|
import { deleteUserIdentity, findUserById, updateUserById } from '#src/queries/user.js';
|
2022-11-21 16:38:24 +08:00
|
|
|
import assertThat from '#src/utils/assert-that.js';
|
2022-11-11 10:50:20 +08:00
|
|
|
|
2022-11-21 16:38:24 +08:00
|
|
|
import { verificationTimeout } from './consts.js';
|
2022-12-05 14:14:21 +08:00
|
|
|
import type { AnonymousRouter } from './types.js';
|
2022-11-11 10:50:20 +08:00
|
|
|
|
2022-12-05 14:14:21 +08:00
|
|
|
export const profileRoute = '/profile';
|
2022-11-11 10:50:20 +08:00
|
|
|
|
|
|
|
export default function profileRoutes<T extends AnonymousRouter>(router: T, provider: Provider) {
|
|
|
|
router.get(profileRoute, async (ctx, next) => {
|
2022-11-16 14:57:41 +08:00
|
|
|
const { accountId: userId } = await provider.Session.get(ctx);
|
2022-11-11 10:50:20 +08:00
|
|
|
|
2022-11-16 14:57:41 +08:00
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
2022-11-11 10:50:20 +08:00
|
|
|
|
2022-11-16 14:57:41 +08:00
|
|
|
const user = await findUserById(userId);
|
2022-11-11 10:50:20 +08:00
|
|
|
|
|
|
|
ctx.body = pick(user, ...userInfoSelectFields);
|
2022-12-05 14:14:21 +08:00
|
|
|
ctx.status = 200;
|
2022-11-11 10:50:20 +08:00
|
|
|
|
|
|
|
return next();
|
|
|
|
});
|
|
|
|
|
2022-11-16 14:57:41 +08:00
|
|
|
router.patch(
|
|
|
|
profileRoute,
|
|
|
|
koaGuard({
|
|
|
|
body: object({
|
|
|
|
name: string().nullable().optional(),
|
|
|
|
avatar: string().nullable().optional(),
|
|
|
|
customData: arbitraryObjectGuard.optional(),
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
async (ctx, next) => {
|
|
|
|
const { accountId: userId } = await provider.Session.get(ctx);
|
|
|
|
|
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
|
|
|
|
|
|
|
const { name, avatar, customData } = ctx.guard.body;
|
|
|
|
|
|
|
|
await updateUserById(userId, { name, avatar, customData });
|
|
|
|
|
|
|
|
ctx.status = 204;
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2022-11-11 10:50:20 +08:00
|
|
|
router.patch(
|
|
|
|
`${profileRoute}/username`,
|
|
|
|
koaGuard({
|
|
|
|
body: object({ username: string().regex(usernameRegEx) }),
|
|
|
|
}),
|
|
|
|
async (ctx, next) => {
|
|
|
|
const userId = await checkSessionHealth(ctx, provider, verificationTimeout);
|
2022-11-16 14:57:41 +08:00
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
2022-11-11 10:50:20 +08:00
|
|
|
|
|
|
|
const { username } = ctx.guard.body;
|
2022-12-05 14:14:21 +08:00
|
|
|
await checkIdentifierCollision({ username }, userId);
|
2022-11-11 10:50:20 +08:00
|
|
|
|
|
|
|
const user = await updateUserById(userId, { username }, 'replace');
|
|
|
|
ctx.body = pick(user, ...userInfoSelectFields);
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
2022-11-11 14:29:37 +08:00
|
|
|
|
2022-11-16 13:32:16 +08:00
|
|
|
router.patch(
|
2022-11-11 14:29:37 +08:00
|
|
|
`${profileRoute}/password`,
|
|
|
|
koaGuard({
|
|
|
|
body: object({ password: string().regex(passwordRegEx) }),
|
|
|
|
}),
|
|
|
|
async (ctx, next) => {
|
|
|
|
const userId = await checkSessionHealth(ctx, provider, verificationTimeout);
|
|
|
|
|
2022-11-16 14:57:41 +08:00
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
2022-11-11 14:29:37 +08:00
|
|
|
|
|
|
|
const { password } = ctx.guard.body;
|
|
|
|
const { passwordEncrypted: oldPasswordEncrypted } = await findUserById(userId);
|
|
|
|
|
|
|
|
assertThat(
|
|
|
|
!oldPasswordEncrypted || !(await argon2Verify({ password, hash: oldPasswordEncrypted })),
|
|
|
|
new RequestError({ code: 'user.same_password', status: 422 })
|
|
|
|
);
|
|
|
|
|
|
|
|
const { passwordEncrypted, passwordEncryptionMethod } = await encryptUserPassword(password);
|
|
|
|
|
|
|
|
await updateUserById(userId, { passwordEncrypted, passwordEncryptionMethod });
|
|
|
|
|
|
|
|
ctx.status = 204;
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
2022-11-16 13:32:16 +08:00
|
|
|
|
|
|
|
router.patch(
|
|
|
|
`${profileRoute}/email`,
|
|
|
|
koaGuard({
|
|
|
|
body: object({ primaryEmail: string().regex(emailRegEx) }),
|
|
|
|
}),
|
|
|
|
async (ctx, next) => {
|
|
|
|
const userId = await checkSessionHealth(ctx, provider, verificationTimeout);
|
|
|
|
|
2022-11-16 14:57:41 +08:00
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
2022-11-16 13:32:16 +08:00
|
|
|
|
|
|
|
const { primaryEmail } = ctx.guard.body;
|
|
|
|
|
2022-12-05 14:14:21 +08:00
|
|
|
await checkIdentifierCollision({ primaryEmail });
|
2022-11-16 13:32:16 +08:00
|
|
|
await updateUserById(userId, { primaryEmail });
|
|
|
|
|
|
|
|
ctx.status = 204;
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
router.delete(`${profileRoute}/email`, async (ctx, next) => {
|
|
|
|
const userId = await checkSessionHealth(ctx, provider, verificationTimeout);
|
|
|
|
|
2022-11-16 14:57:41 +08:00
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
2022-11-16 13:32:16 +08:00
|
|
|
|
|
|
|
const { primaryEmail } = await findUserById(userId);
|
|
|
|
|
2022-12-07 22:00:58 +08:00
|
|
|
assertThat(primaryEmail, new RequestError({ code: 'user.email_not_exist', status: 422 }));
|
2022-11-16 13:32:16 +08:00
|
|
|
|
|
|
|
await updateUserById(userId, { primaryEmail: null });
|
|
|
|
|
|
|
|
ctx.status = 204;
|
|
|
|
|
|
|
|
return next();
|
|
|
|
});
|
2022-11-16 14:35:38 +08:00
|
|
|
|
|
|
|
router.patch(
|
|
|
|
`${profileRoute}/phone`,
|
|
|
|
koaGuard({
|
|
|
|
body: object({ primaryPhone: string().regex(phoneRegEx) }),
|
|
|
|
}),
|
|
|
|
async (ctx, next) => {
|
|
|
|
const userId = await checkSessionHealth(ctx, provider, verificationTimeout);
|
|
|
|
|
2022-11-16 14:57:41 +08:00
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
2022-11-16 14:35:38 +08:00
|
|
|
|
|
|
|
const { primaryPhone } = ctx.guard.body;
|
|
|
|
|
2022-12-05 14:14:21 +08:00
|
|
|
await checkIdentifierCollision({ primaryPhone });
|
2022-11-16 14:35:38 +08:00
|
|
|
await updateUserById(userId, { primaryPhone });
|
|
|
|
|
|
|
|
ctx.status = 204;
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
router.delete(`${profileRoute}/phone`, async (ctx, next) => {
|
|
|
|
const userId = await checkSessionHealth(ctx, provider, verificationTimeout);
|
|
|
|
|
2022-11-16 14:57:41 +08:00
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
2022-11-16 14:35:38 +08:00
|
|
|
|
|
|
|
const { primaryPhone } = await findUserById(userId);
|
|
|
|
|
2022-12-07 22:00:58 +08:00
|
|
|
assertThat(primaryPhone, new RequestError({ code: 'user.phone_not_exist', status: 422 }));
|
2022-11-16 14:35:38 +08:00
|
|
|
|
|
|
|
await updateUserById(userId, { primaryPhone: null });
|
|
|
|
|
|
|
|
ctx.status = 204;
|
|
|
|
|
|
|
|
return next();
|
|
|
|
});
|
2022-11-18 12:25:31 +08:00
|
|
|
|
|
|
|
router.patch(
|
|
|
|
`${profileRoute}/identities`,
|
|
|
|
koaGuard({
|
|
|
|
body: object({
|
|
|
|
connectorId: string(),
|
|
|
|
data: unknown(),
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
async (ctx, next) => {
|
|
|
|
const userId = await checkSessionHealth(ctx, provider, verificationTimeout);
|
|
|
|
|
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
|
|
|
|
|
|
|
const { connectorId, data } = ctx.guard.body;
|
|
|
|
|
|
|
|
const {
|
|
|
|
metadata: { target },
|
|
|
|
} = await getLogtoConnectorById(connectorId);
|
|
|
|
|
|
|
|
const socialUserInfo = await getUserInfoByAuthCode(connectorId, data);
|
|
|
|
const { identities } = await findUserById(userId);
|
|
|
|
|
|
|
|
await updateUserById(userId, {
|
|
|
|
identities: {
|
|
|
|
...identities,
|
|
|
|
[target]: { userId: socialUserInfo.id, details: socialUserInfo },
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
ctx.status = 204;
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
router.delete(
|
|
|
|
`${profileRoute}/identities/:target`,
|
|
|
|
koaGuard({
|
|
|
|
params: object({ target: string() }),
|
|
|
|
}),
|
|
|
|
async (ctx, next) => {
|
|
|
|
const { accountId: userId } = await provider.Session.get(ctx);
|
|
|
|
|
|
|
|
assertThat(userId, new RequestError({ code: 'auth.unauthorized', status: 401 }));
|
|
|
|
|
|
|
|
const { target } = ctx.guard.params;
|
|
|
|
const { identities } = await findUserById(userId);
|
|
|
|
|
|
|
|
assertThat(
|
|
|
|
has(identities, target),
|
2022-12-07 22:00:58 +08:00
|
|
|
new RequestError({ code: 'user.identity_not_exist', status: 404 })
|
2022-11-18 12:25:31 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
await deleteUserIdentity(userId, target);
|
|
|
|
|
|
|
|
ctx.status = 204;
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
2022-11-11 10:50:20 +08:00
|
|
|
}
|