* fix: remove old metadata seed files * feat: wip standard to core relation * fix: lint * fix: merge * fix: remove debug files * feat: add feature flag for core object metadata * fix: remove debug * feat: always disable the standard core relation * fix: missing feature flag * fix: remove debug * fix: feature flag doesn't seems to disable relation * fix: delete .vscode folder, change this in another PR * Update packages/twenty-server/src/workspace/workspace-sync-metadata/reflective-metadata.factory.ts Co-authored-by: Weiko <corentin@twenty.com> * Update packages/twenty-server/src/workspace/workspace-sync-metadata/reflective-metadata.factory.ts Co-authored-by: Weiko <corentin@twenty.com> * Update packages/twenty-server/src/workspace/workspace-sync-metadata/workspace-sync.metadata.service.ts Co-authored-by: Weiko <corentin@twenty.com> * fix: remove optional fields from metadata entities * fix: renamed variable * fix: put back CursorScalarType * fix: delete test command * fix: remove unused workspace standard migration command * fix: drop core object metadata declaration * fix: rename variable * fix: drop creation of core datasource * fix: remove feature flag * fix: drop support of standard to core relations * feat: add user email field on workspace-member standard object * fix: update seed accordingly * fix: missing remove command file * fix: datasource label should remain nullable * fix: better asserts * Remove unused code * Remove unused code --------- Co-authored-by: Weiko <corentin@twenty.com> Co-authored-by: Charles Bochet <charles@twenty.com>
118 lines
3.4 KiB
TypeScript
118 lines
3.4 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-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 { UserService } from './services/user.service';
|
|
|
|
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(
|
|
private readonly userService: UserService,
|
|
private readonly environmentService: EnvironmentService,
|
|
private readonly fileUploadService: FileUploadService,
|
|
) {}
|
|
|
|
@Query(() => User)
|
|
async currentUser(@AuthUser() { id }: User) {
|
|
const user = await this.userService.findById(id, {
|
|
relations: [{ name: 'defaultWorkspace', query: {} }],
|
|
});
|
|
|
|
assert(user, 'User not found');
|
|
|
|
return user;
|
|
}
|
|
|
|
@ResolveField(() => WorkspaceMember, {
|
|
nullable: false,
|
|
})
|
|
async workspaceMember(@Parent() user: User): Promise<WorkspaceMember> {
|
|
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);
|
|
}
|
|
}
|