mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
(core) Show tweaked formula in AI responses
Summary: The formula that's used when the Apply button is clicked, and the formula that's shown in responses from the Formula Assistant should now be the same. Previously, they would differ slightly. Test Plan: Server tests. Reviewers: alexmojaki Reviewed By: alexmojaki Subscribers: alexmojaki Differential Revision: https://phab.getgrist.com/D3977
This commit is contained in:
parent
4cfa033078
commit
05c15e4ec3
@ -635,25 +635,24 @@ export class FormulaAssistant extends Disposable {
|
|||||||
// Get the state of the chat from the column.
|
// Get the state of the chat from the column.
|
||||||
const conversationId = this._chat.conversationId.get();
|
const conversationId = this._chat.conversationId.get();
|
||||||
const prevState = column.chatHistory.peek().get().state;
|
const prevState = column.chatHistory.peek().get().state;
|
||||||
const {reply, suggestedActions, state} = await askAI(gristDoc, {
|
const {reply, suggestedActions, suggestedFormula, state} = await askAI(gristDoc, {
|
||||||
conversationId,
|
conversationId,
|
||||||
column,
|
column,
|
||||||
description,
|
description,
|
||||||
state: prevState,
|
state: prevState,
|
||||||
});
|
});
|
||||||
console.debug('suggestedActions', {suggestedActions, reply, state});
|
console.debug('received formula assistant response: ', {suggestedActions, suggestedFormula, reply, state});
|
||||||
// If back-end is capable of conversation, keep its state.
|
// If back-end is capable of conversation, keep its state.
|
||||||
const chatHistoryNew = column.chatHistory.peek();
|
const chatHistoryNew = column.chatHistory.peek();
|
||||||
const value = chatHistoryNew.get();
|
const value = chatHistoryNew.get();
|
||||||
value.state = state;
|
value.state = state;
|
||||||
const formula = (suggestedActions[0]?.[3] as any)?.formula as string;
|
|
||||||
// If model has a conversational skills (and maintains a history), we might get actually
|
// If model has a conversational skills (and maintains a history), we might get actually
|
||||||
// some markdown text back, so we need to parse it.
|
// some markdown text back, so we need to parse it.
|
||||||
const prettyMessage = state ? (reply || formula || '') : (formula || reply || '');
|
const prettyMessage = state ? (reply || suggestedFormula || '') : (suggestedFormula || reply || '');
|
||||||
// Add it to the chat.
|
// Add it to the chat.
|
||||||
return {
|
return {
|
||||||
message: prettyMessage,
|
message: prettyMessage,
|
||||||
formula,
|
formula: suggestedFormula,
|
||||||
action: suggestedActions[0],
|
action: suggestedActions[0],
|
||||||
sender: 'ai',
|
sender: 'ai',
|
||||||
};
|
};
|
||||||
|
@ -50,6 +50,7 @@ export interface AssistanceRequest {
|
|||||||
*/
|
*/
|
||||||
export interface AssistanceResponse {
|
export interface AssistanceResponse {
|
||||||
suggestedActions: DocAction[];
|
suggestedActions: DocAction[];
|
||||||
|
suggestedFormula?: string;
|
||||||
state?: AssistanceState;
|
state?: AssistanceState;
|
||||||
// If the model can be trusted to issue a self-contained
|
// If the model can be trusted to issue a self-contained
|
||||||
// markdown-friendly string, it can be included here.
|
// markdown-friendly string, it can be included here.
|
||||||
|
@ -198,7 +198,14 @@ export class OpenAIAssistant implements Assistant {
|
|||||||
|
|
||||||
const userIdHash = getUserHash(optSession);
|
const userIdHash = getUserHash(optSession);
|
||||||
const completion: string = await this._getCompletion(messages, userIdHash);
|
const completion: string = await this._getCompletion(messages, userIdHash);
|
||||||
const response = await completionToResponse(doc, request, completion, completion);
|
const response = await completionToResponse(doc, request, completion);
|
||||||
|
if (response.suggestedFormula) {
|
||||||
|
// Show the tweaked version of the suggested formula to the user (i.e. the one that's
|
||||||
|
// copied when the Apply button is clicked).
|
||||||
|
response.reply = replaceMarkdownCode(completion, response.suggestedFormula);
|
||||||
|
} else {
|
||||||
|
response.reply = completion;
|
||||||
|
}
|
||||||
response.state = {messages};
|
response.state = {messages};
|
||||||
doc.logTelemetryEvent(optSession, 'assistantReceive', {
|
doc.logTelemetryEvent(optSession, 'assistantReceive', {
|
||||||
full: {
|
full: {
|
||||||
@ -208,7 +215,7 @@ export class OpenAIAssistant implements Assistant {
|
|||||||
index: messages.length - 1,
|
index: messages.length - 1,
|
||||||
content: completion,
|
content: completion,
|
||||||
},
|
},
|
||||||
suggestedFormula: (response.suggestedActions[0]?.[3] as any)?.formula,
|
suggestedFormula: response.suggestedFormula,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
return response;
|
return response;
|
||||||
@ -407,6 +414,14 @@ export async function sendForCompletion(
|
|||||||
return await assistant.apply(optSession, doc, request);
|
return await assistant.apply(optSession, doc, request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a new Markdown string with the contents of its first multi-line code block
|
||||||
|
* replaced with `replaceValue`.
|
||||||
|
*/
|
||||||
|
export function replaceMarkdownCode(markdown: string, replaceValue: string) {
|
||||||
|
return markdown.replace(/```\w*\n(.*)```/s, '```python\n' + replaceValue + '\n```');
|
||||||
|
}
|
||||||
|
|
||||||
async function makeSchemaPromptV1(session: OptDocSession, doc: AssistanceDoc, request: AssistanceRequest) {
|
async function makeSchemaPromptV1(session: OptDocSession, doc: AssistanceDoc, request: AssistanceRequest) {
|
||||||
if (request.context.type !== 'formula') {
|
if (request.context.type !== 'formula') {
|
||||||
throw new Error('makeSchemaPromptV1 only works for formulas');
|
throw new Error('makeSchemaPromptV1 only works for formulas');
|
||||||
@ -418,23 +433,28 @@ async function makeSchemaPromptV1(session: OptDocSession, doc: AssistanceDoc, re
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async function completionToResponse(doc: AssistanceDoc, request: AssistanceRequest,
|
async function completionToResponse(
|
||||||
completion: string, reply?: string): Promise<AssistanceResponse> {
|
doc: AssistanceDoc,
|
||||||
|
request: AssistanceRequest,
|
||||||
|
completion: string,
|
||||||
|
reply?: string
|
||||||
|
): Promise<AssistanceResponse> {
|
||||||
if (request.context.type !== 'formula') {
|
if (request.context.type !== 'formula') {
|
||||||
throw new Error('completionToResponse only works for formulas');
|
throw new Error('completionToResponse only works for formulas');
|
||||||
}
|
}
|
||||||
completion = await doc.assistanceFormulaTweak(completion);
|
const suggestedFormula = await doc.assistanceFormulaTweak(completion) || undefined;
|
||||||
// Suggest an action only if the completion is non-empty (that is,
|
// Suggest an action only if the completion is non-empty (that is,
|
||||||
// it actually looked like code).
|
// it actually looked like code).
|
||||||
const suggestedActions: DocAction[] = completion ? [[
|
const suggestedActions: DocAction[] = suggestedFormula ? [[
|
||||||
"ModifyColumn",
|
"ModifyColumn",
|
||||||
request.context.tableId,
|
request.context.tableId,
|
||||||
request.context.colId, {
|
request.context.colId, {
|
||||||
formula: completion,
|
formula: suggestedFormula,
|
||||||
}
|
}
|
||||||
]] : [];
|
]] : [];
|
||||||
return {
|
return {
|
||||||
suggestedActions,
|
suggestedActions,
|
||||||
|
suggestedFormula,
|
||||||
reply,
|
reply,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user