mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-18 06:47:02 +08:00
feat(core): add get session graphql api (#12237)
Close [AI-116](https://linear.app/affine-design/issue/AI-116) <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit - **New Features** - Added the ability to retrieve detailed information for a specific Copilot session by its ID, including model metadata and optional models, via the user interface and API. - Session data now includes additional fields such as the model used and a list of optional models. - Enhanced GraphQL queries and UI components to support fetching and displaying these new session details. - **Improvements** - Session lists now provide richer information, including model details, for each session. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
This commit is contained in:
@@ -41,6 +41,7 @@ import {
|
||||
AvailableModels,
|
||||
type ChatHistory,
|
||||
type ChatMessage,
|
||||
type ChatSessionState,
|
||||
type ListHistoriesOptions,
|
||||
SubmittedMessage,
|
||||
} from './types';
|
||||
@@ -277,7 +278,7 @@ class CopilotPromptType {
|
||||
}
|
||||
|
||||
@ObjectType()
|
||||
class CopilotSessionType {
|
||||
export class CopilotSessionType {
|
||||
@Field(() => ID)
|
||||
id!: string;
|
||||
|
||||
@@ -286,6 +287,12 @@ class CopilotSessionType {
|
||||
|
||||
@Field(() => String)
|
||||
promptName!: string;
|
||||
|
||||
@Field(() => String)
|
||||
model!: string;
|
||||
|
||||
@Field(() => [String])
|
||||
optionalModels!: string[];
|
||||
}
|
||||
|
||||
// ================== Resolver ==================
|
||||
@@ -329,6 +336,30 @@ export class CopilotResolver {
|
||||
return (await this.sessions(copilot, user, docId, options)).map(s => s.id);
|
||||
}
|
||||
|
||||
@ResolveField(() => CopilotSessionType, {
|
||||
description: 'Get the session by id',
|
||||
complexity: 2,
|
||||
})
|
||||
async session(
|
||||
@Parent() copilot: CopilotType,
|
||||
@CurrentUser() user: CurrentUser,
|
||||
@Args('sessionId') sessionId: string
|
||||
): Promise<CopilotSessionType> {
|
||||
if (!copilot.workspaceId) {
|
||||
throw new NotFoundException('Workspace not found');
|
||||
}
|
||||
await this.ac
|
||||
.user(user.id)
|
||||
.workspace(copilot.workspaceId)
|
||||
.allowLocal()
|
||||
.assert('Workspace.Copilot');
|
||||
const session = await this.chatSession.getSession(sessionId);
|
||||
if (!session) {
|
||||
throw new NotFoundException('Session not found');
|
||||
}
|
||||
return this.transformToSessionType(session);
|
||||
}
|
||||
|
||||
@ResolveField(() => [CopilotSessionType], {
|
||||
description: 'Get the session list in the workspace',
|
||||
complexity: 2,
|
||||
@@ -339,18 +370,21 @@ export class CopilotResolver {
|
||||
@Args('docId', { nullable: true }) docId?: string,
|
||||
@Args('options', { nullable: true }) options?: QueryChatSessionsInput
|
||||
): Promise<CopilotSessionType[]> {
|
||||
if (!copilot.workspaceId) return [];
|
||||
if (!copilot.workspaceId) {
|
||||
throw new NotFoundException('Workspace not found');
|
||||
}
|
||||
await this.ac
|
||||
.user(user.id)
|
||||
.workspace(copilot.workspaceId)
|
||||
.allowLocal()
|
||||
.assert('Workspace.Copilot');
|
||||
return await this.chatSession.listSessions(
|
||||
const sessions = await this.chatSession.listSessions(
|
||||
user.id,
|
||||
copilot.workspaceId,
|
||||
docId,
|
||||
options
|
||||
);
|
||||
return sessions.map(this.transformToSessionType);
|
||||
}
|
||||
|
||||
@ResolveField(() => [CopilotHistoriesType], {})
|
||||
@@ -556,6 +590,18 @@ export class CopilotResolver {
|
||||
throw new CopilotFailedToCreateMessage(e.message);
|
||||
}
|
||||
}
|
||||
|
||||
private transformToSessionType(
|
||||
session: Omit<ChatSessionState, 'messages'>
|
||||
): CopilotSessionType {
|
||||
return {
|
||||
id: session.sessionId,
|
||||
parentSessionId: session.parentSessionId,
|
||||
promptName: session.prompt.name,
|
||||
model: session.prompt.model,
|
||||
optionalModels: session.prompt.optionalModels,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
@Throttle()
|
||||
|
||||
@@ -307,9 +307,7 @@ export class ChatSessionService {
|
||||
});
|
||||
}
|
||||
|
||||
private async getSession(
|
||||
sessionId: string
|
||||
): Promise<ChatSessionState | undefined> {
|
||||
async getSession(sessionId: string): Promise<ChatSessionState | undefined> {
|
||||
return await this.db.aiSession
|
||||
.findUnique({
|
||||
where: { id: sessionId, deletedAt: null },
|
||||
@@ -414,13 +412,7 @@ export class ChatSessionService {
|
||||
workspaceId: string,
|
||||
docId?: string,
|
||||
options?: { action?: boolean }
|
||||
): Promise<
|
||||
Array<{
|
||||
id: string;
|
||||
parentSessionId: string | null;
|
||||
promptName: string;
|
||||
}>
|
||||
> {
|
||||
): Promise<Omit<ChatSessionState, 'messages'>[]> {
|
||||
return await this.db.aiSession
|
||||
.findMany({
|
||||
where: {
|
||||
@@ -434,17 +426,31 @@ export class ChatSessionService {
|
||||
},
|
||||
select: {
|
||||
id: true,
|
||||
userId: true,
|
||||
workspaceId: true,
|
||||
docId: true,
|
||||
parentSessionId: true,
|
||||
promptName: true,
|
||||
},
|
||||
})
|
||||
.then(sessions =>
|
||||
sessions.map(({ id, parentSessionId, promptName }) => ({
|
||||
id,
|
||||
parentSessionId: parentSessionId || null,
|
||||
promptName,
|
||||
}))
|
||||
);
|
||||
.then(sessions => {
|
||||
return Promise.all(
|
||||
sessions.map(async session => {
|
||||
const prompt = await this.prompt.get(session.promptName);
|
||||
if (!prompt)
|
||||
throw new CopilotPromptNotFound({ name: session.promptName });
|
||||
|
||||
return {
|
||||
sessionId: session.id,
|
||||
userId: session.userId,
|
||||
workspaceId: session.workspaceId,
|
||||
docId: session.docId,
|
||||
parentSessionId: session.parentSessionId,
|
||||
prompt,
|
||||
};
|
||||
})
|
||||
);
|
||||
});
|
||||
}
|
||||
|
||||
async listHistories(
|
||||
|
||||
@@ -133,6 +133,9 @@ type Copilot {
|
||||
"""Get the quota of the user in the workspace"""
|
||||
quota: CopilotQuota!
|
||||
|
||||
"""Get the session by id"""
|
||||
session(sessionId: String!): CopilotSessionType!
|
||||
|
||||
"""Get the session id list in the workspace"""
|
||||
sessionIds(docId: String, options: QueryChatSessionsInput): [String!]! @deprecated(reason: "Use `sessions` instead")
|
||||
|
||||
@@ -318,6 +321,8 @@ type CopilotQuota {
|
||||
|
||||
type CopilotSessionType {
|
||||
id: ID!
|
||||
model: String!
|
||||
optionalModels: [String!]!
|
||||
parentSessionId: ID
|
||||
promptName: String!
|
||||
}
|
||||
|
||||
16
packages/common/graphql/src/graphql/copilot-session-get.gql
Normal file
16
packages/common/graphql/src/graphql/copilot-session-get.gql
Normal file
@@ -0,0 +1,16 @@
|
||||
query getCopilotSession(
|
||||
$workspaceId: String!
|
||||
$sessionId: String!
|
||||
) {
|
||||
currentUser {
|
||||
copilot(workspaceId: $workspaceId) {
|
||||
session(sessionId: $sessionId) {
|
||||
id
|
||||
parentSessionId
|
||||
promptName
|
||||
model
|
||||
optionalModels
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -9,6 +9,8 @@ query getCopilotSessions(
|
||||
id
|
||||
parentSessionId
|
||||
promptName
|
||||
model
|
||||
optionalModels
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -740,6 +740,24 @@ export const forkCopilotSessionMutation = {
|
||||
}`,
|
||||
};
|
||||
|
||||
export const getCopilotSessionQuery = {
|
||||
id: 'getCopilotSessionQuery' as const,
|
||||
op: 'getCopilotSession',
|
||||
query: `query getCopilotSession($workspaceId: String!, $sessionId: String!) {
|
||||
currentUser {
|
||||
copilot(workspaceId: $workspaceId) {
|
||||
session(sessionId: $sessionId) {
|
||||
id
|
||||
parentSessionId
|
||||
promptName
|
||||
model
|
||||
optionalModels
|
||||
}
|
||||
}
|
||||
}
|
||||
}`,
|
||||
};
|
||||
|
||||
export const updateCopilotSessionMutation = {
|
||||
id: 'updateCopilotSessionMutation' as const,
|
||||
op: 'updateCopilotSession',
|
||||
@@ -758,6 +776,8 @@ export const getCopilotSessionsQuery = {
|
||||
id
|
||||
parentSessionId
|
||||
promptName
|
||||
model
|
||||
optionalModels
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -174,6 +174,8 @@ export interface Copilot {
|
||||
histories: Array<CopilotHistories>;
|
||||
/** Get the quota of the user in the workspace */
|
||||
quota: CopilotQuota;
|
||||
/** Get the session by id */
|
||||
session: CopilotSessionType;
|
||||
/**
|
||||
* Get the session id list in the workspace
|
||||
* @deprecated Use `sessions` instead
|
||||
@@ -199,6 +201,10 @@ export interface CopilotHistoriesArgs {
|
||||
options?: InputMaybe<QueryChatHistoriesInput>;
|
||||
}
|
||||
|
||||
export interface CopilotSessionArgs {
|
||||
sessionId: Scalars['String']['input'];
|
||||
}
|
||||
|
||||
export interface CopilotSessionIdsArgs {
|
||||
docId?: InputMaybe<Scalars['String']['input']>;
|
||||
options?: InputMaybe<QueryChatSessionsInput>;
|
||||
@@ -415,6 +421,8 @@ export interface CopilotQuota {
|
||||
export interface CopilotSessionType {
|
||||
__typename?: 'CopilotSessionType';
|
||||
id: Scalars['ID']['output'];
|
||||
model: Scalars['String']['output'];
|
||||
optionalModels: Array<Scalars['String']['output']>;
|
||||
parentSessionId: Maybe<Scalars['ID']['output']>;
|
||||
promptName: Scalars['String']['output'];
|
||||
}
|
||||
@@ -3453,6 +3461,29 @@ export type ForkCopilotSessionMutation = {
|
||||
forkCopilotSession: string;
|
||||
};
|
||||
|
||||
export type GetCopilotSessionQueryVariables = Exact<{
|
||||
workspaceId: Scalars['String']['input'];
|
||||
sessionId: Scalars['String']['input'];
|
||||
}>;
|
||||
|
||||
export type GetCopilotSessionQuery = {
|
||||
__typename?: 'Query';
|
||||
currentUser: {
|
||||
__typename?: 'UserType';
|
||||
copilot: {
|
||||
__typename?: 'Copilot';
|
||||
session: {
|
||||
__typename?: 'CopilotSessionType';
|
||||
id: string;
|
||||
parentSessionId: string | null;
|
||||
promptName: string;
|
||||
model: string;
|
||||
optionalModels: Array<string>;
|
||||
};
|
||||
};
|
||||
} | null;
|
||||
};
|
||||
|
||||
export type UpdateCopilotSessionMutationVariables = Exact<{
|
||||
options: UpdateChatSessionInput;
|
||||
}>;
|
||||
@@ -3479,6 +3510,8 @@ export type GetCopilotSessionsQuery = {
|
||||
id: string;
|
||||
parentSessionId: string | null;
|
||||
promptName: string;
|
||||
model: string;
|
||||
optionalModels: Array<string>;
|
||||
}>;
|
||||
};
|
||||
} | null;
|
||||
@@ -4997,6 +5030,11 @@ export type Queries =
|
||||
variables: CopilotQuotaQueryVariables;
|
||||
response: CopilotQuotaQuery;
|
||||
}
|
||||
| {
|
||||
name: 'getCopilotSessionQuery';
|
||||
variables: GetCopilotSessionQueryVariables;
|
||||
response: GetCopilotSessionQuery;
|
||||
}
|
||||
| {
|
||||
name: 'getCopilotSessionsQuery';
|
||||
variables: GetCopilotSessionsQueryVariables;
|
||||
|
||||
@@ -380,6 +380,10 @@ declare global {
|
||||
docId?: string,
|
||||
options?: { action?: boolean }
|
||||
) => Promise<CopilotSessionType[] | undefined>;
|
||||
getSession: (
|
||||
workspaceId: string,
|
||||
sessionId: string
|
||||
) => Promise<CopilotSessionType | undefined>;
|
||||
updateSession: (sessionId: string, promptName: string) => Promise<string>;
|
||||
}
|
||||
|
||||
|
||||
@@ -11,6 +11,7 @@ import {
|
||||
forkCopilotSessionMutation,
|
||||
getCopilotHistoriesQuery,
|
||||
getCopilotHistoryIdsQuery,
|
||||
getCopilotSessionQuery,
|
||||
getCopilotSessionsQuery,
|
||||
type GraphQLQuery,
|
||||
listContextObjectQuery,
|
||||
@@ -136,6 +137,18 @@ export class CopilotClient {
|
||||
}
|
||||
}
|
||||
|
||||
async getSession(workspaceId: string, sessionId: string) {
|
||||
try {
|
||||
const res = await this.gql({
|
||||
query: getCopilotSessionQuery,
|
||||
variables: { sessionId, workspaceId },
|
||||
});
|
||||
return res.currentUser?.copilot?.session;
|
||||
} catch (err) {
|
||||
throw resolveError(err);
|
||||
}
|
||||
}
|
||||
|
||||
async getSessions(
|
||||
workspaceId: string,
|
||||
docId?: string,
|
||||
|
||||
@@ -579,6 +579,9 @@ Could you make a new website based on these notes and send back just the html fi
|
||||
|
||||
AIProvider.provide('session', {
|
||||
createSession,
|
||||
getSession: async (workspaceId: string, sessionId: string) => {
|
||||
return client.getSession(workspaceId, sessionId);
|
||||
},
|
||||
getSessions: async (
|
||||
workspaceId: string,
|
||||
docId?: string,
|
||||
|
||||
Reference in New Issue
Block a user