mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-04 08:38:34 +00:00
#### PR Dependency Tree * **PR #13434** 👈 This tree was auto-generated by [Charcoal](https://github.com/danerwilliams/charcoal) <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **Style** * Improved clarity of TypeScript error suppression comments across various test files and helper scripts. Comments now specify the reasons for ignoring specific type errors, enhancing code readability for developers. * **Chores** * Updated inline comments without affecting application functionality or user experience. No changes to features, logic, or test outcomes. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
1267 lines
33 KiB
TypeScript
1267 lines
33 KiB
TypeScript
import { randomUUID } from 'node:crypto';
|
|
|
|
import { ProjectRoot } from '@affine-tools/utils/path';
|
|
import { PrismaClient } from '@prisma/client';
|
|
import type { TestFn } from 'ava';
|
|
import ava from 'ava';
|
|
import Sinon from 'sinon';
|
|
|
|
import { AppModule } from '../app.module';
|
|
import { JobQueue } from '../base';
|
|
import { ConfigModule } from '../base/config';
|
|
import { AuthService } from '../core/auth';
|
|
import { DocReader } from '../core/doc';
|
|
import { CopilotContextService } from '../plugins/copilot/context';
|
|
import {
|
|
CopilotEmbeddingJob,
|
|
MockEmbeddingClient,
|
|
} from '../plugins/copilot/embedding';
|
|
import { prompts, PromptService } from '../plugins/copilot/prompt';
|
|
import {
|
|
CopilotProviderFactory,
|
|
CopilotProviderType,
|
|
GeminiGenerativeProvider,
|
|
OpenAIProvider,
|
|
} from '../plugins/copilot/providers';
|
|
import { CopilotStorage } from '../plugins/copilot/storage';
|
|
import { MockCopilotProvider } from './mocks';
|
|
import {
|
|
acceptInviteById,
|
|
createTestingApp,
|
|
createWorkspace,
|
|
inviteUser,
|
|
TestingApp,
|
|
TestUser,
|
|
} from './utils';
|
|
import {
|
|
addContextDoc,
|
|
addContextFile,
|
|
array2sse,
|
|
audioTranscription,
|
|
chatWithImages,
|
|
chatWithStreamObject,
|
|
chatWithText,
|
|
chatWithTextStream,
|
|
chatWithWorkflow,
|
|
claimAudioTranscription,
|
|
cleanObject,
|
|
createCopilotContext,
|
|
createCopilotMessage,
|
|
createCopilotSession,
|
|
createDocCopilotSession,
|
|
createPinnedCopilotSession,
|
|
createWorkspaceCopilotSession,
|
|
forkCopilotSession,
|
|
getCopilotSession,
|
|
getDocSessions,
|
|
getHistories,
|
|
getPinnedSessions,
|
|
getWorkspaceSessions,
|
|
listContext,
|
|
listContextDocAndFiles,
|
|
matchFiles,
|
|
matchWorkspaceDocs,
|
|
sse2array,
|
|
submitAudioTranscription,
|
|
textToEventStream,
|
|
unsplashSearch,
|
|
updateCopilotSession,
|
|
} from './utils/copilot';
|
|
|
|
const test = ava as TestFn<{
|
|
auth: AuthService;
|
|
app: TestingApp;
|
|
db: PrismaClient;
|
|
context: CopilotContextService;
|
|
jobs: CopilotEmbeddingJob;
|
|
prompt: PromptService;
|
|
factory: CopilotProviderFactory;
|
|
storage: CopilotStorage;
|
|
u1: TestUser;
|
|
}>;
|
|
|
|
test.before(async t => {
|
|
const app = await createTestingApp({
|
|
imports: [
|
|
ConfigModule.override({
|
|
copilot: {
|
|
providers: {
|
|
openai: { apiKey: '1' },
|
|
fal: {},
|
|
gemini: { apiKey: '1' },
|
|
},
|
|
unsplash: {
|
|
key: process.env.UNSPLASH_ACCESS_KEY || '1',
|
|
},
|
|
},
|
|
}),
|
|
AppModule,
|
|
],
|
|
tapModule: m => {
|
|
// use real JobQueue for testing
|
|
m.overrideProvider(JobQueue).useClass(JobQueue);
|
|
m.overrideProvider(DocReader).useValue({
|
|
getFullDocContent() {
|
|
return {
|
|
title: '1',
|
|
summary: '1',
|
|
};
|
|
},
|
|
});
|
|
m.overrideProvider(OpenAIProvider).useClass(MockCopilotProvider);
|
|
m.overrideProvider(GeminiGenerativeProvider).useClass(
|
|
class MockGenerativeProvider extends MockCopilotProvider {
|
|
// @ts-expect-error type not typed
|
|
override type: CopilotProviderType = CopilotProviderType.Gemini;
|
|
}
|
|
);
|
|
},
|
|
});
|
|
|
|
const auth = app.get(AuthService);
|
|
const db = app.get(PrismaClient);
|
|
const context = app.get(CopilotContextService);
|
|
const prompt = app.get(PromptService);
|
|
const storage = app.get(CopilotStorage);
|
|
const jobs = app.get(CopilotEmbeddingJob);
|
|
|
|
t.context.app = app;
|
|
t.context.db = db;
|
|
t.context.auth = auth;
|
|
t.context.context = context;
|
|
t.context.prompt = prompt;
|
|
t.context.storage = storage;
|
|
t.context.jobs = jobs;
|
|
});
|
|
|
|
let textPromptName = 'prompt';
|
|
let imagePromptName = 'prompt-image';
|
|
|
|
test.beforeEach(async t => {
|
|
Sinon.restore();
|
|
const { app, prompt } = t.context;
|
|
await prompt.onApplicationBootstrap();
|
|
t.context.u1 = await app.signupV1();
|
|
textPromptName = randomUUID().replaceAll('-', '');
|
|
imagePromptName = randomUUID().replaceAll('-', '');
|
|
|
|
await prompt.set(textPromptName, 'test', [
|
|
{ role: 'system', content: 'hello {{word}}' },
|
|
]);
|
|
|
|
await prompt.set(imagePromptName, 'test-image', [
|
|
{ role: 'system', content: 'hello {{word}}' },
|
|
]);
|
|
});
|
|
|
|
test.after.always(async t => {
|
|
await t.context.app.close();
|
|
});
|
|
|
|
// ==================== session ====================
|
|
|
|
test('should create session correctly', async t => {
|
|
const { app, u1 } = t.context;
|
|
|
|
const assertCreateSession = async (
|
|
workspaceId: string,
|
|
error: string,
|
|
asserter = async (x: any) => {
|
|
t.truthy(await x, error);
|
|
}
|
|
) => {
|
|
await asserter(
|
|
createCopilotSession(app, workspaceId, randomUUID(), textPromptName)
|
|
);
|
|
};
|
|
|
|
{
|
|
const { id } = await createWorkspace(app);
|
|
await assertCreateSession(
|
|
id,
|
|
'should be able to create session with cloud workspace that user can access'
|
|
);
|
|
}
|
|
|
|
{
|
|
await assertCreateSession(
|
|
randomUUID(),
|
|
'should be able to create session with local workspace'
|
|
);
|
|
}
|
|
|
|
{
|
|
const u2 = await app.createUser();
|
|
const { id } = await createWorkspace(app);
|
|
await app.login(u2);
|
|
await assertCreateSession(id, '', async x => {
|
|
await t.throwsAsync(
|
|
x,
|
|
{ instanceOf: Error },
|
|
'should not able to create session with cloud workspace that user cannot access'
|
|
);
|
|
});
|
|
|
|
await app.switchUser(u1);
|
|
const inviteId = await inviteUser(app, id, u2.email);
|
|
await app.login(u2);
|
|
await acceptInviteById(app, id, inviteId, false);
|
|
await assertCreateSession(
|
|
id,
|
|
'should able to create session after user have permission'
|
|
);
|
|
}
|
|
});
|
|
|
|
test('should update session correctly', async t => {
|
|
const { app } = t.context;
|
|
|
|
const assertUpdateSession = async (
|
|
sessionId: string,
|
|
error: string,
|
|
asserter = async (x: any) => {
|
|
t.truthy(await x, error);
|
|
}
|
|
) => {
|
|
await asserter(updateCopilotSession(app, sessionId, textPromptName));
|
|
};
|
|
|
|
{
|
|
const { id: workspaceId } = await createWorkspace(app);
|
|
const docId = randomUUID();
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
workspaceId,
|
|
docId,
|
|
textPromptName
|
|
);
|
|
await assertUpdateSession(
|
|
sessionId,
|
|
'should be able to update session with cloud workspace that user can access'
|
|
);
|
|
}
|
|
|
|
{
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
randomUUID(),
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
await assertUpdateSession(
|
|
sessionId,
|
|
'should be able to update session with local workspace'
|
|
);
|
|
}
|
|
|
|
{
|
|
await app.signupV1();
|
|
const u2 = await app.createUser();
|
|
const { id: workspaceId } = await createWorkspace(app);
|
|
const inviteId = await inviteUser(app, workspaceId, u2.email);
|
|
await app.login(u2);
|
|
await acceptInviteById(app, workspaceId, inviteId, false);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
workspaceId,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
await assertUpdateSession(
|
|
sessionId,
|
|
'should able to update session after user have permission'
|
|
);
|
|
}
|
|
|
|
{
|
|
const sessionId = '123456';
|
|
await assertUpdateSession(sessionId, '', async x => {
|
|
await t.throwsAsync(
|
|
x,
|
|
{ instanceOf: Error },
|
|
'should not able to update invalid session id'
|
|
);
|
|
});
|
|
}
|
|
});
|
|
|
|
test('should fork session correctly', async t => {
|
|
const { app, u1 } = t.context;
|
|
|
|
const assertForkSession = async (
|
|
workspaceId: string,
|
|
docId: string,
|
|
sessionId: string,
|
|
lastMessageId: string | undefined,
|
|
error: string,
|
|
asserter = async (x: any) => {
|
|
const forkedSessionId = await x;
|
|
t.truthy(forkedSessionId, error);
|
|
return forkedSessionId;
|
|
}
|
|
) =>
|
|
await asserter(
|
|
forkCopilotSession(app, workspaceId, docId, sessionId, lastMessageId)
|
|
);
|
|
|
|
// prepare session
|
|
const { id } = await createWorkspace(app);
|
|
const docId = randomUUID();
|
|
const sessionId = await createCopilotSession(app, id, docId, textPromptName);
|
|
|
|
let forkedSessionId: string;
|
|
// should be able to fork session
|
|
{
|
|
for (let i = 0; i < 3; i++) {
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
await chatWithText(app, sessionId, messageId);
|
|
}
|
|
const histories = await getHistories(app, { workspaceId: id, docId });
|
|
const latestMessageId = histories[0].messages.findLast(
|
|
m => m.role === 'assistant'
|
|
)?.id;
|
|
t.truthy(latestMessageId, 'should find last message id');
|
|
|
|
// should be able to fork session
|
|
forkedSessionId = await assertForkSession(
|
|
id,
|
|
docId,
|
|
sessionId,
|
|
latestMessageId!,
|
|
'should be able to fork session with cloud workspace that user can access'
|
|
);
|
|
}
|
|
|
|
// should be able to fork session without latestMessageId (copy all messages)
|
|
{
|
|
forkedSessionId = await assertForkSession(
|
|
id,
|
|
docId,
|
|
sessionId,
|
|
undefined,
|
|
'should be able to fork session without latestMessageId'
|
|
);
|
|
}
|
|
|
|
// should not be able to fork session with wrong latestMessageId
|
|
{
|
|
await assertForkSession(
|
|
id,
|
|
docId,
|
|
sessionId,
|
|
'wrong-message-id',
|
|
'',
|
|
async x => {
|
|
await t.throwsAsync(
|
|
x,
|
|
{ instanceOf: Error },
|
|
'should not able to fork session with wrong latestMessageId'
|
|
);
|
|
}
|
|
);
|
|
}
|
|
|
|
{
|
|
const u2 = await app.signupV1();
|
|
await assertForkSession(id, docId, sessionId, randomUUID(), '', async x => {
|
|
await t.throwsAsync(
|
|
x,
|
|
{ instanceOf: Error },
|
|
'should not able to fork session with cloud workspace that user cannot access'
|
|
);
|
|
});
|
|
|
|
await app.switchUser(u1);
|
|
const inviteId = await inviteUser(app, id, u2.email);
|
|
await app.switchUser(u2);
|
|
await acceptInviteById(app, id, inviteId, false);
|
|
await assertForkSession(id, docId, sessionId, randomUUID(), '', async x => {
|
|
await t.throwsAsync(
|
|
x,
|
|
{ instanceOf: Error },
|
|
'should not able to fork a root session from other user'
|
|
);
|
|
});
|
|
|
|
await app.switchUser(u1);
|
|
const histories = await getHistories(app, { workspaceId: id, docId });
|
|
const latestMessageId = histories
|
|
.find(h => h.sessionId === forkedSessionId)
|
|
?.messages.findLast(m => m.role === 'assistant')?.id;
|
|
t.truthy(latestMessageId, 'should find latest message id');
|
|
|
|
await app.switchUser(u2);
|
|
await assertForkSession(
|
|
id,
|
|
docId,
|
|
forkedSessionId,
|
|
latestMessageId!,
|
|
'should able to fork a forked session created by other user'
|
|
);
|
|
}
|
|
});
|
|
|
|
test('should be able to use test provider', async t => {
|
|
const { app } = t.context;
|
|
|
|
const { id } = await createWorkspace(app);
|
|
t.truthy(
|
|
await createCopilotSession(app, id, randomUUID(), textPromptName),
|
|
'failed to create session'
|
|
);
|
|
});
|
|
|
|
// ==================== message ====================
|
|
|
|
test('should create message correctly', async t => {
|
|
const { app } = t.context;
|
|
|
|
{
|
|
const { id } = await createWorkspace(app);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
t.truthy(messageId, 'should be able to create message with valid session');
|
|
}
|
|
|
|
{
|
|
// with attachment url
|
|
{
|
|
const { id } = await createWorkspace(app);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId, undefined, [
|
|
'http://example.com/cat.jpg',
|
|
]);
|
|
t.truthy(messageId, 'should be able to create message with url link');
|
|
}
|
|
|
|
// with attachment
|
|
{
|
|
const { id } = await createWorkspace(app);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const smallestPng =
|
|
'';
|
|
const pngData = await fetch(smallestPng).then(res => res.arrayBuffer());
|
|
const messageId = await createCopilotMessage(
|
|
app,
|
|
sessionId,
|
|
undefined,
|
|
undefined,
|
|
new File([new Uint8Array(pngData)], '1.png', { type: 'image/png' })
|
|
);
|
|
t.truthy(messageId, 'should be able to create message with blob');
|
|
}
|
|
|
|
// with attachments
|
|
{
|
|
const { id } = await createWorkspace(app);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const smallestPng =
|
|
'';
|
|
const pngData = await fetch(smallestPng).then(res => res.arrayBuffer());
|
|
const messageId = await createCopilotMessage(
|
|
app,
|
|
sessionId,
|
|
undefined,
|
|
undefined,
|
|
undefined,
|
|
[new File([new Uint8Array(pngData)], '1.png', { type: 'image/png' })]
|
|
);
|
|
t.truthy(messageId, 'should be able to create message with blobs');
|
|
}
|
|
}
|
|
|
|
{
|
|
await t.throwsAsync(
|
|
createCopilotMessage(app, randomUUID()),
|
|
{ instanceOf: Error },
|
|
'should not able to create message with invalid session'
|
|
);
|
|
}
|
|
});
|
|
|
|
// ==================== chat ====================
|
|
|
|
test('should be able to chat with api', async t => {
|
|
const { app, storage } = t.context;
|
|
|
|
Sinon.stub(storage, 'handleRemoteLink').resolvesArg(2);
|
|
|
|
const { id } = await createWorkspace(app);
|
|
{
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
const ret = await chatWithText(app, sessionId, messageId);
|
|
t.is(ret, 'generate text to text', 'should be able to chat with text');
|
|
|
|
const ret2 = await chatWithTextStream(app, sessionId, messageId);
|
|
t.is(
|
|
ret2,
|
|
textToEventStream('generate text to text stream', messageId),
|
|
'should be able to chat with text stream'
|
|
);
|
|
}
|
|
|
|
{
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
imagePromptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
const ret3 = await chatWithImages(app, sessionId, messageId);
|
|
t.is(
|
|
array2sse(sse2array(ret3).filter(e => e.event !== 'event')),
|
|
textToEventStream(
|
|
['https://example.com/test-image.jpg', 'hello '],
|
|
messageId,
|
|
'attachment'
|
|
),
|
|
'should be able to chat with images'
|
|
);
|
|
}
|
|
|
|
{
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
|
|
const ret4 = await chatWithStreamObject(app, sessionId, messageId);
|
|
|
|
const objects = Array.from('generate text to object stream').map(data =>
|
|
JSON.stringify({ type: 'text-delta', textDelta: data })
|
|
);
|
|
|
|
t.is(
|
|
ret4,
|
|
textToEventStream(objects, messageId),
|
|
'should be able to chat with stream object'
|
|
);
|
|
}
|
|
|
|
Sinon.restore();
|
|
});
|
|
|
|
test('should be able to chat with api by workflow', async t => {
|
|
const { app } = t.context;
|
|
|
|
const { id } = await createWorkspace(app);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
'workflow:presentation'
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId, 'apple company');
|
|
const ret = await chatWithWorkflow(app, sessionId, messageId);
|
|
t.is(
|
|
array2sse(sse2array(ret).filter(e => e.event !== 'event')),
|
|
textToEventStream(['generate text to text stream'], messageId),
|
|
'should be able to chat with workflow'
|
|
);
|
|
});
|
|
|
|
test('should be able to chat with special image model', async t => {
|
|
const { app, storage } = t.context;
|
|
|
|
Sinon.stub(storage, 'handleRemoteLink').resolvesArg(2);
|
|
|
|
const { id } = await createWorkspace(app);
|
|
|
|
const testWithModel = async (promptName: string, finalPrompt: string) => {
|
|
const model = prompts.find(p => p.name === promptName)?.model;
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
promptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId, 'some-tag', [
|
|
`https://example.com/${promptName}.jpg`,
|
|
]);
|
|
const ret3 = await chatWithImages(app, sessionId, messageId);
|
|
t.is(
|
|
ret3,
|
|
textToEventStream(
|
|
[`https://example.com/${model}.jpg`, finalPrompt],
|
|
messageId,
|
|
'attachment'
|
|
),
|
|
'should be able to chat with images'
|
|
);
|
|
};
|
|
|
|
await testWithModel('Generate image', 'some-tag');
|
|
await testWithModel(
|
|
'Convert to sticker',
|
|
'convert this image to sticker. you need to identify the subject matter and warp a circle of white stroke around the subject matter and with transparent background. some-tag'
|
|
);
|
|
await testWithModel(
|
|
'Upscale image',
|
|
'make the image more detailed. some-tag'
|
|
);
|
|
await testWithModel(
|
|
'Remove background',
|
|
'Keep the subject and remove other non-subject items. Transparent background. some-tag'
|
|
);
|
|
|
|
Sinon.restore();
|
|
});
|
|
|
|
test('should be able to retry with api', async t => {
|
|
const { app, storage } = t.context;
|
|
|
|
Sinon.stub(storage, 'handleRemoteLink').resolvesArg(2);
|
|
|
|
// normal chat
|
|
{
|
|
const { id } = await createWorkspace(app);
|
|
const docId = randomUUID();
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
docId,
|
|
textPromptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
// chat 2 times
|
|
await chatWithText(app, sessionId, messageId);
|
|
await chatWithText(app, sessionId, messageId);
|
|
|
|
const histories = await getHistories(app, { workspaceId: id, docId });
|
|
t.deepEqual(
|
|
histories.map(h => h.messages.map(m => m.content)),
|
|
[['generate text to text', 'generate text to text']],
|
|
'should be able to list history'
|
|
);
|
|
}
|
|
|
|
// retry chat
|
|
{
|
|
const { id } = await createWorkspace(app);
|
|
const docId = randomUUID();
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
docId,
|
|
textPromptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
await chatWithText(app, sessionId, messageId);
|
|
// retry without message id
|
|
await chatWithText(app, sessionId);
|
|
|
|
// should only have 1 message
|
|
const histories = await getHistories(app, { workspaceId: id, docId });
|
|
t.snapshot(
|
|
cleanObject(histories),
|
|
'should be able to list history after retry'
|
|
);
|
|
}
|
|
|
|
// retry chat with new message id
|
|
{
|
|
const { id } = await createWorkspace(app);
|
|
const docId = randomUUID();
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
docId,
|
|
textPromptName
|
|
);
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
await chatWithText(app, sessionId, messageId);
|
|
// retry with new message id
|
|
const newMessageId = await createCopilotMessage(app, sessionId);
|
|
await chatWithText(app, sessionId, newMessageId, '', true);
|
|
|
|
// should only have 1 message
|
|
const histories = await getHistories(app, { workspaceId: id, docId });
|
|
t.snapshot(
|
|
cleanObject(histories),
|
|
'should be able to list history after retry'
|
|
);
|
|
}
|
|
|
|
Sinon.restore();
|
|
});
|
|
|
|
test('should reject message from different session', async t => {
|
|
const { app } = t.context;
|
|
|
|
const { id } = await createWorkspace(app);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const anotherSessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const anotherMessageId = await createCopilotMessage(app, anotherSessionId);
|
|
await t.throwsAsync(
|
|
chatWithText(app, sessionId, anotherMessageId),
|
|
{ instanceOf: Error },
|
|
'should reject message from different session'
|
|
);
|
|
});
|
|
|
|
test('should reject request from different user', async t => {
|
|
const { app, u1 } = t.context;
|
|
|
|
const u2 = await app.createUser();
|
|
const { id } = await createWorkspace(app);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
id,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
|
|
// should reject message from different user
|
|
{
|
|
await app.login(u2);
|
|
await t.throwsAsync(
|
|
createCopilotMessage(app, sessionId),
|
|
{ instanceOf: Error },
|
|
'should reject message from different user'
|
|
);
|
|
}
|
|
|
|
// should reject chat from different user
|
|
{
|
|
await app.switchUser(u1);
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
{
|
|
await app.switchUser(u2);
|
|
await t.throwsAsync(
|
|
chatWithText(app, sessionId, messageId),
|
|
{ instanceOf: Error },
|
|
'should reject chat from different user'
|
|
);
|
|
}
|
|
}
|
|
});
|
|
|
|
// ==================== history ====================
|
|
|
|
test('should be able to list history', async t => {
|
|
const { app } = t.context;
|
|
|
|
const { id: workspaceId } = await createWorkspace(app);
|
|
const docId = randomUUID();
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
workspaceId,
|
|
docId,
|
|
textPromptName
|
|
);
|
|
|
|
const messageId = await createCopilotMessage(app, sessionId, 'hello');
|
|
await chatWithText(app, sessionId, messageId);
|
|
|
|
{
|
|
const histories = await getHistories(app, { workspaceId, docId });
|
|
t.deepEqual(
|
|
histories.map(h => h.messages.map(m => m.content)),
|
|
[['hello', 'generate text to text']],
|
|
'should be able to list history'
|
|
);
|
|
}
|
|
|
|
{
|
|
const histories = await getHistories(app, {
|
|
workspaceId,
|
|
docId,
|
|
options: { messageOrder: 'desc' },
|
|
});
|
|
t.deepEqual(
|
|
histories.map(h => h.messages.map(m => m.content)),
|
|
[['generate text to text', 'hello']],
|
|
'should be able to list history'
|
|
);
|
|
}
|
|
});
|
|
|
|
test('should reject request that user have not permission', async t => {
|
|
const { app, u1 } = t.context;
|
|
|
|
const u2 = await app.createUser();
|
|
const { id: workspaceId } = await createWorkspace(app);
|
|
|
|
// should reject request that user have not permission
|
|
{
|
|
await app.login(u2);
|
|
await t.throwsAsync(
|
|
getHistories(app, { workspaceId }),
|
|
{ instanceOf: Error },
|
|
'should reject request that user have not permission'
|
|
);
|
|
}
|
|
|
|
// should able to list history after user have permission
|
|
{
|
|
await app.switchUser(u1);
|
|
const inviteId = await inviteUser(app, workspaceId, u2.email);
|
|
await app.switchUser(u2);
|
|
await acceptInviteById(app, workspaceId, inviteId, false);
|
|
|
|
t.deepEqual(
|
|
await getHistories(app, { workspaceId }),
|
|
[],
|
|
'should able to list history after user have permission'
|
|
);
|
|
}
|
|
|
|
{
|
|
const docId = randomUUID();
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
workspaceId,
|
|
docId,
|
|
textPromptName
|
|
);
|
|
|
|
const messageId = await createCopilotMessage(app, sessionId);
|
|
await chatWithText(app, sessionId, messageId);
|
|
|
|
const histories = await getHistories(app, { workspaceId, docId });
|
|
t.deepEqual(
|
|
histories.map(h => h.messages.map(m => m.content)),
|
|
[['generate text to text']],
|
|
'should able to list history'
|
|
);
|
|
|
|
await app.switchUser(u1);
|
|
t.deepEqual(
|
|
await getHistories(app, { workspaceId }),
|
|
[],
|
|
'should not list history created by another user'
|
|
);
|
|
}
|
|
});
|
|
|
|
test('should be able to search image from unsplash', async t => {
|
|
const { app } = t.context;
|
|
|
|
const resp = await unsplashSearch(app);
|
|
t.not(resp.status, 404, 'route should be exists');
|
|
});
|
|
|
|
test('should be able to manage context', async t => {
|
|
const { app, context, jobs } = t.context;
|
|
|
|
const { id: workspaceId } = await createWorkspace(app);
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
workspaceId,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
|
|
// use mocked embedding client
|
|
Sinon.stub(context, 'embeddingClient').get(() => new MockEmbeddingClient());
|
|
Sinon.stub(jobs, 'embeddingClient').get(() => new MockEmbeddingClient());
|
|
|
|
{
|
|
await t.throwsAsync(
|
|
createCopilotContext(app, workspaceId, randomUUID()),
|
|
{ instanceOf: Error },
|
|
'should throw error if create context with invalid session id'
|
|
);
|
|
|
|
const context = await createCopilotContext(app, workspaceId, sessionId);
|
|
|
|
const list = await listContext(app, workspaceId, sessionId);
|
|
t.deepEqual(
|
|
list.map(f => ({ id: f.id })),
|
|
[{ id: context }],
|
|
'should list context'
|
|
);
|
|
}
|
|
|
|
const fs = await import('node:fs');
|
|
const buffer = fs.readFileSync(
|
|
ProjectRoot.join('packages/common/native/fixtures/sample.pdf').toFileUrl()
|
|
);
|
|
|
|
// match files
|
|
{
|
|
const contextId = await createCopilotContext(app, workspaceId, sessionId);
|
|
|
|
const { id: fileId } = await addContextFile(
|
|
app,
|
|
contextId,
|
|
'fileId1',
|
|
'sample.pdf',
|
|
buffer
|
|
);
|
|
|
|
const { files } =
|
|
(await listContextDocAndFiles(app, workspaceId, sessionId, contextId)) ||
|
|
{};
|
|
t.snapshot(
|
|
cleanObject(files, ['id', 'error', 'createdAt']),
|
|
'should list context files'
|
|
);
|
|
|
|
// wait for processing
|
|
{
|
|
let { files } =
|
|
(await listContextDocAndFiles(
|
|
app,
|
|
workspaceId,
|
|
sessionId,
|
|
contextId
|
|
)) || {};
|
|
|
|
while (files?.[0].status !== 'finished') {
|
|
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
({ files } =
|
|
(await listContextDocAndFiles(
|
|
app,
|
|
workspaceId,
|
|
sessionId,
|
|
contextId
|
|
)) || {});
|
|
}
|
|
}
|
|
|
|
const result = (await matchFiles(app, contextId, 'test', 1))!;
|
|
t.is(result.length, 1, 'should match context');
|
|
t.is(result[0].fileId, fileId, 'should match file id');
|
|
}
|
|
|
|
// match docs
|
|
{
|
|
const sessionId = await createCopilotSession(
|
|
app,
|
|
workspaceId,
|
|
randomUUID(),
|
|
textPromptName
|
|
);
|
|
const contextId = await createCopilotContext(app, workspaceId, sessionId);
|
|
|
|
const docId = 'docId1';
|
|
await t.context.db.snapshot.create({
|
|
data: {
|
|
workspaceId: workspaceId,
|
|
id: docId,
|
|
blob: Buffer.from([1, 1]),
|
|
state: Buffer.from([1, 1]),
|
|
updatedAt: new Date(),
|
|
createdAt: new Date(),
|
|
},
|
|
});
|
|
|
|
await addContextDoc(app, contextId, docId);
|
|
|
|
const { docs } =
|
|
(await listContextDocAndFiles(app, workspaceId, sessionId, contextId)) ||
|
|
{};
|
|
t.snapshot(
|
|
cleanObject(docs, ['error', 'createdAt']),
|
|
'should list context docs'
|
|
);
|
|
|
|
// wait for processing
|
|
{
|
|
let { docs } =
|
|
(await listContextDocAndFiles(
|
|
app,
|
|
workspaceId,
|
|
sessionId,
|
|
contextId
|
|
)) || {};
|
|
|
|
while (docs?.[0].status !== 'finished') {
|
|
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
({ docs } =
|
|
(await listContextDocAndFiles(
|
|
app,
|
|
workspaceId,
|
|
sessionId,
|
|
contextId
|
|
)) || {});
|
|
}
|
|
}
|
|
|
|
const result = (await matchWorkspaceDocs(app, contextId, 'test', 1))!;
|
|
t.is(result.length, 1, 'should match context');
|
|
t.is(result[0].docId, docId, 'should match doc id');
|
|
}
|
|
});
|
|
|
|
test('should be able to transcript', async t => {
|
|
const { app } = t.context;
|
|
|
|
const { id: workspaceId } = await createWorkspace(app);
|
|
|
|
for (const [provider, func] of [
|
|
[GeminiGenerativeProvider, 'text'],
|
|
[GeminiGenerativeProvider, 'structure'],
|
|
] as const) {
|
|
Sinon.stub(app.get(provider), func).resolves(
|
|
JSON.stringify([
|
|
{ a: 'A', s: 30, e: 45, t: 'Hello, everyone.' },
|
|
{
|
|
a: 'B',
|
|
s: 46,
|
|
e: 70,
|
|
t: 'Hi, thank you for joining the meeting today.',
|
|
},
|
|
])
|
|
);
|
|
}
|
|
|
|
{
|
|
const job = await submitAudioTranscription(app, workspaceId, '1', '1.mp3', [
|
|
Buffer.from([1, 1]),
|
|
]);
|
|
t.snapshot(
|
|
cleanObject([job], ['id']),
|
|
'should submit audio transcription job'
|
|
);
|
|
t.truthy(job.id, 'should have job id');
|
|
|
|
// wait for processing
|
|
{
|
|
let { status } =
|
|
(await audioTranscription(app, workspaceId, job.id)) || {};
|
|
|
|
while (status !== 'finished') {
|
|
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
({ status } =
|
|
(await audioTranscription(app, workspaceId, job.id)) || {});
|
|
}
|
|
}
|
|
|
|
{
|
|
const result = await claimAudioTranscription(app, job.id);
|
|
t.snapshot(
|
|
cleanObject([result], ['id']),
|
|
'should claim audio transcription job'
|
|
);
|
|
}
|
|
}
|
|
|
|
{
|
|
// sliced audio
|
|
const job = await submitAudioTranscription(app, workspaceId, '2', '2.mp3', [
|
|
Buffer.from([1, 1]),
|
|
Buffer.from([1, 2]),
|
|
]);
|
|
t.snapshot(
|
|
cleanObject([job], ['id']),
|
|
'should submit audio transcription job'
|
|
);
|
|
t.truthy(job.id, 'should have job id');
|
|
|
|
// wait for processing
|
|
{
|
|
let { status } =
|
|
(await audioTranscription(app, workspaceId, job.id)) || {};
|
|
|
|
while (status !== 'finished') {
|
|
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
({ status } =
|
|
(await audioTranscription(app, workspaceId, job.id)) || {});
|
|
}
|
|
}
|
|
|
|
{
|
|
const result = await claimAudioTranscription(app, job.id);
|
|
t.snapshot(
|
|
cleanObject([result], ['id']),
|
|
'should claim audio transcription job'
|
|
);
|
|
}
|
|
}
|
|
});
|
|
|
|
test('should create different session types and validate prompt constraints', async t => {
|
|
const { app } = t.context;
|
|
const { id: workspaceId } = await createWorkspace(app);
|
|
|
|
const validateSession = async (
|
|
description: string,
|
|
workspaceId: string,
|
|
createPromise: Promise<string>
|
|
) => {
|
|
const sessionId = await createPromise;
|
|
|
|
t.truthy(sessionId, description);
|
|
t.snapshot(
|
|
cleanObject(
|
|
[await getCopilotSession(app, workspaceId, sessionId)],
|
|
['id', 'workspaceId', 'promptName']
|
|
),
|
|
`should create session with ${description}`
|
|
);
|
|
return sessionId;
|
|
};
|
|
|
|
await validateSession(
|
|
'should create workspace session with text prompt',
|
|
workspaceId,
|
|
createWorkspaceCopilotSession(app, workspaceId, textPromptName)
|
|
);
|
|
await validateSession(
|
|
'should create pinned session with text prompt',
|
|
workspaceId,
|
|
createPinnedCopilotSession(app, workspaceId, 'pinned-doc', textPromptName)
|
|
);
|
|
await validateSession(
|
|
'should create doc session with text prompt',
|
|
workspaceId,
|
|
createDocCopilotSession(app, workspaceId, 'normal-doc', textPromptName)
|
|
);
|
|
});
|
|
|
|
test('should list histories for different session types correctly', async t => {
|
|
const { app } = t.context;
|
|
const { id: workspaceId } = await createWorkspace(app);
|
|
const pinnedDocId = 'pinned-doc';
|
|
const docId = 'normal-doc';
|
|
|
|
// create sessions and add messages
|
|
const [workspaceSessionId, pinnedSessionId, docSessionId] = await Promise.all(
|
|
[
|
|
createWorkspaceCopilotSession(app, workspaceId, textPromptName),
|
|
createPinnedCopilotSession(app, workspaceId, pinnedDocId, textPromptName),
|
|
createDocCopilotSession(app, workspaceId, docId, textPromptName),
|
|
]
|
|
);
|
|
|
|
await Promise.all([
|
|
createCopilotMessage(app, workspaceSessionId, 'workspace message'),
|
|
createCopilotMessage(app, pinnedSessionId, 'pinned message'),
|
|
createCopilotMessage(app, docSessionId, 'doc message'),
|
|
]);
|
|
|
|
const testHistoryQuery = async (
|
|
queryFn: () => Promise<any[]>,
|
|
opts: {
|
|
sessionIds?: string[];
|
|
sessionId?: string;
|
|
pinned?: boolean;
|
|
isEmpty?: boolean;
|
|
},
|
|
description: string
|
|
) => {
|
|
const s = await queryFn();
|
|
|
|
if (opts.isEmpty) {
|
|
t.is(s.length, 0, `should return ${description}`);
|
|
return;
|
|
}
|
|
|
|
if (opts.sessionIds) {
|
|
t.is(s.length, opts.sessionIds.length, `should return ${description}`);
|
|
const ids = s.map(h => h.sessionId).sort((a, b) => a.localeCompare(b));
|
|
const expectedIds = opts.sessionIds.sort((a, b) => a.localeCompare(b));
|
|
t.deepEqual(ids, expectedIds, `should return correct ${description}`);
|
|
} else if (opts.sessionId) {
|
|
t.is(s.length, 1, `should return ${description}`);
|
|
t.is(
|
|
s[0].sessionId,
|
|
opts.sessionId,
|
|
`should return correct ${description}`
|
|
);
|
|
if (opts.pinned !== undefined) {
|
|
t.is(s[0].pinned, opts.pinned, `pinned status for ${description}`);
|
|
}
|
|
}
|
|
};
|
|
|
|
// test for getHistories
|
|
await testHistoryQuery(
|
|
() => getHistories(app, { workspaceId, docId: null }),
|
|
{ sessionId: workspaceSessionId },
|
|
'workspace session history'
|
|
);
|
|
await testHistoryQuery(
|
|
() => getHistories(app, { workspaceId, docId: pinnedDocId }),
|
|
{ sessionId: pinnedSessionId },
|
|
'pinned session history'
|
|
);
|
|
await testHistoryQuery(
|
|
() => getHistories(app, { workspaceId, docId }),
|
|
{ sessionId: docSessionId },
|
|
'doc session history'
|
|
);
|
|
|
|
// test for getWorkspaceSessions
|
|
await testHistoryQuery(
|
|
() => getWorkspaceSessions(app, { workspaceId }),
|
|
{ sessionId: workspaceSessionId, pinned: false },
|
|
'workspace-level sessions'
|
|
);
|
|
|
|
// test for getDocSessions
|
|
await testHistoryQuery(
|
|
() =>
|
|
getDocSessions(app, { workspaceId, docId, options: { pinned: false } }),
|
|
{ sessionId: docSessionId, pinned: false },
|
|
'doc sessions'
|
|
);
|
|
|
|
await testHistoryQuery(
|
|
() => getDocSessions(app, { workspaceId, docId: pinnedDocId }),
|
|
{ sessionId: pinnedSessionId, pinned: true },
|
|
'pinned doc sessions'
|
|
);
|
|
|
|
// test for getPinnedSessions
|
|
await testHistoryQuery(
|
|
() => getPinnedSessions(app, { workspaceId }),
|
|
{ sessionId: pinnedSessionId, pinned: true },
|
|
'pinned sessions'
|
|
);
|
|
|
|
await testHistoryQuery(
|
|
() => getPinnedSessions(app, { workspaceId, docId: pinnedDocId }),
|
|
{ sessionId: pinnedSessionId, pinned: true },
|
|
'pinned session for specific doc'
|
|
);
|
|
|
|
await testHistoryQuery(
|
|
() => getPinnedSessions(app, { workspaceId, docId }),
|
|
{ isEmpty: true },
|
|
'no pinned sessions for non-pinned doc'
|
|
);
|
|
});
|