* Seed UserWorkspace for existing demo/dev users * add workspaces field to currentUser * new token generation endpoint for switching workspace * lint fix * include dependency * requested fixes * resolver test pass * changing defaultWorkspace and workspaceMember when switching workspaces * tests fix * requested changes * delete user/workspace edge case handled * after merge * requested changes * :wq! * workspace manytoone relation * lint fix / import fix * gql codegen * Fix migrations and generateJWT * migration fix * relations fix --------- Co-authored-by: martmull <martmull@hotmail.fr>
129 lines
3.8 KiB
TypeScript
129 lines
3.8 KiB
TypeScript
import {
|
|
Resolver,
|
|
Query,
|
|
Args,
|
|
Parent,
|
|
ResolveField,
|
|
Mutation,
|
|
} from '@nestjs/graphql';
|
|
import { ForbiddenException, UseGuards } from '@nestjs/common';
|
|
|
|
import crypto from 'crypto';
|
|
|
|
import { FileUpload, GraphQLUpload } from 'graphql-upload';
|
|
|
|
import { SupportDriver } from 'src/integrations/environment/interfaces/support.interface';
|
|
import { FileFolder } from 'src/core/file/interfaces/file-folder.interface';
|
|
|
|
import { AuthUser } from 'src/decorators/auth/auth-user.decorator';
|
|
import { EnvironmentService } from 'src/integrations/environment/environment.service';
|
|
import { streamToBuffer } from 'src/utils/stream-to-buffer';
|
|
import { FileUploadService } from 'src/core/file/services/file-upload.service';
|
|
import { assert } from 'src/utils/assert';
|
|
import { JwtAuthGuard } from 'src/guards/jwt.auth.guard';
|
|
import { User } from 'src/core/user/user.entity';
|
|
import { WorkspaceMember } from 'src/core/user/dtos/workspace-member.dto';
|
|
import { UserWorkspaceService } from 'src/core/user-workspace/user-workspace.service';
|
|
|
|
import { UserService } from './services/user.service';
|
|
import { InjectRepository } from '@nestjs/typeorm';
|
|
import { Repository } from 'typeorm';
|
|
|
|
const getHMACKey = (email?: string, key?: string | null) => {
|
|
if (!email || !key) return null;
|
|
|
|
const hmac = crypto.createHmac('sha256', key);
|
|
|
|
return hmac.update(email).digest('hex');
|
|
};
|
|
|
|
@UseGuards(JwtAuthGuard)
|
|
@Resolver(() => User)
|
|
export class UserResolver {
|
|
constructor(
|
|
@InjectRepository(User, 'core')
|
|
private readonly userRepository: Repository<User>,
|
|
private readonly userService: UserService,
|
|
private readonly userWorkspaceService: UserWorkspaceService,
|
|
private readonly environmentService: EnvironmentService,
|
|
private readonly fileUploadService: FileUploadService,
|
|
) {}
|
|
|
|
@Query(() => User)
|
|
async currentUser(@AuthUser() { id }: User): Promise<User> {
|
|
const user = await this.userRepository.findOne({
|
|
where: {
|
|
id,
|
|
},
|
|
relations: ['defaultWorkspace', 'workspaces', 'workspaces.workspace'],
|
|
});
|
|
|
|
assert(user, 'User not found');
|
|
|
|
return user;
|
|
}
|
|
|
|
@ResolveField(() => WorkspaceMember, {
|
|
nullable: true,
|
|
})
|
|
async workspaceMember(
|
|
@Parent() user: User,
|
|
): Promise<WorkspaceMember | undefined> {
|
|
return this.userService.loadWorkspaceMember(user);
|
|
}
|
|
|
|
@ResolveField(() => String, {
|
|
nullable: true,
|
|
})
|
|
supportUserHash(@Parent() parent: User): string | null {
|
|
if (this.environmentService.getSupportDriver() !== SupportDriver.Front) {
|
|
return null;
|
|
}
|
|
const key = this.environmentService.getSupportFrontHMACKey();
|
|
|
|
return getHMACKey(parent.email, key);
|
|
}
|
|
|
|
@Mutation(() => String)
|
|
async uploadProfilePicture(
|
|
@AuthUser() { id }: User,
|
|
@Args({ name: 'file', type: () => GraphQLUpload })
|
|
{ createReadStream, filename, mimetype }: FileUpload,
|
|
): Promise<string> {
|
|
if (!id) {
|
|
throw new Error('User not found');
|
|
}
|
|
|
|
const stream = createReadStream();
|
|
const buffer = await streamToBuffer(stream);
|
|
const fileFolder = FileFolder.ProfilePicture;
|
|
|
|
const { paths } = await this.fileUploadService.uploadImage({
|
|
file: buffer,
|
|
filename,
|
|
mimeType: mimetype,
|
|
fileFolder,
|
|
});
|
|
|
|
return paths[0];
|
|
}
|
|
|
|
@Mutation(() => User)
|
|
async deleteUser(@AuthUser() { id: userId, defaultWorkspace }: User) {
|
|
// Get the list of demo workspace IDs
|
|
const demoWorkspaceIds = this.environmentService.getDemoWorkspaceIds();
|
|
|
|
const currentUserWorkspaceId = defaultWorkspace.id;
|
|
|
|
// Check if the user's default workspace ID is in the list of demo workspace IDs
|
|
if (demoWorkspaceIds.includes(currentUserWorkspaceId)) {
|
|
throw new ForbiddenException(
|
|
'Deletion of users with a default demo workspace is not allowed.',
|
|
);
|
|
}
|
|
|
|
// Proceed with user deletion
|
|
return this.userService.deleteUser(userId);
|
|
}
|
|
}
|