feat(api): add user search endpoint and normalize response data and validation
This commit is contained in:
parent
91f4c524b9
commit
c7b7d61cec
14 changed files with 574 additions and 368 deletions
|
@ -1,10 +1,17 @@
|
|||
import zod from 'zod';
|
||||
import { prisma } from '@/prisma';
|
||||
import zod from 'zod/v4';
|
||||
|
||||
export const userEmailSchema = zod
|
||||
.string()
|
||||
export const userEmailClientSchema = zod
|
||||
.email('Invalid email address')
|
||||
.min(3, 'Email is required');
|
||||
|
||||
export const userEmailSchema = userEmailClientSchema.refine(async (val) => {
|
||||
const existingUser = await prisma.user.findUnique({
|
||||
where: { email: val },
|
||||
});
|
||||
return !existingUser;
|
||||
}, 'Email in use by another account');
|
||||
|
||||
export const userFirstNameSchema = zod
|
||||
.string()
|
||||
.min(1, 'First name is required')
|
||||
|
@ -15,20 +22,40 @@ export const userLastNameSchema = zod
|
|||
.min(1, 'Last name is required')
|
||||
.max(32, 'Last name must be at most 32 characters long');
|
||||
|
||||
export const userNameSchema = zod
|
||||
export const userNameClientSchema = zod
|
||||
.string()
|
||||
.min(3, 'Username is required')
|
||||
.max(32, 'Username must be at most 32 characters long')
|
||||
.regex(
|
||||
/^[a-zA-Z0-9_]+$/,
|
||||
'Username can only contain letters, numbers, and underscores',
|
||||
);
|
||||
)
|
||||
.refine((val) => !disallowedUsernames.includes(val?.toLowerCase() || ''), {
|
||||
error: 'Username is not allowed',
|
||||
});
|
||||
|
||||
export const loginSchema = zod.object({
|
||||
email: userEmailSchema.or(userNameSchema),
|
||||
export const userNameSchema = userNameClientSchema.refine(async (val) => {
|
||||
const existingUser = await prisma.user.findUnique({
|
||||
where: { name: val },
|
||||
});
|
||||
return !existingUser;
|
||||
}, 'Username in use by another account');
|
||||
|
||||
export const loginClientSchema = zod.object({
|
||||
email: userEmailClientSchema.or(userNameClientSchema),
|
||||
password: zod.string().min(1, 'Password is required'),
|
||||
});
|
||||
|
||||
export const userIdSchema = zod
|
||||
.string()
|
||||
.min(1, 'User ID is required')
|
||||
.refine(async (val) => {
|
||||
const user = await prisma.user.findUnique({
|
||||
where: { id: val },
|
||||
});
|
||||
return !!user;
|
||||
}, 'User not found');
|
||||
|
||||
export const registerSchema = zod
|
||||
.object({
|
||||
firstName: userFirstNameSchema,
|
||||
|
@ -45,7 +72,7 @@ export const registerSchema = zod
|
|||
username: userNameSchema,
|
||||
})
|
||||
.refine((data) => data.password === data.confirmPassword, {
|
||||
message: 'Passwords do not match',
|
||||
error: 'Passwords do not match',
|
||||
path: ['confirmPassword'],
|
||||
})
|
||||
.refine(
|
||||
|
@ -55,7 +82,39 @@ export const registerSchema = zod
|
|||
!data.password.includes(data.email) &&
|
||||
!data.password.includes(data.username),
|
||||
{
|
||||
message:
|
||||
error:
|
||||
'Password cannot contain your first name, last name, email, or username',
|
||||
path: ['password'],
|
||||
},
|
||||
);
|
||||
|
||||
export const registerClientSchema = zod
|
||||
.object({
|
||||
firstName: userFirstNameSchema,
|
||||
lastName: userLastNameSchema,
|
||||
email: userEmailClientSchema,
|
||||
password: zod
|
||||
.string()
|
||||
.min(8, 'Password must be at least 8 characters long')
|
||||
.max(128, 'Password must be at most 128 characters long'),
|
||||
confirmPassword: zod
|
||||
.string()
|
||||
.min(8, 'Password must be at least 8 characters long')
|
||||
.max(128, 'Password must be at most 128 characters long'),
|
||||
username: userNameClientSchema,
|
||||
})
|
||||
.refine((data) => data.password === data.confirmPassword, {
|
||||
error: 'Passwords do not match',
|
||||
path: ['confirmPassword'],
|
||||
})
|
||||
.refine(
|
||||
(data) =>
|
||||
!data.password.includes(data.firstName) &&
|
||||
!data.password.includes(data.lastName) &&
|
||||
!data.password.includes(data.email) &&
|
||||
!data.password.includes(data.username),
|
||||
{
|
||||
error:
|
||||
'Password cannot contain your first name, last name, email, or username',
|
||||
path: ['password'],
|
||||
},
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue