diff --git a/migration/1723583534955-AddUserEmailVerificationFields.ts b/migration/1723583534955-AddUserEmailVerificationFields.ts new file mode 100644 index 000000000..07921e89e --- /dev/null +++ b/migration/1723583534955-AddUserEmailVerificationFields.ts @@ -0,0 +1,29 @@ +import { MigrationInterface, QueryRunner } from 'typeorm'; + +export class AddUserEmailVerificationFields1723583534955 + implements MigrationInterface +{ + public async up(queryRunner: QueryRunner): Promise { + await queryRunner.query(` + ALTER TABLE "user" + ADD "emailConfirmationToken" character varying, + ADD "emailConfirmationTokenExpiredAt" TIMESTAMP, + ADD "emailConfirmed" boolean DEFAULT false, + ADD "emailConfirmationSent" boolean DEFAULT false, + ADD "emailConfirmationSentAt" TIMESTAMP, + ADD "emailConfirmedAt" TIMESTAMP; + `); + } + + public async down(queryRunner: QueryRunner): Promise { + await queryRunner.query(` + ALTER TABLE "user" + DROP COLUMN "emailConfirmationToken", + DROP COLUMN "emailConfirmationTokenExpiredAt", + DROP COLUMN "emailConfirmed", + DROP COLUMN "emailConfirmationSent", + DROP COLUMN "emailConfirmationSentAt", + DROP COLUMN "emailConfirmedAt"; + `); + } +} diff --git a/src/adapters/notifications/MockNotificationAdapter.ts b/src/adapters/notifications/MockNotificationAdapter.ts index 870f37275..97008f20a 100644 --- a/src/adapters/notifications/MockNotificationAdapter.ts +++ b/src/adapters/notifications/MockNotificationAdapter.ts @@ -34,6 +34,15 @@ export class MockNotificationAdapter implements NotificationAdapterInterface { return Promise.resolve(undefined); } + async sendUserEmailConfirmation(params: { + email: string; + user: User; + token: string; + }) { + logger.debug('MockNotificationAdapter sendUserEmailConfirmation', params); + return Promise.resolve(undefined); + } + userSuperTokensCritical(): Promise { return Promise.resolve(undefined); } diff --git a/src/adapters/notifications/NotificationAdapterInterface.ts b/src/adapters/notifications/NotificationAdapterInterface.ts index 1e02155e9..827b168c4 100644 --- a/src/adapters/notifications/NotificationAdapterInterface.ts +++ b/src/adapters/notifications/NotificationAdapterInterface.ts @@ -62,6 +62,12 @@ export interface NotificationAdapterInterface { token: string; }): Promise; + sendUserEmailConfirmation(params: { + email: string; + user: User; + token: string; + }): Promise; + userSuperTokensCritical(params: { user: User; eventName: UserStreamBalanceWarning; diff --git a/src/adapters/notifications/NotificationCenterAdapter.ts b/src/adapters/notifications/NotificationCenterAdapter.ts index 039f25525..e0200e8a4 100644 --- a/src/adapters/notifications/NotificationCenterAdapter.ts +++ b/src/adapters/notifications/NotificationCenterAdapter.ts @@ -90,6 +90,29 @@ export class NotificationCenterAdapter implements NotificationAdapterInterface { } } + // todo: use different eventName specific to Qacc (to show correct icon and description) + // todo: add the new eventName to the notification service and add the schema to Ortto + async sendUserEmailConfirmation(params: { + email: string; + user: User; + token: string; + }): Promise { + const { email, user, token } = params; + try { + await callSendNotification({ + eventName: NOTIFICATIONS_EVENT_NAMES.SEND_EMAIL_CONFIRMATION, + segment: { + payload: { + email, + verificationLink: `${dappUrl}/verification/user/${user.walletAddress}/${token}`, + }, + }, + }); + } catch (e) { + logger.error('sendUserEmailConfirmation >> error', e); + } + } + async userSuperTokensCritical(params: { user: User; eventName: UserStreamBalanceWarning; diff --git a/src/entities/user.ts b/src/entities/user.ts index 08e45daa0..6fae47a1d 100644 --- a/src/entities/user.ts +++ b/src/entities/user.ts @@ -188,6 +188,30 @@ export class User extends BaseEntity { @Field(_type => Float, { nullable: true }) activeQFMBDScore?: number; + @Field(_type => Boolean, { nullable: false }) + @Column({ default: false }) + emailConfirmed: boolean; + + @Field(_type => String, { nullable: true }) + @Column('text', { nullable: true }) + emailConfirmationToken: string | null; + + @Field(_type => Date, { nullable: true }) + @Column('timestamptz', { nullable: true }) + emailConfirmationTokenExpiredAt: Date | null; + + @Field(_type => Boolean, { nullable: true }) + @Column({ default: false }) + emailConfirmationSent: boolean; + + @Field(_type => Date, { nullable: true }) + @Column({ type: 'timestamptz', nullable: true }) + emailConfirmationSentAt: Date | null; + + @Field(_type => Date, { nullable: true }) + @Column({ type: 'timestamptz', nullable: true }) + emailConfirmedAt: Date | null; + @Field(_type => Int, { nullable: true }) async donationsCount() { return await Donation.createQueryBuilder('donation') diff --git a/src/repositories/userRepository.test.ts b/src/repositories/userRepository.test.ts index 84336a6d5..4b482d91b 100644 --- a/src/repositories/userRepository.test.ts +++ b/src/repositories/userRepository.test.ts @@ -11,11 +11,14 @@ import { User, UserRole } from '../entities/user'; import { findAdminUserByEmail, findAllUsers, + findUserByEmailConfirmationToken, findUserById, findUserByWalletAddress, findUsersWhoDonatedToProjectExcludeWhoLiked, findUsersWhoLikedProjectExcludeProjectOwner, findUsersWhoSupportProject, + updateUserEmailConfirmationStatus, + updateUserEmailConfirmationToken, } from './userRepository'; import { Reaction } from '../entities/reaction'; @@ -44,6 +47,19 @@ describe( findUsersWhoDonatedToProjectTestCases, ); +describe( + 'userRepository.findUserByEmailConfirmationToken', + findUserByEmailConfirmationTokenTestCases, +); +describe( + 'userRepository.updateUserEmailConfirmationStatus', + updateUserEmailConfirmationStatusTestCases, +); +describe( + 'userRepository.updateUserEmailConfirmationToken', + updateUserEmailConfirmationTokenTestCases, +); + function findUsersWhoDonatedToProjectTestCases() { it('should find wallet addresses of who donated to a project, exclude who liked', async () => { const project = await saveProjectDirectlyToDb(createProjectData()); @@ -489,3 +505,107 @@ function findUsersWhoSupportProjectTestCases() { ); }); } + +function findUserByEmailConfirmationTokenTestCases() { + it('should return a user if a valid email confirmation token is provided', async () => { + await User.create({ + email: 'test@example.com', + emailConfirmationToken: 'validToken123', + loginType: 'wallet', + }).save(); + + const foundUser = await findUserByEmailConfirmationToken('validToken123'); + assert.isNotNull(foundUser); + assert.equal(foundUser!.email, 'test@example.com'); + assert.equal(foundUser!.emailConfirmationToken, 'validToken123'); + }); + + it('should return null if no user is found with the provided email confirmation token', async () => { + const foundUser = await findUserByEmailConfirmationToken('invalidToken123'); + assert.isNull(foundUser); + }); +} + +function updateUserEmailConfirmationStatusTestCases() { + it('should update the email confirmation status of a user', async () => { + const user = await User.create({ + email: 'test@example.com', + emailConfirmed: false, + emailConfirmationToken: 'validToken123', + loginType: 'wallet', + }).save(); + + await updateUserEmailConfirmationStatus({ + userId: user.id, + emailConfirmed: true, + emailConfirmationTokenExpiredAt: null, + emailConfirmationToken: null, + emailConfirmationSentAt: null, + }); + + // Using findOne with options object + const updatedUser = await User.findOne({ where: { id: user.id } }); + assert.isNotNull(updatedUser); + assert.isTrue(updatedUser!.emailConfirmed); + assert.isNull(updatedUser!.emailConfirmationToken); + }); + + it('should not update any user if the userId does not exist', async () => { + const result = await updateUserEmailConfirmationStatus({ + userId: 999, // non-existent userId + emailConfirmed: true, + emailConfirmationTokenExpiredAt: null, + emailConfirmationToken: null, + emailConfirmationSentAt: null, + }); + + assert.equal(result.affected, 0); // No rows should be affected + }); +} + +function updateUserEmailConfirmationTokenTestCases() { + it('should update the email confirmation token and expiry date for a user', async () => { + const user = await User.create({ + email: 'test@example.com', + loginType: 'wallet', + }).save(); + + const newToken = 'newToken123'; + const newExpiryDate = new Date(Date.now() + 3600 * 1000); // 1 hour from now + const sentAtDate = new Date(); + + await updateUserEmailConfirmationToken({ + userId: user.id, + emailConfirmationToken: newToken, + emailConfirmationTokenExpiredAt: newExpiryDate, + emailConfirmationSentAt: sentAtDate, + }); + + // Using findOne with options object + const updatedUser = await User.findOne({ where: { id: user.id } }); + assert.isNotNull(updatedUser); + assert.equal(updatedUser!.emailConfirmationToken, newToken); + assert.equal( + updatedUser!.emailConfirmationTokenExpiredAt!.getTime(), + newExpiryDate.getTime(), + ); + assert.equal( + updatedUser!.emailConfirmationSentAt!.getTime(), + sentAtDate.getTime(), + ); + }); + + it('should throw an error if the userId does not exist', async () => { + try { + await updateUserEmailConfirmationToken({ + userId: 999, // non-existent userId + emailConfirmationToken: 'newToken123', + emailConfirmationTokenExpiredAt: new Date(), + emailConfirmationSentAt: new Date(), + }); + assert.fail('Expected an error to be thrown'); + } catch (error) { + assert.equal(error.message, 'User not found'); + } + }); +} diff --git a/src/repositories/userRepository.ts b/src/repositories/userRepository.ts index a9fb86155..985926596 100644 --- a/src/repositories/userRepository.ts +++ b/src/repositories/userRepository.ts @@ -1,3 +1,4 @@ +import { UpdateResult } from 'typeorm'; import { publicSelectionFields, User, UserRole } from '../entities/user'; import { Donation } from '../entities/donation'; import { Reaction } from '../entities/reaction'; @@ -177,3 +178,67 @@ export const findUsersWhoSupportProject = async ( } return users; }; + +export const findUserByEmailConfirmationToken = async ( + emailConfirmationToken: string, +): Promise => { + return User.createQueryBuilder('user') + .where({ + emailConfirmationToken, + }) + .getOne(); +}; + +export const updateUserEmailConfirmationStatus = async (params: { + userId: number; + emailConfirmed: boolean; + emailConfirmationTokenExpiredAt: Date | null; + emailConfirmationToken: string | null; + emailConfirmationSentAt: Date | null; +}): Promise => { + const { + userId, + emailConfirmed, + emailConfirmationTokenExpiredAt, + emailConfirmationToken, + emailConfirmationSentAt, + } = params; + + return User.createQueryBuilder() + .update(User) + .set({ + emailConfirmed, + emailConfirmationTokenExpiredAt, + emailConfirmationToken, + emailConfirmationSentAt, + }) + .where('id = :userId', { userId }) + .execute(); +}; + +export const updateUserEmailConfirmationToken = async (params: { + userId: number; + emailConfirmationToken: string; + emailConfirmationTokenExpiredAt: Date; + emailConfirmationSentAt: Date; +}): Promise => { + const { + userId, + emailConfirmationToken, + emailConfirmationTokenExpiredAt, + emailConfirmationSentAt, + } = params; + + const user = await findUserById(userId); + if (!user) { + throw new Error('User not found'); + } + + user.emailConfirmationToken = emailConfirmationToken; + user.emailConfirmationTokenExpiredAt = emailConfirmationTokenExpiredAt; + user.emailConfirmationSentAt = emailConfirmationSentAt; + user.emailConfirmed = false; + + await user.save(); + return user; +}; diff --git a/src/resolvers/userResolver.test.ts b/src/resolvers/userResolver.test.ts index 10ca74ea6..38772227a 100644 --- a/src/resolvers/userResolver.test.ts +++ b/src/resolvers/userResolver.test.ts @@ -6,6 +6,7 @@ import { User } from '../entities/user'; import { createDonationData, createProjectData, + generateConfirmationEmailToken, generateRandomEtheriumAddress, generateTestAccessToken, graphqlUrl, @@ -18,15 +19,27 @@ import { refreshUserScores, updateUser, userByAddress, + userVerificationConfirmEmail, + userVerificationSendEmailConfirmation, } from '../../test/graphqlQueries'; import { errorMessages } from '../utils/errorMessages'; import { DONATION_STATUS } from '../entities/donation'; import { getGitcoinAdapter } from '../adapters/adaptersFactory'; import { updateUserTotalDonated } from '../services/userService'; +import { findUserById } from '../repositories/userRepository'; +import { sleep } from '../utils/utils'; describe('updateUser() test cases', updateUserTestCases); describe('userByAddress() test cases', userByAddressTestCases); describe('refreshUserScores() test cases', refreshUserScoresTestCases); +describe( + 'userVerificationSendEmailConfirmation() test cases', + userVerificationSendEmailConfirmationTestCases, +); +describe( + 'userVerificationConfirmEmail() test cases', + userVerificationConfirmEmailTestCases, +); // TODO I think we can delete addUserVerification query // describe('addUserVerification() test cases', addUserVerificationTestCases); function refreshUserScoresTestCases() { @@ -658,3 +671,163 @@ function updateUserTestCases() { assert.equal(updatedUser?.url, updateUserData.url); }); } + +function userVerificationSendEmailConfirmationTestCases() { + it('should send email confirmation for user email verification', async () => { + const user = await saveUserDirectlyToDb(generateRandomEtheriumAddress()); + user.email = 'test@example.com'; + user.emailConfirmed = false; + await user.save(); + + const accessToken = await generateTestAccessToken(user.id); + const result = await axios.post( + graphqlUrl, + { + query: userVerificationSendEmailConfirmation, + variables: { + userId: user.id, + }, + }, + { + headers: { + Authorization: `Bearer ${accessToken}`, + }, + }, + ); + + assert.isOk(result.data.data.userVerificationSendEmailConfirmation); + assert.isFalse( + result.data.data.userVerificationSendEmailConfirmation.emailConfirmed, + ); + assert.equal( + result.data.data.userVerificationSendEmailConfirmation.email, + 'test@example.com', + ); + assert.equal( + result.data.data.userVerificationSendEmailConfirmation + .emailConfirmationSent, + true, + ); + assert.isNotNull( + result.data.data.userVerificationSendEmailConfirmation + .emailConfirmationToken, + ); + }); + + it('should throw error when sending email confirmation if email is already confirmed', async () => { + const user = await saveUserDirectlyToDb(generateRandomEtheriumAddress()); + user.email = 'test@example.com'; + user.emailConfirmed = true; + await user.save(); + + const accessToken = await generateTestAccessToken(user.id); + const result = await axios.post( + graphqlUrl, + { + query: userVerificationSendEmailConfirmation, + variables: { + userId: user.id, + }, + }, + { + headers: { + Authorization: `Bearer ${accessToken}`, + }, + }, + ); + + assert.equal( + result.data.errors[0].message, + errorMessages.YOU_ALREADY_VERIFIED_THIS_EMAIL, + ); + }); +} + +function userVerificationConfirmEmailTestCases() { + it('should confirm user email verification', async () => { + const user = await saveUserDirectlyToDb(generateRandomEtheriumAddress()); + user.email = 'test@example.com'; + user.emailConfirmed = false; + await user.save(); + + const accessToken = await generateTestAccessToken(user.id); + const emailConfirmationSentResult = await axios.post( + graphqlUrl, + { + query: userVerificationSendEmailConfirmation, + variables: { + userId: user.id, + }, + }, + { + headers: { + Authorization: `Bearer ${accessToken}`, + }, + }, + ); + + const token = + emailConfirmationSentResult.data.data + .userVerificationSendEmailConfirmation.emailConfirmationToken; + + const result = await axios.post( + graphqlUrl, + { + query: userVerificationConfirmEmail, + variables: { + emailConfirmationToken: token, + }, + }, + { + headers: { + Authorization: `Bearer ${accessToken}`, + }, + }, + ); + + assert.isOk(result.data.data.userVerificationConfirmEmail); + assert.equal( + result.data.data.userVerificationConfirmEmail.emailConfirmed, + true, + ); + assert.isNotNull( + result.data.data.userVerificationConfirmEmail.emailConfirmedAt, + ); + }); + + it('should throw error when confirm email token is invalid or expired for user email verification', async () => { + const user = await saveUserDirectlyToDb(generateRandomEtheriumAddress()); + user.email = 'test@example.com'; + user.emailConfirmed = false; + await user.save(); + + const accessToken = await generateTestAccessToken(user.id); + const token = await generateConfirmationEmailToken(user.id); + user.emailConfirmationToken = token; + await user.save(); + await sleep(500); // Simulating token expiration or invalidity + + const result = await axios.post( + graphqlUrl, + { + query: userVerificationConfirmEmail, + variables: { + emailConfirmationToken: token, + }, + }, + { + headers: { + Authorization: `Bearer ${accessToken}`, + }, + }, + ); + + assert.equal(result.data.errors[0].message, 'jwt expired'); + const userReinitializedEmailParams = await findUserById(user.id); + + assert.isFalse(userReinitializedEmailParams!.emailConfirmed); + assert.isFalse(userReinitializedEmailParams!.emailConfirmationSent); + assert.isNotOk(userReinitializedEmailParams!.emailConfirmationSentAt); + assert.isNull(userReinitializedEmailParams!.emailConfirmationToken); + }); +} diff --git a/src/resolvers/userResolver.ts b/src/resolvers/userResolver.ts index 4a1861cfb..a6150ab55 100644 --- a/src/resolvers/userResolver.ts +++ b/src/resolvers/userResolver.ts @@ -8,13 +8,17 @@ import { Resolver, } from 'type-graphql'; import { Repository } from 'typeorm'; +import * as jwt from 'jsonwebtoken'; +import moment from 'moment'; import { User } from '../entities/user'; +import config from '../config'; import { AccountVerificationInput } from './types/accountVerificationInput'; import { ApolloContext } from '../types/ApolloContext'; import { i18n, translationErrorMessagesKeys } from '../utils/errorMessages'; import { validateEmail } from '../utils/validators/commonValidators'; import { + findUserByEmailConfirmationToken, findUserById, findUserByWalletAddress, } from '../repositories/userRepository'; @@ -176,6 +180,14 @@ export class UserResolver { if (!validateEmail(email)) { throw new Error(i18n.__(translationErrorMessagesKeys.INVALID_EMAIL)); } + if (dbUser.email !== email) { + dbUser.emailConfirmed = false; + dbUser.emailConfirmationSent = false; + dbUser.emailConfirmationToken = null; + dbUser.emailConfirmationTokenExpiredAt = null; + dbUser.emailConfirmationSentAt = null; + dbUser.emailConfirmedAt = null; + } dbUser.email = email; } if (url !== undefined) { @@ -231,4 +243,112 @@ export class UserResolver { return true; } + + @Mutation(_returns => User) + async userVerificationSendEmailConfirmation( + @Arg('userId') userId: number, + @Ctx() { req: { user } }: ApolloContext, + ): Promise { + try { + const currentUserId = user?.userId; + if (!currentUserId || currentUserId != userId) { + throw new Error(i18n.__(translationErrorMessagesKeys.UN_AUTHORIZED)); + } + + const userToVerify = await findUserById(userId); + + if (!userToVerify) { + throw new Error(i18n.__(translationErrorMessagesKeys.USER_NOT_FOUND)); + } + + const email = userToVerify.email; + if (!email) { + throw new Error( + i18n.__(translationErrorMessagesKeys.NO_EMAIL_PROVIDED), + ); + } + if (userToVerify.emailConfirmed) { + throw new Error( + i18n.__(translationErrorMessagesKeys.YOU_ALREADY_VERIFIED_THIS_EMAIL), + ); + } + + const token = jwt.sign( + { userId }, + config.get('MAILER_JWT_SECRET') as string, + { expiresIn: '5m' }, + ); + + userToVerify.emailConfirmationTokenExpiredAt = moment() + .add(5, 'minutes') + .toDate(); + userToVerify.emailConfirmationToken = token; + userToVerify.emailConfirmationSent = true; + userToVerify.emailConfirmed = false; + userToVerify.emailConfirmationSentAt = new Date(); + await userToVerify.save(); + + await getNotificationAdapter().sendUserEmailConfirmation({ + email, + user: userToVerify, + token, + }); + + return userToVerify; + } catch (e) { + logger.error('userVerificationSendEmailConfirmation() error', e); + throw e; + } + } + + @Mutation(_returns => User) + async userVerificationConfirmEmail( + @Arg('emailConfirmationToken') emailConfirmationToken: string, + ): Promise { + try { + const secret = config.get('MAILER_JWT_SECRET') as string; + + const isValidToken = await findUserByEmailConfirmationToken( + emailConfirmationToken, + ); + + if (!isValidToken) { + throw new Error(i18n.__(translationErrorMessagesKeys.USER_NOT_FOUND)); + } + + const decodedJwt: any = jwt.verify(emailConfirmationToken, secret); + const userId = decodedJwt.userId; + const user = await findUserById(userId); + + if (!user) { + throw new Error(i18n.__(translationErrorMessagesKeys.USER_NOT_FOUND)); + } + + user.emailConfirmationTokenExpiredAt = null; + user.emailConfirmationToken = null; + user.emailConfirmedAt = new Date(); + user.emailConfirmed = true; + await user.save(); + + return user; + } catch (e) { + const user = await findUserByEmailConfirmationToken( + emailConfirmationToken, + ); + + if (!user) { + throw new Error(i18n.__(translationErrorMessagesKeys.USER_NOT_FOUND)); + } + + user.emailConfirmed = false; + user.emailConfirmationTokenExpiredAt = null; + user.emailConfirmationSent = false; + user.emailConfirmationSentAt = null; + user.emailConfirmationToken = null; + + await user.save(); + logger.error('userVerificationConfirmEmail() error', e); + throw e; + } + } } diff --git a/src/utils/errorMessages.ts b/src/utils/errorMessages.ts index b3e524b36..a8f8dfa13 100644 --- a/src/utils/errorMessages.ts +++ b/src/utils/errorMessages.ts @@ -319,4 +319,5 @@ export const translationErrorMessagesKeys = { DRAFT_DONATION_DISABLED: 'DRAFT_DONATION_DISABLED', EVM_SUPPORT_ONLY: 'EVM_SUPPORT_ONLY', ABC_NOT_FOUND: 'ABC_NOT_FOUND', + NO_EMAIL_PROVIDED: 'NO_EMAIL_PROVIDED', }; diff --git a/src/utils/locales/en.json b/src/utils/locales/en.json index 7739b2952..ca706cc5f 100644 --- a/src/utils/locales/en.json +++ b/src/utils/locales/en.json @@ -101,5 +101,6 @@ "EVM_SUPPORT_ONLY": "Only EVM support", "INVALID_PROJECT_ID": "INVALID_PROJECT_ID", "TX_NOT_FOUND": "TX_NOT_FOUND", - "ABC_NOT_FOUND": "Abc not found" + "ABC_NOT_FOUND": "Abc not found", + "NO_EMAIL_PROVIDED": "No email address provided." } diff --git a/src/utils/locales/es.json b/src/utils/locales/es.json index e7d051c5d..468f99d42 100644 --- a/src/utils/locales/es.json +++ b/src/utils/locales/es.json @@ -98,5 +98,6 @@ "PROJECT_UPDATE_CONTENT_LENGTH_SIZE_EXCEEDED": "El contenido es demasiado largo", "DRAFT_DONATION_DISABLED": "El borrador de donación está deshabilitado", "EVM_SUPPORT_ONLY": "Solo se admite EVM", - "ABC_NOT_FOUND": "ABC no encontrado" + "ABC_NOT_FOUND": "ABC no encontrado", + "NO_EMAIL_PROVIDED": "No se ha proporcionado una dirección de correo electrónico." } diff --git a/test/graphqlQueries.ts b/test/graphqlQueries.ts index 9ed8aaea5..de77ba21f 100644 --- a/test/graphqlQueries.ts +++ b/test/graphqlQueries.ts @@ -1989,3 +1989,33 @@ export const fetchDonationMetricsQuery = ` } } `; + +export const userVerificationSendEmailConfirmation = ` + mutation userVerificationSendEmailConfirmation($userId: Float!){ + userVerificationSendEmailConfirmation(userId: $userId) { + id + email + emailConfirmed + emailConfirmationToken + emailConfirmationTokenExpiredAt + emailConfirmationSent + emailConfirmationSentAt + emailConfirmedAt + } + } +`; + +export const userVerificationConfirmEmail = ` + mutation userVerificationConfirmEmail($emailConfirmationToken: String!){ + userVerificationConfirmEmail(emailConfirmationToken: $emailConfirmationToken) { + id + email + emailConfirmed + emailConfirmationToken + emailConfirmationTokenExpiredAt + emailConfirmationSent + emailConfirmationSentAt + emailConfirmedAt + } + } +`;