import { SignJWT } from 'jose' import { IRequestOptions, RestClient } from 'typed-rest-client' import { CONFIG } from '@/config' import { LogError } from '@/server/LogError' import { backendLogger as logger } from '@/server/logger' import { GetUser } from './model/GetUser' import { PostUser } from './model/PostUser' import { PostUserError } from './model/PostUserError' import { UsersResponse } from './model/UsersResponse' /** * HumHubClient as singleton class */ export class HumHubClient { // eslint-disable-next-line no-use-before-define private static instance: HumHubClient private restClient: RestClient // eslint-disable-next-line no-useless-constructor, @typescript-eslint/no-empty-function private constructor() { this.restClient = new RestClient('gradido-backend', CONFIG.HUMHUB_API_URL) logger.info('create rest client for', CONFIG.HUMHUB_API_URL) } public static getInstance(): HumHubClient | undefined { if (!CONFIG.HUMHUB_ACTIVE || !CONFIG.HUMHUB_API_URL) { logger.info(`humhub are disabled via config...`) return } if (!HumHubClient.instance) { HumHubClient.instance = new HumHubClient() } return HumHubClient.instance } protected async createRequestOptions( queryParams?: Record, ): Promise { const requestOptions: IRequestOptions = { additionalHeaders: { authorization: 'Bearer ' + (await this.createJWTToken()) }, } if (queryParams) { requestOptions.queryParameters = { params: queryParams } } return requestOptions } private async createJWTToken(): Promise { const secret = new TextEncoder().encode(CONFIG.HUMHUB_JWT_KEY) const token = await new SignJWT({ 'urn:gradido:claim': true, uid: 1 }) .setProtectedHeader({ alg: 'HS512' }) .setIssuedAt() .setIssuer('urn:gradido:issuer') .setAudience('urn:gradido:audience') .setExpirationTime('5m') .sign(secret) return token } /** * Get all users from humhub * https://marketplace.humhub.com/module/rest/docs/html/user.html#tag/User/paths/~1user/get * @param page The number of page of the result set >= 0 * @param limit The numbers of items to return per page, Default: 20, [1 .. 50] * @returns list of users */ public async users(page = 0, limit = 20): Promise { const options = await this.createRequestOptions({ page, limit }) const response = await this.restClient.get('/api/v1/user', options) if (response.statusCode !== 200) { throw new LogError('error requesting users from humhub', response) } return response.result } /** * get user by email * https://marketplace.humhub.com/module/rest/docs/html/user.html#tag/User/paths/~1user~1get-by-email/get * @param email for user search * @returns user object if found */ public async userByEmail(email: string): Promise { const options = await this.createRequestOptions({ email }) const response = await this.restClient.get('/api/v1/user/get-by-email', options) // 404 = user not found if (response.statusCode === 404) { return null } return response.result } /** * create user * https://marketplace.humhub.com/module/rest/docs/html/user.html#tag/User/paths/~1user/post * @param user for saving on humhub instance */ public async createUser(user: PostUser): Promise { const options = await this.createRequestOptions() try { const response = await this.restClient.create('/api/v1/user', user, options) if (response.statusCode !== 200) { throw new LogError('error creating user on humhub', { user, response }) } } catch (error) { throw new LogError('error on creating new user', { user, error: JSON.stringify(error, null, 2), }) } } /** * update user * https://marketplace.humhub.com/module/rest/docs/html/user.html#tag/User/operation/updateUser * @param user user object to update * @param humhubUserId humhub user id * @returns updated user object on success */ public async updateUser(user: PostUser, humhubUserId: number): Promise { const options = await this.createRequestOptions() const response = await this.restClient.update( `/api/v1/user/${humhubUserId}`, user, options, ) if (response.statusCode === 400) { throw new LogError('Invalid user supplied', { user, response }) } else if (response.statusCode === 404) { throw new LogError('User not found', { user, response }) } return response.result } } // new RestClient('gradido', 'api/v1/')