Files
zitadel/apps/login/src/lib/zitadel.ts
2025-01-29 16:00:21 +01:00

1439 lines
30 KiB
TypeScript

import { Client, create, Duration } from "@zitadel/client";
import { makeReqCtx } from "@zitadel/client/v2";
import { IdentityProviderService } from "@zitadel/proto/zitadel/idp/v2/idp_service_pb";
import { TextQueryMethod } from "@zitadel/proto/zitadel/object/v2/object_pb";
import {
CreateCallbackRequest,
OIDCService,
} from "@zitadel/proto/zitadel/oidc/v2/oidc_service_pb";
import { Organization } from "@zitadel/proto/zitadel/org/v2/org_pb";
import { OrganizationService } from "@zitadel/proto/zitadel/org/v2/org_service_pb";
import { RequestChallenges } from "@zitadel/proto/zitadel/session/v2/challenge_pb";
import {
Checks,
SessionService,
} from "@zitadel/proto/zitadel/session/v2/session_service_pb";
import { LoginSettings } from "@zitadel/proto/zitadel/settings/v2/login_settings_pb";
import { SettingsService } from "@zitadel/proto/zitadel/settings/v2/settings_service_pb";
import { SendEmailVerificationCodeSchema } from "@zitadel/proto/zitadel/user/v2/email_pb";
import type { RedirectURLsJson } from "@zitadel/proto/zitadel/user/v2/idp_pb";
import {
NotificationType,
SendPasswordResetLinkSchema,
} from "@zitadel/proto/zitadel/user/v2/password_pb";
import {
SearchQuery,
SearchQuerySchema,
} from "@zitadel/proto/zitadel/user/v2/query_pb";
import {
SendInviteCodeSchema,
User,
UserState,
} from "@zitadel/proto/zitadel/user/v2/user_pb";
import {
AddHumanUserRequest,
ResendEmailCodeRequest,
ResendEmailCodeRequestSchema,
SendEmailCodeRequestSchema,
SetPasswordRequest,
SetPasswordRequestSchema,
UserService,
VerifyPasskeyRegistrationRequest,
VerifyU2FRegistrationRequest,
} from "@zitadel/proto/zitadel/user/v2/user_service_pb";
import { unstable_cacheLife as cacheLife } from "next/cache";
import { createServiceForHost } from "./service";
const useCache = process.env.DEBUG !== "true";
async function cacheWrapper<T>(callback: Promise<T>) {
"use cache";
cacheLife("hours");
return callback;
}
export async function getBrandingSettings({
serviceUrl,
serviceRegion,
organization,
}: {
serviceUrl: string;
serviceRegion: string;
organization?: string;
}) {
const settingsService: Client<typeof SettingsService> =
await createServiceForHost(SettingsService, serviceUrl, serviceRegion);
const callback = settingsService
.getBrandingSettings({ ctx: makeReqCtx(organization) }, {})
.then((resp) => (resp.settings ? resp.settings : undefined));
return useCache ? cacheWrapper(callback) : callback;
}
export async function getLoginSettings({
serviceUrl,
serviceRegion,
organization,
}: {
serviceUrl: string;
serviceRegion: string;
organization?: string;
}) {
const settingsService: Client<typeof SettingsService> =
await createServiceForHost(SettingsService, serviceUrl, serviceRegion);
const callback = settingsService
.getLoginSettings({ ctx: makeReqCtx(organization) }, {})
.then((resp) => (resp.settings ? resp.settings : undefined));
return useCache ? cacheWrapper(callback) : callback;
}
export async function listIDPLinks({
serviceUrl,
serviceRegion,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.listIDPLinks({ userId }, {});
}
export async function addOTPEmail({
serviceUrl,
serviceRegion,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.addOTPEmail({ userId }, {});
}
export async function addOTPSMS({
serviceUrl,
serviceRegion,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.addOTPSMS({ userId }, {});
}
export async function registerTOTP({
serviceUrl,
serviceRegion,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.registerTOTP({ userId }, {});
}
export async function getGeneralSettings({
serviceUrl,
serviceRegion,
}: {
serviceUrl: string;
serviceRegion: string;
}) {
const settingsService: Client<typeof SettingsService> =
await createServiceForHost(SettingsService, serviceUrl, serviceRegion);
const callback = settingsService
.getGeneralSettings({}, {})
.then((resp) => resp.supportedLanguages);
return useCache ? cacheWrapper(callback) : callback;
}
export async function getLegalAndSupportSettings({
serviceUrl,
serviceRegion,
organization,
}: {
serviceUrl: string;
serviceRegion: string;
organization?: string;
}) {
const settingsService: Client<typeof SettingsService> =
await createServiceForHost(SettingsService, serviceUrl, serviceRegion);
const callback = settingsService
.getLegalAndSupportSettings({ ctx: makeReqCtx(organization) }, {})
.then((resp) => (resp.settings ? resp.settings : undefined));
return useCache ? cacheWrapper(callback) : callback;
}
export async function getPasswordComplexitySettings({
serviceUrl,
serviceRegion,
organization,
}: {
serviceUrl: string;
serviceRegion: string;
organization?: string;
}) {
const settingsService: Client<typeof SettingsService> =
await createServiceForHost(SettingsService, serviceUrl, serviceRegion);
const callback = settingsService
.getPasswordComplexitySettings({ ctx: makeReqCtx(organization) })
.then((resp) => (resp.settings ? resp.settings : undefined));
return useCache ? cacheWrapper(callback) : callback;
}
export async function createSessionFromChecks({
serviceUrl,
serviceRegion,
checks,
challenges,
lifetime,
}: {
serviceUrl: string;
serviceRegion: string;
checks: Checks;
challenges: RequestChallenges | undefined;
lifetime?: Duration;
}) {
const sessionService: Client<typeof SessionService> =
await createServiceForHost(SessionService, serviceUrl, serviceRegion);
return sessionService.createSession({ checks, challenges, lifetime }, {});
}
export async function createSessionForUserIdAndIdpIntent({
serviceUrl,
serviceRegion,
userId,
idpIntent,
lifetime,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
idpIntent: {
idpIntentId?: string | undefined;
idpIntentToken?: string | undefined;
};
lifetime?: Duration;
}) {
const sessionService: Client<typeof SessionService> =
await createServiceForHost(SessionService, serviceUrl, serviceRegion);
return sessionService.createSession({
checks: {
user: {
search: {
case: "userId",
value: userId,
},
},
idpIntent,
},
lifetime,
});
}
export async function setSession({
serviceUrl,
serviceRegion,
sessionId,
sessionToken,
challenges,
checks,
lifetime,
}: {
serviceUrl: string;
serviceRegion: string;
sessionId: string;
sessionToken: string;
challenges: RequestChallenges | undefined;
checks?: Checks;
lifetime?: Duration;
}) {
const sessionService: Client<typeof SessionService> =
await createServiceForHost(SessionService, serviceUrl, serviceRegion);
return sessionService.setSession(
{
sessionId,
sessionToken,
challenges,
checks: checks ? checks : {},
metadata: {},
lifetime,
},
{},
);
}
export async function getSession({
serviceUrl,
serviceRegion,
sessionId,
sessionToken,
}: {
serviceUrl: string;
serviceRegion: string;
sessionId: string;
sessionToken: string;
}) {
const sessionService: Client<typeof SessionService> =
await createServiceForHost(SessionService, serviceUrl, serviceRegion);
return sessionService.getSession({ sessionId, sessionToken }, {});
}
export async function deleteSession({
serviceUrl,
serviceRegion,
sessionId,
sessionToken,
}: {
serviceUrl: string;
serviceRegion: string;
sessionId: string;
sessionToken: string;
}) {
const sessionService: Client<typeof SessionService> =
await createServiceForHost(SessionService, serviceUrl, serviceRegion);
return sessionService.deleteSession({ sessionId, sessionToken }, {});
}
type ListSessionsCommand = {
serviceUrl: string;
serviceRegion: string;
ids: string[];
};
export async function listSessions({
serviceUrl,
serviceRegion,
ids,
}: ListSessionsCommand) {
const sessionService: Client<typeof SessionService> =
await createServiceForHost(SessionService, serviceUrl, serviceRegion);
return sessionService.listSessions(
{
queries: [
{
query: {
case: "idsQuery",
value: { ids },
},
},
],
},
{},
);
}
export type AddHumanUserData = {
serviceUrl: string;
serviceRegion: string;
firstName: string;
lastName: string;
email: string;
password: string | undefined;
organization: string | undefined;
};
export async function addHumanUser({
serviceUrl,
serviceRegion,
email,
firstName,
lastName,
password,
organization,
}: AddHumanUserData) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.addHumanUser({
email: {
email,
verification: {
case: "isVerified",
value: false,
},
},
username: email,
profile: { givenName: firstName, familyName: lastName },
organization: organization
? { org: { case: "orgId", value: organization } }
: undefined,
passwordType: password
? { case: "password", value: { password } }
: undefined,
});
}
export async function addHuman({
serviceUrl,
serviceRegion,
request,
}: {
serviceUrl: string;
serviceRegion: string;
request: AddHumanUserRequest;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.addHumanUser(request);
}
export async function verifyTOTPRegistration({
serviceUrl,
serviceRegion,
code,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
code: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.verifyTOTPRegistration({ code, userId }, {});
}
export async function getUserByID({
serviceUrl,
serviceRegion,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.getUserByID({ userId }, {});
}
export async function verifyInviteCode({
serviceUrl,
serviceRegion,
userId,
verificationCode,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
verificationCode: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.verifyInviteCode({ userId, verificationCode }, {});
}
export async function resendInviteCode({
serviceUrl,
serviceRegion,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.resendInviteCode({ userId }, {});
}
export async function sendEmailCode({
serviceUrl,
serviceRegion,
userId,
urlTemplate,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
urlTemplate: string;
}) {
let medium = create(SendEmailCodeRequestSchema, { userId });
medium = create(SendEmailCodeRequestSchema, {
...medium,
verification: {
case: "sendCode",
value: create(SendEmailVerificationCodeSchema, {
urlTemplate,
}),
},
});
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.sendEmailCode(medium, {});
}
export async function createInviteCode({
serviceUrl,
serviceRegion,
urlTemplate,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
urlTemplate: string;
userId: string;
}) {
let medium = create(SendInviteCodeSchema, {
applicationName: "Typescript Login",
});
medium = {
...medium,
urlTemplate,
};
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.createInviteCode(
{
userId,
verification: {
case: "sendCode",
value: medium,
},
},
{},
);
}
export type ListUsersCommand = {
serviceUrl: string;
serviceRegion: string;
loginName?: string;
userName?: string;
email?: string;
phone?: string;
organizationId?: string;
};
export async function listUsers({
serviceUrl,
serviceRegion,
loginName,
userName,
phone,
email,
organizationId,
}: ListUsersCommand) {
const queries: SearchQuery[] = [];
// either use loginName or userName, email, phone
if (loginName) {
queries.push(
create(SearchQuerySchema, {
query: {
case: "loginNameQuery",
value: {
loginName,
method: TextQueryMethod.EQUALS,
},
},
}),
);
} else if (userName || email || phone) {
const orQueries: SearchQuery[] = [];
if (userName) {
const userNameQuery = create(SearchQuerySchema, {
query: {
case: "userNameQuery",
value: {
userName,
method: TextQueryMethod.EQUALS,
},
},
});
orQueries.push(userNameQuery);
}
if (email) {
const emailQuery = create(SearchQuerySchema, {
query: {
case: "emailQuery",
value: {
emailAddress: email,
method: TextQueryMethod.EQUALS,
},
},
});
orQueries.push(emailQuery);
}
if (phone) {
const phoneQuery = create(SearchQuerySchema, {
query: {
case: "phoneQuery",
value: {
number: phone,
method: TextQueryMethod.EQUALS,
},
},
});
orQueries.push(phoneQuery);
}
queries.push(
create(SearchQuerySchema, {
query: {
case: "orQuery",
value: {
queries: orQueries,
},
},
}),
);
}
if (organizationId) {
queries.push(
create(SearchQuerySchema, {
query: {
case: "organizationIdQuery",
value: {
organizationId,
},
},
}),
);
}
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.listUsers({ queries });
}
export type SearchUsersCommand = {
serviceUrl: string;
serviceRegion: string;
searchValue: string;
loginSettings: LoginSettings;
organizationId?: string;
suffix?: string;
};
const PhoneQuery = (searchValue: string) =>
create(SearchQuerySchema, {
query: {
case: "phoneQuery",
value: {
number: searchValue,
method: TextQueryMethod.EQUALS,
},
},
});
const LoginNameQuery = (searchValue: string) =>
create(SearchQuerySchema, {
query: {
case: "loginNameQuery",
value: {
loginName: searchValue,
method: TextQueryMethod.EQUALS,
},
},
});
const EmailQuery = (searchValue: string) =>
create(SearchQuerySchema, {
query: {
case: "emailQuery",
value: {
emailAddress: searchValue,
method: TextQueryMethod.EQUALS,
},
},
});
/**
* this is a dedicated search function to search for users from the loginname page
* it searches users based on the loginName or userName and org suffix combination, and falls back to email and phone if no users are found
* */
export async function searchUsers({
serviceUrl,
serviceRegion,
searchValue,
loginSettings,
organizationId,
suffix,
}: SearchUsersCommand) {
const queries: SearchQuery[] = [];
// if a suffix is provided, we search for the userName concatenated with the suffix
if (suffix) {
const searchValueWithSuffix = `${searchValue}@${suffix}`;
const loginNameQuery = LoginNameQuery(searchValueWithSuffix);
queries.push(loginNameQuery);
} else {
const loginNameQuery = LoginNameQuery(searchValue);
queries.push(loginNameQuery);
}
if (organizationId) {
queries.push(
create(SearchQuerySchema, {
query: {
case: "organizationIdQuery",
value: {
organizationId,
},
},
}),
);
}
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
const loginNameResult = await userService.listUsers({ queries });
if (!loginNameResult || !loginNameResult.details) {
return { error: "An error occurred." };
}
if (loginNameResult.result.length > 1) {
return { error: "Multiple users found" };
}
if (loginNameResult.result.length == 1) {
return loginNameResult;
}
const emailAndPhoneQueries: SearchQuery[] = [];
if (
loginSettings.disableLoginWithEmail &&
loginSettings.disableLoginWithPhone
) {
return { error: "User not found in the system" };
} else if (loginSettings.disableLoginWithEmail && searchValue.length <= 20) {
const phoneQuery = PhoneQuery(searchValue);
emailAndPhoneQueries.push(phoneQuery);
} else if (loginSettings.disableLoginWithPhone) {
const emailQuery = EmailQuery(searchValue);
emailAndPhoneQueries.push(emailQuery);
} else {
const emailAndPhoneOrQueries: SearchQuery[] = [];
const emailQuery = EmailQuery(searchValue);
emailAndPhoneOrQueries.push(emailQuery);
let phoneQuery;
if (searchValue.length <= 20) {
phoneQuery = PhoneQuery(searchValue);
emailAndPhoneOrQueries.push(phoneQuery);
}
emailAndPhoneQueries.push(
create(SearchQuerySchema, {
query: {
case: "orQuery",
value: {
queries: emailAndPhoneOrQueries,
},
},
}),
);
}
if (organizationId) {
queries.push(
create(SearchQuerySchema, {
query: {
case: "organizationIdQuery",
value: {
organizationId,
},
},
}),
);
}
const emailOrPhoneResult = await userService.listUsers({
queries: emailAndPhoneQueries,
});
if (!emailOrPhoneResult || !emailOrPhoneResult.details) {
return { error: "An error occurred." };
}
if (emailOrPhoneResult.result.length > 1) {
return { error: "Multiple users found." };
}
if (emailOrPhoneResult.result.length == 1) {
return loginNameResult;
}
return { error: "User not found in the system" };
}
export async function getDefaultOrg({
serviceUrl,
serviceRegion,
}: {
serviceUrl: string;
serviceRegion: string;
}): Promise<Organization | null> {
const orgService: Client<typeof OrganizationService> =
await createServiceForHost(OrganizationService, serviceUrl, serviceRegion);
return orgService
.listOrganizations(
{
queries: [
{
query: {
case: "defaultQuery",
value: {},
},
},
],
},
{},
)
.then((resp) => (resp?.result && resp.result[0] ? resp.result[0] : null));
}
export async function getOrgsByDomain({
serviceUrl,
serviceRegion,
domain,
}: {
serviceUrl: string;
serviceRegion: string;
domain: string;
}) {
const orgService: Client<typeof OrganizationService> =
await createServiceForHost(OrganizationService, serviceUrl, serviceRegion);
return orgService.listOrganizations(
{
queries: [
{
query: {
case: "domainQuery",
value: { domain, method: TextQueryMethod.EQUALS },
},
},
],
},
{},
);
}
export async function startIdentityProviderFlow({
serviceUrl,
serviceRegion,
idpId,
urls,
}: {
serviceUrl: string;
serviceRegion: string;
idpId: string;
urls: RedirectURLsJson;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.startIdentityProviderIntent({
idpId,
content: {
case: "urls",
value: urls,
},
});
}
export async function retrieveIdentityProviderInformation({
serviceUrl,
serviceRegion,
idpIntentId,
idpIntentToken,
}: {
serviceUrl: string;
serviceRegion: string;
idpIntentId: string;
idpIntentToken: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.retrieveIdentityProviderIntent({
idpIntentId,
idpIntentToken,
});
}
export async function getAuthRequest({
serviceUrl,
serviceRegion,
authRequestId,
}: {
serviceUrl: string;
serviceRegion: string;
authRequestId: string;
}) {
const oidcService = await createServiceForHost(
OIDCService,
serviceUrl,
serviceRegion,
);
return oidcService.getAuthRequest({
authRequestId,
});
}
export async function createCallback({
serviceUrl,
serviceRegion,
req,
}: {
serviceUrl: string;
serviceRegion: string;
req: CreateCallbackRequest;
}) {
const oidcService = await createServiceForHost(
OIDCService,
serviceUrl,
serviceRegion,
);
return oidcService.createCallback(req);
}
export async function verifyEmail({
serviceUrl,
serviceRegion,
userId,
verificationCode,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
verificationCode: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.verifyEmail(
{
userId,
verificationCode,
},
{},
);
}
export async function resendEmailCode({
serviceUrl,
serviceRegion,
userId,
urlTemplate,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
urlTemplate: string;
}) {
let request: ResendEmailCodeRequest = create(ResendEmailCodeRequestSchema, {
userId,
});
const medium = create(SendEmailVerificationCodeSchema, {
urlTemplate,
});
request = { ...request, verification: { case: "sendCode", value: medium } };
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.resendEmailCode(request, {});
}
export async function retrieveIDPIntent({
serviceUrl,
serviceRegion,
id,
token,
}: {
serviceUrl: string;
serviceRegion: string;
id: string;
token: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.retrieveIdentityProviderIntent(
{ idpIntentId: id, idpIntentToken: token },
{},
);
}
export async function getIDPByID({
serviceUrl,
serviceRegion,
id,
}: {
serviceUrl: string;
serviceRegion: string;
id: string;
}) {
const idpService: Client<typeof IdentityProviderService> =
await createServiceForHost(
IdentityProviderService,
serviceUrl,
serviceRegion,
);
return idpService.getIDPByID({ id }, {}).then((resp) => resp.idp);
}
export async function addIDPLink({
serviceUrl,
serviceRegion,
idp,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
idp: { id: string; userId: string; userName: string };
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.addIDPLink(
{
idpLink: {
userId: idp.userId,
idpId: idp.id,
userName: idp.userName,
},
userId,
},
{},
);
}
export async function passwordReset({
serviceUrl,
serviceRegion,
userId,
urlTemplate,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
urlTemplate?: string;
}) {
let medium = create(SendPasswordResetLinkSchema, {
notificationType: NotificationType.Email,
});
medium = {
...medium,
urlTemplate,
};
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.passwordReset(
{
userId,
medium: {
case: "sendLink",
value: medium,
},
},
{},
);
}
export async function setUserPassword({
serviceUrl,
serviceRegion,
userId,
password,
user,
code,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
password: string;
user: User;
code?: string;
}) {
let payload = create(SetPasswordRequestSchema, {
userId,
newPassword: {
password,
},
});
// check if the user has no password set in order to set a password
if (!code) {
const authmethods = await listAuthenticationMethodTypes({
serviceUrl,
serviceRegion,
userId,
});
// if the user has no authmethods set, we can set a password otherwise we need a code
if (
!(authmethods.authMethodTypes.length === 0) &&
user.state !== UserState.INITIAL
) {
return { error: "Provide a code to set a password" };
}
}
if (code) {
payload = {
...payload,
verification: {
case: "verificationCode",
value: code,
},
};
}
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.setPassword(payload, {}).catch((error) => {
// throw error if failed precondition (ex. User is not yet initialized)
if (error.code === 9 && error.message) {
return { error: error.message };
} else {
throw error;
}
});
}
export async function setPassword({
serviceUrl,
serviceRegion,
payload,
}: {
serviceUrl: string;
serviceRegion: string;
payload: SetPasswordRequest;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.setPassword(payload, {});
}
/**
*
* @param host
* @param userId the id of the user where the email should be set
* @returns the newly set email
*/
export async function createPasskeyRegistrationLink({
serviceUrl,
serviceRegion,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.createPasskeyRegistrationLink({
userId,
medium: {
case: "returnCode",
value: {},
},
});
}
/**
*
* @param host
* @param userId the id of the user where the email should be set
* @param domain the domain on which the factor is registered
* @returns the newly set email
*/
export async function registerU2F({
serviceUrl,
serviceRegion,
userId,
domain,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
domain: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.registerU2F({
userId,
domain,
});
}
/**
*
* @param host
* @param request the request object for verifying U2F registration
* @returns the result of the verification
*/
export async function verifyU2FRegistration({
serviceUrl,
serviceRegion,
request,
}: {
serviceUrl: string;
serviceRegion: string;
request: VerifyU2FRegistrationRequest;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.verifyU2FRegistration(request, {});
}
/**
*
* @param host
* @param orgId the organization ID
* @param linking_allowed whether linking is allowed
* @returns the active identity providers
*/
export async function getActiveIdentityProviders({
serviceUrl,
serviceRegion,
orgId,
linking_allowed,
}: {
serviceUrl: string;
serviceRegion: string;
orgId?: string;
linking_allowed?: boolean;
}) {
const props: any = { ctx: makeReqCtx(orgId) };
if (linking_allowed) {
props.linkingAllowed = linking_allowed;
}
const settingsService: Client<typeof SettingsService> =
await createServiceForHost(SettingsService, serviceUrl, serviceRegion);
return settingsService.getActiveIdentityProviders(props, {});
}
/**
*
* @param host
* @param request the request object for verifying passkey registration
* @returns the result of the verification
*/
export async function verifyPasskeyRegistration({
serviceUrl,
serviceRegion,
request,
}: {
serviceUrl: string;
serviceRegion: string;
request: VerifyPasskeyRegistrationRequest;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.verifyPasskeyRegistration(request, {});
}
/**
*
* @param host
* @param userId the id of the user where the email should be set
* @param code the code for registering the passkey
* @param domain the domain on which the factor is registered
* @returns the newly set email
*/
export async function registerPasskey({
serviceUrl,
serviceRegion,
userId,
code,
domain,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
code: { id: string; code: string };
domain: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.registerPasskey({
userId,
code,
domain,
});
}
/**
*
* @param host
* @param userId the id of the user where the email should be set
* @returns the list of authentication method types
*/
export async function listAuthenticationMethodTypes({
serviceUrl,
serviceRegion,
userId,
}: {
serviceUrl: string;
serviceRegion: string;
userId: string;
}) {
const userService: Client<typeof UserService> = await createServiceForHost(
UserService,
serviceUrl,
serviceRegion,
);
return userService.listAuthenticationMethodTypes({
userId,
});
}