Auth
RelayBox Auth handles user authentication, session management, and multi-factor authentication (MFA). It provides methods for signing up, logging in, refreshing tokens, managing user sessions, and OAuth provider login. Auth provides a higher level of securty and interaction capability for your application.
Getting started
To initialize auth, simply pass a public key to the RelayBox constructor. A public key consists of the first 2 parts of any API key.
For example, if your API key is...
12345.abcde.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
Your public key will be...
12345.abcde
The third part of the API key is the secret key used to sign authentication tokens and should never be exposed to end users. To generate an API key, head over to the dashboard and locate the "API keys" section under any of your applications.
const relayBox = new RelayBox({
publicKey: xxxx.xxxx // Replace with your public key
});
Once initialized you can now access auth functionality within your application via the auth namespace. For example...
const authUserSession = await relayBox.auth.getSession();
Auth provides the means to create and manage every aspect of the realtime user journey through your applications. Users created using the auth service have full access to realtime services once system authentication is complete.
The AuthUserSession Object
After successful authentication, the authentication service returns an object containing data about the logged-in user, along with the credentials required to access realtime services.
The authentication tokens within the session object can also be used as a standalone authentication mechanism for your own backend services through the use of cryptographically signed JWT tokens. See the @relaybox/rest documentation for more information.
An AuthUserSession object returned by the authentication service looks like this...
{
"session": {
"token": "xxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"refreshToken": "xxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"expiresIn": 300,
"expiresAt": 1726174042498,
"destroyAt": 1726177342498,
"authStorageType": "persist"
},
"user": {
"id": "af8f910e-c72c-4335-a104-48212ad65c1f",
"orgId": "8f6a2232-63e2-4ff7-969f-417739d155ed",
"appId": "675e81c4-2b88-4167-a33b-0d564d2548a5",
"username": "ShinyKrill709",
"clientId": "0fyWl1VWj8Nx",
"email": "hello@universe.com",
"createdAt": "2024-09-10T18:14:48.758Z",
"updatedAt": "2024-09-12T11:22:30.194Z",
"verifiedAt": "2024-09-10T18:19:45.952Z",
"authMfaEnabled": true,
"blockedAt": null,
"identities": [
{
"id": "654e6295-6927-4ce2-98b1-dca79a868f05",
"provider": "github",
"providerId": "123456",
"verifiedAt": "2024-09-10T18:14:48.908"
},
{
"id": "fc7c67c7-315d-4bfd-8ae3-8e58bc36d4ae",
"provider": "google",
"providerId": "123456",
"verifiedAt": "2024-09-10T20:25:42.719"
},
{
"id": "55734265-ba2a-4cf8-b41b-3cd63d097a04",
"provider": "email",
"providerId": null,
"verifiedAt": "2024-09-10T18:19:45.952"
}
],
"factors": [
{
"id": "0aa7af26-3313-40ae-9857-24ae66ea939e",
"type": "totp",
"verifiedAt": "2024-09-10T18:21:07.231"
}
]
}
}
Error handling
Auth operations are asynchronous and may result in errors being thrown. All errors follow a standard format and can be handled using try/catch or Promise.catch() syntax.
Handled errors will follow a standard format and contain the following data...
{
name: string, // error name
status: number, // HTTP status code
message: string, // human readable error message
data: any // data relevant to the failure
}
A real-world example might look something like the following...
try {
const session = await relayBox.auth.signIn({
email: 'unregistered@relaybox.net',
password: 'my-super-secure-but-incorrect-password'
});
} catch (err) {
console.log(err.name); // UnauthorizedError
console.log(err.status); // 401
console.log(err.message); // Login failed
console.log(err.data); // { message: 'Login failed' }
}
By using named errors and status codes, we hope this will assist with handling errors and presenting a clean UI to your userbase when something inevitably goes wrong.
auth.onAuthEvent()
Authentication events are emmited from all auth service actions.
relayBox.auth.onAuthEvent('SIGN_IN', (session: any) => {
console.log(session)
});
Event | Description |
---|---|
SIGN_UP | Emitted following successful user registration. |
VERIFY | Emitted following successful account verification. |
SIGN_IN | Emitted following successful sign in (including MFA verify). |
SIGN_OUT | Emitted following user sign out. |
GET_SESSION | Emitted after sucessful user session request. |
RESEND_VERIFICATION | Emitted after regitration verification code has been resent. |
MFA_ENROLL | Emitted following successful MFA enrollment. |
MFA_CHALLENGE | Emitted after a sucessful MFA challenge. |
MFA_REQUIRED | Emitted when MFA verification is requred following authentication. |
TOKEN_REFRESH | Emitted following successful token refresh request. |
SESSION_EXPIRED | Emitted following session expiry. The event will be emitted when an auth token is being refreshed automatically due to expiry or during a manual tokenRefresh operation. |
PASSWORD_RESET | Emitted following initiation of password reset request. |
PASSWORD_CONFIRM | Emitted following succesful password reset confirmation. |
auth.signUp()
Asynchronous function to register a new user account.
const response = await relayBox.auth.signUp({
email: 'hello@universe.com',
password: 'my-super-secure-password',
username: 'ShinyKrill709',
firstName: 'Fringe',
lastName: 'Fudge'
});
Options | Description | Type |
---|---|---|
Valid email address: The email address to be associated with the account. A verification code will be sent to this email address to confirm ownership of the account. | string | |
password | Account password: The password to be associated with the account. | string |
username (optional) | Account username: The username to be associated with the account. If not provided, a unique username will be automatically generated. | string |
firstName (optional) | User first name: The first name of the user to be associated with the account. | string |
lastName (optional) | Users last name: The last name of the user to be associated with the account. | string |
Responses
200 OK Object containing the new user id and client id.
{
"message": "Registration successful",
"id": "12345678-1234-1234-1234-123456789012",
"clientId": "_Tv-388ztXmr"
}
400 Bad Request SchemaValidationError
Request schema validation errors follow a distinct structure across all auth endpoints providing feedback on the fields that failed validation.
{
"name": "SchemaValidationError",
"message": "Schema validation failed",
"data": [
{
"code": "too_small",
"minimum": 5,
"type": "string",
"inclusive": true,
"exact": false,
"message": "String must contain at least 5 character(s)",
"path": [
"password"
]
}
]
}
401 Unauthorized AuthenticationError
To mitigate the opportunity for enumeration attacks, a standard 401 Unauthorized error will be returned if the email address or username conflicts with an existing account.
{
"name": "AuthenticationError",
"message": "Registration failed",
}
400 Bad Request PasswordRegexError
Password regex errors are returned when a supplied password fails to meet criteria set via the application dashboard.
{
"name": "PasswordRegexError",
"message": "Password does not match required pattern"
}
Example:
The following example demonstrates a user sign up request containing both optional and required fields.
const response = await relayBox.auth.signUp({
email: 'hello@universe.com',
password: 'my-super-secure-password',
username: 'ShinyKrill709',
firstName: 'Fringe',
lastName: 'Fudge'
});
auth.signIn()
Asynchronous function to authenticate a user using email and passsword.
const session = await relayBox.auth.signIn({
email: 'hello@universe.com',
password: 'my-super-secure-password'
});
Options | Description | Type |
---|---|---|
Valid email address: The verified email address associated the user account. | string | |
password | User password: The password associated with the account. | string |
Responses
200 OK AuthUserSession
{
"session": {
"token": "xxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"refreshToken": "xxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"expiresIn": 300,
"expiresAt": 1726174042498,
"destroyAt": 1726177342498,
"authStorageType": "persist"
},
"user": {
"id": "12345678-1234-1234-1234-123456789012",
"orgId": "12345678-1234-1234-1234-123456789012",
"appId": "12345678-1234-1234-1234-123456789012",
"username": "ShinyKrill709",
"clientId": "0fyWl1VWj8Nx",
"email": "hello@universe.com",
"createdAt": "2024-09-10T18:14:48.758Z",
"updatedAt": "2024-09-12T11:22:30.194Z",
"verifiedAt": "2024-09-10T18:19:45.952Z",
"authMfaEnabled": true,
"blockedAt": null,
"identities": [
{
"id": "12345678-1234-1234-1234-123456789012",
"provider": "github",
"providerId": "123456",
"verifiedAt": "2024-09-10T18:14:48.908"
},
{
"id": "12345678-1234-1234-1234-123456789012",
"provider": "google",
"providerId": "123456",
"verifiedAt": "2024-09-10T20:25:42.719"
},
{
"id": "12345678-1234-1234-1234-123456789012",
"provider": "email",
"providerId": null,
"verifiedAt": "2024-09-10T18:19:45.952"
}
],
"factors": [
{
"id": "12345678-1234-1234-1234-123456789012",
"type": "totp",
"verifiedAt": "2024-09-10T18:21:07.231"
}
]
}
}
200 OK Restricted AuthUserSession
When a user has enrolled in multi factor authentication, the user sign in response object will contain the factors associated with the account along with some basic user information.
{
"user": {
"username": "ShinyKrill709",
"authMfaEnabled": true,
"factors": [
{
"id": "12345678-1234-1234-1234-123456789012",
"type": "totp",
"verifiedAt": "2024-09-10T18:21:07.231"
}
]
},
"session": null,
"tmpToken": "xxxxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
}
401 Unauthorized UnauthorizedError
To mitigate the opportunity for enumeration attacks, a standard 401 Unauthorized error will be returned if the sign in request fails due to account information validation.
{
"name": "UnauthorizedError",
"message": "Login failed"
}
Example:
const session = await relayBox.auth.signIn({
email: 'hello@universe.com',
password: 'my-super-secure-password'
});
auth.verify()
Asynchronous function to verify ownership of an email address using the code sent during account registration.
const response = await relayBox.auth.verify({
email: 'hello@universe.com',
code: 'xxxxxx'
});
Options | Description | Type |
---|---|---|
Valid email address: The email address used to create the user account. | string | |
code | Six digit verification code: The verification code sent to the user's email address following registration. | string |
Responses
200 OK
{
"message": "User verification successful",
}
400 Bad Request SchemaValidationError
Request schema validation errors follow a distinct structure across all auth endpoints providing feedback on the fields that failed validation.
{
"name": "SchemaValidationError",
"message": "Schema validation failed",
"data": [
{
"code": "too_small",
"minimum": 6,
"type": "string",
"inclusive": true,
"exact": true,
"message": "String must contain exactly 6 character(s)",
"path": [
"code"
]
}
]
}
401 Unauthorized UnauthorizedError
To mitigate the opportunity for enumeration attacks, a standard 401 Unauthorized error will be returned if code verification fails. More specific information about the error can be found in the dashboard user activity logs.
{
"name": "AuthenticationError",
"message": "User verification failed",
}
Example:
const session = await relayBox.auth.verify({
email: 'hello@universe.com',
code: 'xxxxxx'
});
auth.passwordReset()
Asynchronous function to initiate a password reset request. A verification code will be sent to the email address provided if previously registered with the application.
const result = await relayBox.auth.passwordReset({
email: 'hello@universe.com',
});
Options | Description | Type |
---|---|---|
Valid email address: The email address to send the password reset request to. | string |
Responses
200 OK
To mitigate the opportunity for enumeration attacks, the response will always return a success message.
{
message: 'Password reset request initialized'
}
Example:
const result = await relayBox.auth.passwordReset({
email: 'hello@universe.com',
});
auth.passwordConfirm()
Asynchronous function to confirm a verification code sent via email following a password reset request.
const response = await relayBox.auth.passwordConfirm({
email: 'hello@universe.com',
password: 'my-new-super-secure-password',
code: 'xxxxxx'
});
Options | Description | Type |
---|---|---|
Valid email address: The email address used to initiate the password reset request. | string | |
password | New password: The password the user wishes to associate with their account. | string |
code | Six digit verification code: The verification code sent to the user's email address. | string |
Responses
200 OK
{
"message": "Password reset successful"
}
400 Bad Request SchemaValidationError
Request schema validation errors follow a distinct structure across all auth endpoints providing feedback on the fields that failed validation.
{
"name": "SchemaValidationError",
"message": "Schema validation failed",
"data": [
{
"code": "too_small",
"minimum": 6,
"type": "string",
"inclusive": true,
"exact": true,
"message": "String must contain exactly 6 character(s)",
"path": [
"code"
]
}
]
}
400 Bad Request PasswordRegexError
Password regex errors are returned when a supplied password fails to meet criteria set via the application dashboard.
{
"name": "PasswordRegexError",
"message": "Password does not match required pattern"
}
401 Unauthorized UnauthorizedError
To mitigate the opportunity for enumeration attacks, a standard 401 Unauthorized error will be returned if the email address is not recognized or the verification code has expired or is invalid.
{
"name": "AuthenticationError",
"message": "Password reset failed",
}
Example:
const response = await relayBox.auth.passwordConfirm({
email: 'hello@universe.com',
password: 'my-new-super-secure-password',
code: '123456'
});
auth.resendVerification()
Asynchronous function to resend a new user signup verification code.
const response = await relayBox.auth.resendVerification({
email: 'hello@universe.com',
});
Options | Description | Type |
---|---|---|
Valid email address: The email address associated with the account to be verified. | string |
Responses
200 OK
To mitigate enumeration attacks, the response will be a success message unless schema validation fails.
{
"message": "Verification code sent"
}
400 Bad Request SchemaValidationError
Request schema validation errors follow a distinct structure across all auth endpoints providing feedback on the fields that failed validation.
{
"name": "SchemaValidationError",
"message": "Schema validation failed",
"data": [
{
"validation": "email",
"code": "invalid_string",
"message": "Invalid email",
"path": [
"email"
]
}
]
}
Example:
const session = await relayBox.auth.resendVerification({
email: 'hello@universe.com'
});
auth.signOut()
Synchronous function to sign a user out of the application. Destroys the current session including authentication and refresh tokens.
relayBox.auth.signOut();
Example:
relayBox.auth.signOut();
auth.tokenRefresh()
Asynchronous function to refresh the logged in user's authentication token. This is a helper method that should be used with caution. The recommended approach is to set the token expiry and let the application automatically handle refreshing the token at regular intervals.
const result = await relayBox.auth.tokenRefresh();
Responses
200 OK
A sucessful response will contain a new authentication token, along with relevant expiry information.
{
"token": "xxxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"expiresIn": 300, // Seconds
"expiresAt": 1726219018607 // Milliseconds since epoch
}
403 Forbidden TokenExpiredError
If the refresh token used to refresh the authentication token has expired, the response will contain a TokenExpiredError with a status code of 403.
{
"name": "TokenExpiredError",
"message": "jwt expired"
}
Example:
const tokenResponse = await relayBox.auth.tokenRefresh();
auth.getSession()
Asynchronous function to return the session data for the logged in user
const result = await relayBox.auth.getSession({
verify: false
});
Options | Description | Type |
---|---|---|
verify | Verify session data: If true, an HTTP request will be sent via the auth service to fetch session data directly from source. If false, the session data currently in memory will be returned | boolean |
Responses
200 OK
Returns the session data for the logged in user.
{
"session": {
"token": "xxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"refreshToken": "xxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"expiresIn": 300,
"expiresAt": 1726174042498,
"destroyAt": 1726177342498,
"authStorageType": "persist"
},
"user": {
"id": "12345678-1234-1234-1234-123456789012",
"orgId": "12345678-1234-1234-1234-123456789012",
"appId": "12345678-1234-1234-1234-123456789012",
"username": "ShinyKrill709",
"clientId": "0fyWl1VWj8Nx",
"email": "hello@universe.com",
"createdAt": "2024-09-10T18:14:48.758Z",
"updatedAt": "2024-09-12T11:22:30.194Z",
"verifiedAt": "2024-09-10T18:19:45.952Z",
"authMfaEnabled": true,
"blockedAt": null,
"identities": [
{
"id": "12345678-1234-1234-1234-123456789012",
"provider": "github",
"providerId": "123456",
"verifiedAt": "2024-09-10T18:14:48.908"
},
{
"id": "12345678-1234-1234-1234-123456789012",
"provider": "google",
"providerId": "123456",
"verifiedAt": "2024-09-10T20:25:42.719"
},
{
"id": "12345678-1234-1234-1234-123456789012",
"provider": "email",
"providerId": null,
"verifiedAt": "2024-09-10T18:19:45.952"
}
],
"factors": [
{
"id": "12345678-1234-1234-1234-123456789012",
"type": "totp",
"verifiedAt": "2024-09-10T18:21:07.231"
}
]
}
}
403 Forbidden Forbidden Error
In case of failure, the response will return an HTTP authorizer error.
{
"statusCode": 403,
"error": "Forbidden",
"message": "User is not authorized to access this resource"
}
Example
const authUserSession = await relayBox.auth.getSession({
verify: true
});
auth.signInWithOAuth()
Synchronous function to initiate user sign in via a 3rd party social identity provider.
relayBox.auth.signInWithOauth({
provider: 'google | github'
});
// Following successful authentication
relayBox.auth.onAuthEvent('SIGN_IN | MFA_REQUIRED', (session: any) => {
console.log(session)
})
Options | Description | Type |
---|---|---|
provider | Name of social identity provider: The social identity provider to be used for oauth authentication. Currently, 'google' and 'github' are supported. | string |
This action opens a popup window to the social identity provider. The user will be redirected back to the application once authorization is complete.
Either SIGN_IN or MFA_REQUIRED will be emitted to the auth.onAuthEvent() listener along with the relevant data to complete the request.
Example:
relayBox.auth.signInWithOauth({
provider: 'google | github'
});
auth.mfa.enroll()
Asynchronous function to enroll a user for multi factor authentication.
const response = await relayBox.auth.mfa.enroll({
type: 'totp'
});
Options | Description | Type |
---|---|---|
type | MFA type: Currently only "totp" is supported. | string |
Responses
200 OK
Returns the necessary data for enrolling a user in multi factor authentication.
{
"id": "12345678-1234-1234-1234-123456789012",
"type": "totp",
"secret": "J4GDGXAJBM2UYJKSJ5FCI2TYHBQSO5TD",
"qrCodeUri": ""
}
Attribute | Description | Type |
---|---|---|
id | The factor id: Required by auth.challenge() and auth.verify() to authenticate the user | string |
type | MFA type: Currently only "totp" is supported. | string |
secret | MFA secret key: Can be used in place of a QR code to register an authentication device. | string |
qrCodeUri | MFA QR code: Base64 encoded image data used by an authenticator app to register a new device. | string |
403 Forbidden DuplicateKeyError
If the user is already enrolled for multi factor authentication, the response will return a DuplicateKeyError.
{
"name": "DuplicateKeyError",
"message": "MFA type \"totp\" already verified for user"
}
Example:
const response = await relayBox.auth.mfa.enroll({
type: 'totp'
});
auth.mfa.challenge()
Asynchronous function to request a multi factor authentication challenge.
const challenge = await relayBox.auth.challenge({
factorId: 'a7821a11-42a2-494b-abdb-0a26cfa01421'
});
Options | Description | Type |
---|---|---|
factorId | MFA factor ID: The ID of the factor to be verified. This will be in uuid format. The factor ID will be the id returned from mfa.enroll() or can be found in the array of factors in the authUserSession object following sign in. | string (uuid) |
Returns
Return value includes the challenge id and the verify method. The verify method is the final step in the multi factor authentication process. It accepts an authenticator app code as a parameter and returns an authUserSession object.
{
id: challenge.id,
verify: ({ code }) => Promise<AuthUserSession>;
};
Responses
200 OKReturns the challenge id and expirty time in milliseconds since the epoch.
{
"id": "12345678-1234-1234-1234-123456789012",
"expiresAt": 1726234496406
}
400 Bad Request SchemaValidationError
Request schema validation errors follow a distinct structure across all auth endpoints providing feedback on the fields that failed validation.
{
"name": "SchemaValidationError",
"message": "Schema validation failed",
"data": [
{
"validation": "uuid",
"code": "invalid_string",
"message": "Invalid uuid",
"path": [
"factorId"
]
}
]
}
403 Forbidden ValidationError
If the factor id is invalid, the response will return a ValidationError.
{
"name": "ValidationError",
"message": "Invalid factor id"
}
Example:
const mfaEnrollResponse = await relayBox.auth.mfa.enroll({
type: 'totp'
});
const response = await relayBox.auth.mfa.challenge({
factorId: mfaEnrollResponse.id
});
auth.mfa.verify()
Asynchronous function to verify a user's multi factor authentication factor challenge.
const response = await relayBox.auth.verify({
factorId: 'a7821a11-42a2-494b-abdb-0a26cfa01421',
challengeId: 'a7821a11-42a2-494b-abdb-0a26cfa01421',
code: '123456',
autoChallenge: false
});
Options | Description | Type |
---|---|---|
factorId | MFA factor ID: The identifier of the factor to be verified. | string |
challengeId (conditional) | Short lived challenge ID: The id of the specfic challenge to be verified. Condition: Required if autoChallenge is false. | string |
code | Six digit code: The time sensitive code from an authenticator app. | string |
autoChallenge (optional) | Automatic challenge creation: If true, a challenge will be automatically created on the server when the verify request is recieved. In this scenario, a challenge id is not required. | string |
Responses
200 OK AuthUserSession
{
"session": {
"token": "xxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"refreshToken": "xxxxx.xxxxxxx.xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"expiresIn": 300,
"expiresAt": 1726174042498,
"destroyAt": 1726177342498,
"authStorageType": "persist"
},
"user": {
"id": "af8f910e-c72c-4335-a104-48212ad65c1f",
"orgId": "8f6a2232-63e2-4ff7-969f-417739d155ed",
"appId": "675e81c4-2b88-4167-a33b-0d564d2548a5",
"username": "ShinyKrill709",
"clientId": "0fyWl1VWj8Nx",
"email": "hello@universe.com",
"createdAt": "2024-09-10T18:14:48.758Z",
"updatedAt": "2024-09-12T11:22:30.194Z",
"verifiedAt": "2024-09-10T18:19:45.952Z",
"authMfaEnabled": true,
"blockedAt": null,
"identities": [
{
"id": "654e6295-6927-4ce2-98b1-dca79a868f05",
"provider": "github",
"providerId": "123456",
"verifiedAt": "2024-09-10T18:14:48.908"
},
{
"id": "fc7c67c7-315d-4bfd-8ae3-8e58bc36d4ae",
"provider": "google",
"providerId": "123456",
"verifiedAt": "2024-09-10T20:25:42.719"
},
{
"id": "55734265-ba2a-4cf8-b41b-3cd63d097a04",
"provider": "email",
"providerId": null,
"verifiedAt": "2024-09-10T18:19:45.952"
}
],
"factors": [
{
"id": "0aa7af26-3313-40ae-9857-24ae66ea939e",
"type": "totp",
"verifiedAt": "2024-09-10T18:21:07.231"
}
]
}
}
403 Forbidden ValidationError
When an invalid factor id is provided, the response will return a 403 ValidationError.
{
"name": "ValidationError",
"message": "Invalid mfa factor id"
}
403 Forbidden ValidationError
When an invalid factor id is provided, the response will return a 403 ValidationError.
{
"name": "ValidationError",
"message": "Unable to verify mfa code"
}
400 Bad Request SchemaValidationError
Request schema validation errors follow a distinct structure across all auth endpoints providing feedback on the fields that failed validation.
{
"data": [
{
"code": "invalid_type",
"expected": "string",
"received": "undefined",
"path": [
"factorId"
],
"message": "Required"
}
],
"name": "SchemaValidationError",
"message": "Schema validation failed"
}
403 Forbidden ForbiddenError
When the challenge has expired, the response will return a 403 ForbiddenError.
{
"statusCode": 403,
"error": "Forbidden",
"message": "User is not authorized to access this resource"
}
400 Bad request ValidationError
If neither a challengeId or autoChallenge is provided, the response will return a 400 ValidationError.
{
"name": "ValidationError",
"message": "\"challengeId\" or \"autoChallenge\" required"
}
Example:
// MFA verify with auto challenge
const session = await relayBox.auth.mfa.verify({
factorId: '12345678-1234-1234-1234-123456789012',
code: '123456',
autoChallenge: true
});
// MFA verify with challenge id
const session = await relayBox.auth.mfa.verify({
factorId: '12345678-1234-1234-1234-123456789012',
challengeId: '12345678-1234-1234-1234-123456789012',
code: '123456'
});
// MFA verify with challenge return values
const challenge = relayBox.auth.mfa.challenge({
factorId: '12345678-1234-1234-1234-123456789012'
})
const response = challenge.verify({
code: '123456'
})