AFFiNE/apps/server/tests/app.e2e.ts

210 lines
4.7 KiB
TypeScript
Raw Normal View History

import { ok } from 'node:assert';
import { randomUUID } from 'node:crypto';
2023-04-19 02:14:25 +03:00
2023-06-21 09:08:32 +03:00
import { Transformer } from '@napi-rs/image';
import type { INestApplication } from '@nestjs/common';
2023-04-19 02:14:25 +03:00
import { Test } from '@nestjs/testing';
import { hashSync } from '@node-rs/argon2';
import { type User } from '@prisma/client';
import ava, { type TestFn } from 'ava';
import type { Express } from 'express';
2023-06-21 09:08:32 +03:00
import graphqlUploadExpress from 'graphql-upload/graphqlUploadExpress.mjs';
2023-04-19 02:14:25 +03:00
import request from 'supertest';
import { AppModule } from '../src/app';
import { PrismaService } from '../src/prisma/service';
2023-04-19 02:14:25 +03:00
const gql = '/graphql';
const test = ava as TestFn<{
app: INestApplication;
}>;
2023-04-19 02:14:25 +03:00
class FakePrisma {
fakeUser: User = {
id: randomUUID(),
name: 'Alex Yang',
avatarUrl: '',
email: 'alex.yang@example.org',
password: hashSync('123456'),
emailVerified: new Date(),
createdAt: new Date(),
};
get user() {
// eslint-disable-next-line @typescript-eslint/no-this-alias
const prisma = this;
return {
async findFirst() {
return prisma.fakeUser;
},
async findUnique() {
return this.findFirst();
},
async update() {
return this.findFirst();
},
};
}
}
2023-04-28 06:49:44 +03:00
test.beforeEach(async t => {
2023-09-01 22:41:29 +03:00
const module = await Test.createTestingModule({
imports: [AppModule],
})
.overrideProvider(PrismaService)
.useClass(FakePrisma)
.compile();
t.context.app = module.createNestApplication({
2023-09-01 22:41:29 +03:00
cors: true,
bodyParser: true,
2023-04-19 02:14:25 +03:00
});
t.context.app.use(
2023-09-01 22:41:29 +03:00
graphqlUploadExpress({
maxFileSize: 10 * 1024 * 1024,
maxFiles: 5,
})
);
await t.context.app.init();
2023-09-01 22:41:29 +03:00
});
2023-04-19 02:14:25 +03:00
test.afterEach.always(async t => {
await t.context.app.close();
2023-09-01 22:41:29 +03:00
});
2023-04-19 02:14:25 +03:00
test('should init app', async t => {
t.is(typeof t.context.app, 'object');
await request(t.context.app.getHttpServer())
2023-09-01 22:41:29 +03:00
.post(gql)
.send({
query: `
query {
error
}
`,
2023-09-01 22:41:29 +03:00
})
.expect(400);
const { token } = await createToken(t.context.app);
await request(t.context.app.getHttpServer())
2023-09-01 22:41:29 +03:00
.post(gql)
.auth(token, { type: 'bearer' })
.send({
query: `
query {
__typename
2023-04-19 02:14:25 +03:00
}
`,
2023-09-01 22:41:29 +03:00
})
.expect(200)
.expect(res => {
t.is(res.body.data.__typename, 'Query');
2023-09-01 22:41:29 +03:00
});
});
2023-04-19 02:14:25 +03:00
test('should find default user', async t => {
const { token } = await createToken(t.context.app);
await request(t.context.app.getHttpServer())
2023-09-01 22:41:29 +03:00
.post(gql)
.auth(token, { type: 'bearer' })
.send({
query: `
2023-04-19 02:14:25 +03:00
query {
user(email: "alex.yang@example.org") {
email
avatarUrl
2023-04-19 02:14:25 +03:00
}
}
`,
2023-09-01 22:41:29 +03:00
})
.expect(200)
.expect(res => {
t.is(res.body.data.user.email, 'alex.yang@example.org');
2023-09-01 22:41:29 +03:00
});
});
2023-06-21 09:08:32 +03:00
test('should be able to upload avatar and remove it', async t => {
const { token, id } = await createToken(t.context.app);
2023-09-01 22:41:29 +03:00
const png = await Transformer.fromRgbaPixels(
Buffer.alloc(400 * 400 * 4).fill(255),
400,
400
).png();
2023-06-21 09:08:32 +03:00
await request(t.context.app.getHttpServer())
2023-09-01 22:41:29 +03:00
.post(gql)
.auth(token, { type: 'bearer' })
.field(
'operations',
JSON.stringify({
name: 'uploadAvatar',
query: `mutation uploadAvatar($avatar: Upload!) {
uploadAvatar(avatar: $avatar) {
2023-06-21 09:08:32 +03:00
id
name
avatarUrl
email
}
}
`,
2023-09-01 22:41:29 +03:00
variables: { id, avatar: null },
})
)
.field('map', JSON.stringify({ '0': ['variables.avatar'] }))
.attach('0', png, 'avatar.png')
.expect(200)
.expect(res => {
t.is(res.body.data.uploadAvatar.id, id);
2023-09-01 22:41:29 +03:00
});
await request(t.context.app.getHttpServer())
.post(gql)
.auth(token, { type: 'bearer' })
.set({ 'x-request-id': 'test', 'x-operation-name': 'test' })
.send({
query: `
mutation removeAvatar {
removeAvatar {
success
}
}
`,
})
.expect(200)
.expect(res => {
t.is(res.body.data.removeAvatar.success, true);
});
2023-04-19 02:14:25 +03:00
});
2023-06-21 09:08:32 +03:00
async function createToken(app: INestApplication<Express>): Promise<{
id: string;
token: string;
}> {
let token;
let id;
await request(app.getHttpServer())
.post(gql)
.send({
query: `
mutation {
signIn(email: "alex.yang@example.org", password: "123456") {
id
token {
token
}
}
}
`,
})
.expect(200)
.expect(res => {
id = res.body.data.signIn.id;
ok(
typeof res.body.data.signIn.token.token === 'string',
'res.body.data.signIn.token.token is not a string'
);
token = res.body.data.signIn.token.token;
});
return { token: token!, id: id! };
}