2024-10-30 18:50:32 +01:00
|
|
|
import fetch from "node-fetch";
|
2024-10-28 19:44:50 +01:00
|
|
|
import {Page} from "@playwright/test";
|
|
|
|
|
import {registerWithPasskey} from "./register";
|
|
|
|
|
import {loginWithPasskey, loginWithPassword} from "./login";
|
|
|
|
|
import {changePassword} from "./password";
|
|
|
|
|
|
|
|
|
|
export interface userProps {
|
|
|
|
|
email: string;
|
|
|
|
|
firstName: string;
|
|
|
|
|
lastName: string;
|
|
|
|
|
organization: string;
|
|
|
|
|
password: string;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class User {
|
|
|
|
|
private readonly props: userProps;
|
|
|
|
|
private user: string;
|
|
|
|
|
|
|
|
|
|
constructor(userProps: userProps) {
|
|
|
|
|
this.props = userProps;
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-30 18:50:32 +01:00
|
|
|
async ensure(page: Page) {
|
2024-10-28 19:44:50 +01:00
|
|
|
await this.remove()
|
|
|
|
|
|
|
|
|
|
const body = {
|
|
|
|
|
username: this.props.email,
|
|
|
|
|
organization: {
|
|
|
|
|
orgId: this.props.organization
|
|
|
|
|
},
|
|
|
|
|
profile: {
|
|
|
|
|
givenName: this.props.firstName,
|
|
|
|
|
familyName: this.props.lastName,
|
|
|
|
|
},
|
|
|
|
|
email: {
|
|
|
|
|
email: this.props.email,
|
|
|
|
|
isVerified: true,
|
|
|
|
|
},
|
|
|
|
|
password: {
|
|
|
|
|
password: this.props.password!,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const response = await fetch(process.env.ZITADEL_API_URL! + "/v2/users/human", {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
body: JSON.stringify(body),
|
|
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
|
'Authorization': "Bearer " + process.env.ZITADEL_SERVICE_USER_TOKEN!
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
if (response.statusCode >= 400 && response.statusCode != 409) {
|
|
|
|
|
const error = 'HTTP Error: ' + response.statusCode + ' - ' + response.statusMessage;
|
|
|
|
|
console.error(error);
|
|
|
|
|
throw new Error(error);
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async remove() {
|
|
|
|
|
const response = await fetch(process.env.ZITADEL_API_URL! + "/v2/users/" + this.userId(), {
|
|
|
|
|
method: 'DELETE',
|
|
|
|
|
headers: {
|
|
|
|
|
'Authorization': "Bearer " + process.env.ZITADEL_SERVICE_USER_TOKEN!
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
if (response.statusCode >= 400 && response.statusCode != 404) {
|
|
|
|
|
const error = 'HTTP Error: ' + response.statusCode + ' - ' + response.statusMessage;
|
|
|
|
|
console.error(error);
|
|
|
|
|
throw new Error(error);
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-30 18:50:32 +01:00
|
|
|
public setUserId(userId: string) {
|
|
|
|
|
this.user = userId
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-28 19:44:50 +01:00
|
|
|
public userId() {
|
|
|
|
|
return this.user;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public username() {
|
|
|
|
|
return this.props.email;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public password() {
|
|
|
|
|
return this.props.password;
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-30 18:50:32 +01:00
|
|
|
public firstname() {
|
|
|
|
|
return this.props.firstName
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public lastname() {
|
|
|
|
|
return this.props.lastName
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-28 19:44:50 +01:00
|
|
|
public fullName() {
|
|
|
|
|
return this.props.firstName + " " + this.props.lastName
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async login(page: Page) {
|
|
|
|
|
await loginWithPassword(page, this.username(), this.password())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async changePassword(page: Page, password: string) {
|
|
|
|
|
await loginWithPassword(page, this.username(), this.password())
|
|
|
|
|
await changePassword(page, this.username(), password)
|
|
|
|
|
this.props.password = password
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export class PasswordUser extends User {
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-30 18:50:32 +01:00
|
|
|
enum OtpType {
|
|
|
|
|
time = "time-based",
|
|
|
|
|
sms = "sms",
|
|
|
|
|
email = "email",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface otpUserProps {
|
2024-10-28 19:44:50 +01:00
|
|
|
email: string;
|
|
|
|
|
firstName: string;
|
|
|
|
|
lastName: string;
|
|
|
|
|
organization: string;
|
2024-10-30 18:50:32 +01:00
|
|
|
type: OtpType,
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|
|
|
|
|
|
2024-10-30 18:50:32 +01:00
|
|
|
export class PasswordUserWithOTP extends User {
|
|
|
|
|
private type: OtpType
|
|
|
|
|
private code: string
|
|
|
|
|
|
|
|
|
|
constructor(props: otpUserProps) {
|
|
|
|
|
super({
|
|
|
|
|
email: props.email,
|
|
|
|
|
firstName: props.firstName,
|
|
|
|
|
lastName: props.lastName,
|
|
|
|
|
organization: props.organization,
|
|
|
|
|
password: ""
|
|
|
|
|
})
|
|
|
|
|
this.type = props.type
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|
|
|
|
|
|
2024-10-30 18:50:32 +01:00
|
|
|
async ensure(page: Page) {
|
|
|
|
|
await super.ensure(page)
|
2024-10-28 19:44:50 +01:00
|
|
|
|
2024-10-30 18:50:32 +01:00
|
|
|
const body = {
|
|
|
|
|
username: this.props.email,
|
|
|
|
|
organization: {
|
|
|
|
|
orgId: this.props.organization
|
|
|
|
|
},
|
|
|
|
|
profile: {
|
|
|
|
|
givenName: this.props.firstName,
|
|
|
|
|
familyName: this.props.lastName,
|
|
|
|
|
},
|
|
|
|
|
email: {
|
|
|
|
|
email: this.props.email,
|
|
|
|
|
isVerified: true,
|
|
|
|
|
},
|
|
|
|
|
password: {
|
|
|
|
|
password: this.props.password!,
|
|
|
|
|
}
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|
2024-10-30 18:50:32 +01:00
|
|
|
|
|
|
|
|
const response = await fetch(process.env.ZITADEL_API_URL! + "/v2/users/human", {
|
2024-10-28 19:44:50 +01:00
|
|
|
method: 'POST',
|
2024-10-30 18:50:32 +01:00
|
|
|
body: JSON.stringify(body),
|
2024-10-28 19:44:50 +01:00
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
|
'Authorization': "Bearer " + process.env.ZITADEL_SERVICE_USER_TOKEN!
|
|
|
|
|
}
|
|
|
|
|
});
|
2024-10-30 18:50:32 +01:00
|
|
|
if (response.statusCode >= 400 && response.statusCode != 409) {
|
|
|
|
|
const error = 'HTTP Error: ' + response.statusCode + ' - ' + response.statusMessage;
|
2024-10-28 19:44:50 +01:00
|
|
|
console.error(error);
|
|
|
|
|
throw new Error(error);
|
|
|
|
|
}
|
2024-10-30 18:50:32 +01:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface passkeyUserProps {
|
|
|
|
|
email: string;
|
|
|
|
|
firstName: string;
|
|
|
|
|
lastName: string;
|
|
|
|
|
organization: string;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export class PasskeyUser extends User {
|
|
|
|
|
constructor(props: passkeyUserProps) {
|
|
|
|
|
super({
|
|
|
|
|
email: props.email,
|
|
|
|
|
firstName: props.firstName,
|
|
|
|
|
lastName: props.lastName,
|
|
|
|
|
organization: props.organization,
|
|
|
|
|
password: ""
|
|
|
|
|
})
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|
|
|
|
|
|
2024-10-30 18:50:32 +01:00
|
|
|
public async ensure(page: Page) {
|
|
|
|
|
await this.remove()
|
|
|
|
|
await registerWithPasskey(page, this.firstname(), this.lastname(), this.username())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async login(page: Page) {
|
|
|
|
|
await loginWithPasskey(page, this.username())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async remove() {
|
|
|
|
|
const resp = await getUserByUsername(this.username())
|
|
|
|
|
if (!resp || !resp.result || !resp.result[0]) {
|
|
|
|
|
return
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|
2024-10-30 18:50:32 +01:00
|
|
|
this.setUserId(resp.result[0].userId)
|
|
|
|
|
await super.remove()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function getUserByUsername(username: string) {
|
|
|
|
|
const listUsersBody = {
|
|
|
|
|
queries: [{
|
|
|
|
|
userNameQuery: {
|
|
|
|
|
userName: username,
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|
2024-10-30 18:50:32 +01:00
|
|
|
}]
|
|
|
|
|
}
|
|
|
|
|
const jsonBody = JSON.stringify(listUsersBody)
|
|
|
|
|
const registerResponse = await fetch(process.env.ZITADEL_API_URL! + "/v2/users", {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
body: jsonBody,
|
|
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
|
'Authorization': "Bearer " + process.env.ZITADEL_SERVICE_USER_TOKEN!
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|
2024-10-30 18:50:32 +01:00
|
|
|
});
|
|
|
|
|
if (registerResponse.statusCode >= 400) {
|
|
|
|
|
const error = 'HTTP Error: ' + registerResponse.statusCode + ' - ' + registerResponse.statusMessage;
|
|
|
|
|
console.error(error);
|
|
|
|
|
throw new Error(error);
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|
2024-10-30 18:50:32 +01:00
|
|
|
const respJson = await registerResponse.json()
|
|
|
|
|
return respJson
|
2024-10-28 19:44:50 +01:00
|
|
|
}
|