Fix Input losing focus (#382)

* fix(PaLM2): input losing focus on message stream ending

* fix(askOpenAI.js): fix typo in variable name from newUserMassageId to newUserMessageId

* feat(chatgpt-browser.js, askBingAI.js, askChatGPTBrowser.js): add onEventMessage callback to browserClient

Add onEventMessage callback to browserClient to handle event messages from the server. In askChatGPTBrowser.js, add a getPartialMessage variable to store the partial message text. In askBingAI.js, fix a typo in the variable name newUserMassageId to newUserMessageId. In askChatGPTBrowser.js, remove the preSendRequest parameter and move the sendMessage call to the onEventMessage callback. In askChatGPTBrowser.js, add a check for null or undefined value of getPartialMessage before appending it to the error message.

* fix(bing): input no longer loses input focus as convoId is persisted from beginning of convo

* refactor(Input): remove unused code and fix input autofocus
feat(package.json): add e2e:test-auth script to test authentication flow with saved storage
This commit is contained in:
Danny Avila 2023-05-26 14:32:13 -04:00 committed by GitHub
parent 11b98d3d13
commit c0845ad0b1
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 59 additions and 72 deletions

View file

@ -8,6 +8,7 @@ const browserClient = async ({
model, model,
token, token,
onProgress, onProgress,
onEventMessage,
abortController, abortController,
userId userId
}) => { }) => {
@ -30,7 +31,7 @@ const browserClient = async ({
}; };
const client = new ChatGPTBrowserClient(clientOptions, store); const client = new ChatGPTBrowserClient(clientOptions, store);
let options = { onProgress, abortController }; let options = { onProgress, onEventMessage, abortController };
if (!!parentMessageId && !!conversationId) { if (!!parentMessageId && !!conversationId) {
options = { ...options, parentMessageId, conversationId }; options = { ...options, parentMessageId, conversationId };

View file

@ -24,6 +24,10 @@ const convoSchema = mongoose.Schema(
examples: [{ type: mongoose.Schema.Types.Mixed }], examples: [{ type: mongoose.Schema.Types.Mixed }],
...conversationPreset, ...conversationPreset,
// for bingAI only // for bingAI only
bingConversationId: {
type: String,
default: null
},
jailbreakConversationId: { jailbreakConversationId: {
type: String, type: String,
default: null default: null

View file

@ -129,10 +129,15 @@ const ask = async ({
} }
}); });
const abortController = new AbortController(); const abortController = new AbortController();
let bingConversationId = null;
if (!isNewConversation) {
const convo = await getConvo(req.user.id, conversationId);
bingConversationId = convo.bingConversationId;
}
let response = await askBing({ let response = await askBing({
text, text,
parentMessageId: userParentMessageId, parentMessageId: userParentMessageId,
conversationId, conversationId: bingConversationId ?? conversationId,
...endpointOption, ...endpointOption,
onProgress: progressCallback.call(null, { onProgress: progressCallback.call(null, {
res, res,
@ -147,7 +152,7 @@ const ask = async ({
const newConversationId = endpointOption?.jailbreak const newConversationId = endpointOption?.jailbreak
? response.jailbreakConversationId ? response.jailbreakConversationId
: response.conversationId || conversationId; : response.conversationId || conversationId;
const newUserMassageId = const newUserMessageId =
response.parentMessageId || response.details.requestId || userMessageId; response.parentMessageId || response.details.requestId || userMessageId;
const newResponseMessageId = response.messageId || response.details.messageId; const newResponseMessageId = response.messageId || response.details.messageId;
@ -156,10 +161,11 @@ const ask = async ({
response.response || response.details.spokenText || '**Bing refused to answer.**'; response.response || response.details.spokenText || '**Bing refused to answer.**';
let responseMessage = { let responseMessage = {
conversationId: newConversationId, conversationId,
bingConversationId: newConversationId,
messageId: responseMessageId, messageId: responseMessageId,
newMessageId: newResponseMessageId, newMessageId: newResponseMessageId,
parentMessageId: overrideParentMessageId || newUserMassageId, parentMessageId: overrideParentMessageId || newUserMessageId,
sender: endpointOption?.jailbreak ? 'Sydney' : 'BingAI', sender: endpointOption?.jailbreak ? 'Sydney' : 'BingAI',
text: await handleText(response, true), text: await handleText(response, true),
suggestions: suggestions:
@ -173,31 +179,7 @@ const ask = async ({
await saveMessage(responseMessage); await saveMessage(responseMessage);
responseMessage.messageId = newResponseMessageId; responseMessage.messageId = newResponseMessageId;
// STEP2 update the convosation. let conversationUpdate = { conversationId, bingConversationId: newConversationId, endpoint: 'bingAI' };
// First update conversationId if needed
// Note!
// Bing API will not use our conversationId at the first time,
// so change the placeholder conversationId to the real one.
// Attition: the api will also create new conversationId while using invalid userMessage.parentMessageId,
// but in this situation, don't change the conversationId, but create new convo.
let conversationUpdate = { conversationId: newConversationId, endpoint: 'bingAI' };
if (conversationId != newConversationId)
if (isNewConversation) {
// change the conversationId to new one
conversationUpdate = {
...conversationUpdate,
conversationId: conversationId,
newConversationId: newConversationId
};
} else {
// create new conversation
conversationUpdate = {
...conversationUpdate,
...endpointOption
};
}
if (endpointOption?.jailbreak) { if (endpointOption?.jailbreak) {
conversationUpdate.jailbreak = true; conversationUpdate.jailbreak = true;
@ -210,20 +192,16 @@ const ask = async ({
} }
await saveConvo(req.user.id, conversationUpdate); await saveConvo(req.user.id, conversationUpdate);
conversationId = newConversationId; userMessage.messageId = newUserMessageId;
// STEP3 update the user message
userMessage.conversationId = newConversationId;
userMessage.messageId = newUserMassageId;
// If response has parentMessageId, the fake userMessage.messageId should be updated to the real one. // If response has parentMessageId, the fake userMessage.messageId should be updated to the real one.
if (!overrideParentMessageId) if (!overrideParentMessageId)
await saveMessage({ await saveMessage({
...userMessage, ...userMessage,
messageId: userMessageId, messageId: userMessageId,
newMessageId: newUserMassageId newMessageId: newUserMessageId
}); });
userMessageId = newUserMassageId; userMessageId = newUserMessageId;
sendMessage(res, { sendMessage(res, {
title: await getConvoTitle(req.user.id, conversationId), title: await getConvoTitle(req.user.id, conversationId),

View file

@ -76,7 +76,6 @@ const ask = async ({
userMessage, userMessage,
endpointOption, endpointOption,
conversationId, conversationId,
preSendRequest = true,
overrideParentMessageId = null, overrideParentMessageId = null,
req, req,
res res
@ -92,10 +91,8 @@ const ask = async ({
'X-Accel-Buffering': 'no' 'X-Accel-Buffering': 'no'
}); });
if (preSendRequest) sendMessage(res, { message: userMessage, created: true });
let responseMessageId = crypto.randomUUID(); let responseMessageId = crypto.randomUUID();
let getPartialMessage = null;
try { try {
let lastSavedTimestamp = 0; let lastSavedTimestamp = 0;
const { onProgress: progressCallback, getPartialText } = createOnProgress({ const { onProgress: progressCallback, getPartialText } = createOnProgress({
@ -116,15 +113,30 @@ const ask = async ({
} }
} }
}); });
getPartialMessage = getPartialText;
const abortController = new AbortController(); const abortController = new AbortController();
let response = await browserClient({ let response = await browserClient({
text, text,
parentMessageId: userParentMessageId, parentMessageId: userParentMessageId,
conversationId, conversationId,
...endpointOption, ...endpointOption,
onProgress: progressCallback.call(null, { res, text }),
abortController, abortController,
userId userId,
onProgress: progressCallback.call(null, { res, text }),
onEventMessage: (eventMessage) => {
let data = null;
try {
data = JSON.parse(eventMessage.data);
} catch (e) {
return;
}
sendMessage(res, {
message: { ...userMessage, conversationId: data.conversation_id },
created: true
});
}
}); });
console.log('CLIENT RESPONSE', response); console.log('CLIENT RESPONSE', response);
@ -212,8 +224,8 @@ const ask = async ({
parentMessageId: overrideParentMessageId || userMessageId, parentMessageId: overrideParentMessageId || userMessageId,
unfinished: false, unfinished: false,
cancelled: false, cancelled: false,
error: true, // error: true,
text: error.message text: `${getPartialMessage() ?? ''}\n\nError message: "${error.message}"`
}; };
await saveMessage(errorMessage); await saveMessage(errorMessage);
handleError(res, errorMessage); handleError(res, errorMessage);

View file

@ -1,5 +1,6 @@
const express = require('express'); const express = require('express');
const router = express.Router(); const router = express.Router();
const crypto = require('crypto');
const { titleConvo } = require('../../../app/'); const { titleConvo } = require('../../../app/');
const GoogleClient = require('../../../app/google/GoogleClient'); const GoogleClient = require('../../../app/google/GoogleClient');
const { saveMessage, getConvoTitle, saveConvo, getConvo } = require('../../../models'); const { saveMessage, getConvoTitle, saveConvo, getConvo } = require('../../../models');
@ -7,7 +8,7 @@ const { handleError, sendMessage, createOnProgress } = require('./handlers');
const requireJwtAuth = require('../../../middleware/requireJwtAuth'); const requireJwtAuth = require('../../../middleware/requireJwtAuth');
router.post('/', requireJwtAuth, async (req, res) => { router.post('/', requireJwtAuth, async (req, res) => {
const { endpoint, text, parentMessageId, conversationId } = req.body; const { endpoint, text, parentMessageId, conversationId: oldConversationId } = req.body;
if (text.length === 0) return handleError(res, { text: 'Prompt empty or too short' }); if (text.length === 0) return handleError(res, { text: 'Prompt empty or too short' });
if (endpoint !== 'google') return handleError(res, { text: 'Illegal request' }); if (endpoint !== 'google') return handleError(res, { text: 'Illegal request' });
@ -31,6 +32,8 @@ router.post('/', requireJwtAuth, async (req, res) => {
return handleError(res, { text: `Illegal request: model` }); return handleError(res, { text: `Illegal request: model` });
} }
const conversationId = oldConversationId || crypto.randomUUID();
// eslint-disable-next-line no-use-before-define // eslint-disable-next-line no-use-before-define
return await ask({ return await ask({
text, text,
@ -64,6 +67,8 @@ const ask = async ({ text, endpointOption, parentMessageId = null, conversationI
if (!conversationId) { if (!conversationId) {
conversationId = data.conversationId; conversationId = data.conversationId;
} }
sendMessage(res, { message: userMessage, created: true });
}; };
const { onProgress: progressCallback } = createOnProgress({ const { onProgress: progressCallback } = createOnProgress({

View file

@ -190,7 +190,7 @@ const ask = async ({
console.log('CLIENT RESPONSE', response); console.log('CLIENT RESPONSE', response);
const newConversationId = response.conversationId || conversationId; const newConversationId = response.conversationId || conversationId;
const newUserMassageId = response.parentMessageId || userMessageId; const newUserMessageId = response.parentMessageId || userMessageId;
const newResponseMessageId = response.messageId; const newResponseMessageId = response.messageId;
// STEP1 generate response message // STEP1 generate response message
@ -200,7 +200,7 @@ const ask = async ({
conversationId: newConversationId, conversationId: newConversationId,
messageId: responseMessageId, messageId: responseMessageId,
newMessageId: newResponseMessageId, newMessageId: newResponseMessageId,
parentMessageId: overrideParentMessageId || newUserMassageId, parentMessageId: overrideParentMessageId || newUserMessageId,
text: await handleText(response), text: await handleText(response),
sender: endpointOption?.chatGptLabel || 'ChatGPT', sender: endpointOption?.chatGptLabel || 'ChatGPT',
unfinished: false, unfinished: false,
@ -234,16 +234,16 @@ const ask = async ({
// STEP3 update the user message // STEP3 update the user message
userMessage.conversationId = newConversationId; userMessage.conversationId = newConversationId;
userMessage.messageId = newUserMassageId; userMessage.messageId = newUserMessageId;
// If response has parentMessageId, the fake userMessage.messageId should be updated to the real one. // If response has parentMessageId, the fake userMessage.messageId should be updated to the real one.
if (!overrideParentMessageId) if (!overrideParentMessageId)
await saveMessage({ await saveMessage({
...userMessage, ...userMessage,
messageId: userMessageId, messageId: userMessageId,
newMessageId: newUserMassageId newMessageId: newUserMessageId
}); });
userMessageId = newUserMassageId; userMessageId = newUserMessageId;
sendMessage(res, { sendMessage(res, {
title: await getConvoTitle(req.user.id, conversationId), title: await getConvoTitle(req.user.id, conversationId),

View file

@ -5,7 +5,6 @@ import OpenAIOptions from './OpenAIOptions';
import ChatGPTOptions from './ChatGPTOptions'; import ChatGPTOptions from './ChatGPTOptions';
import BingAIOptions from './BingAIOptions'; import BingAIOptions from './BingAIOptions';
import GoogleOptions from './GoogleOptions'; import GoogleOptions from './GoogleOptions';
// import BingStyles from './BingStyles';
import NewConversationMenu from './NewConversationMenu'; import NewConversationMenu from './NewConversationMenu';
import AdjustToneButton from './AdjustToneButton'; import AdjustToneButton from './AdjustToneButton';
import Footer from './Footer'; import Footer from './Footer';
@ -21,7 +20,6 @@ export default function TextChat({ isSearchView = false }) {
const conversation = useRecoilValue(store.conversation); const conversation = useRecoilValue(store.conversation);
const latestMessage = useRecoilValue(store.latestMessage); const latestMessage = useRecoilValue(store.latestMessage);
const [text, setText] = useRecoilState(store.text); const [text, setText] = useRecoilState(store.text);
// const [text, setText] = useState('');
const endpointsConfig = useRecoilValue(store.endpointsConfig); const endpointsConfig = useRecoilValue(store.endpointsConfig);
const isSubmitting = useRecoilValue(store.isSubmitting); const isSubmitting = useRecoilValue(store.isSubmitting);
@ -30,8 +28,6 @@ export default function TextChat({ isSearchView = false }) {
const disabled = false; const disabled = false;
const { ask, stopGenerating } = useMessageHandler(); const { ask, stopGenerating } = useMessageHandler();
// const bingStylesRef = useRef(null);
const [showBingToneSetting, setShowBingToneSetting] = useState(false); const [showBingToneSetting, setShowBingToneSetting] = useState(false);
const isNotAppendable = latestMessage?.unfinished & !isSubmitting || latestMessage?.error; const isNotAppendable = latestMessage?.unfinished & !isSubmitting || latestMessage?.error;
@ -39,25 +35,15 @@ export default function TextChat({ isSearchView = false }) {
// auto focus to input, when enter a conversation. // auto focus to input, when enter a conversation.
useEffect(() => { useEffect(() => {
if (conversation?.conversationId !== 'search') inputRef.current?.focus(); if (conversation?.conversationId !== 'search') inputRef.current?.focus();
// setText('');
}, [conversation?.conversationId]); }, [conversation?.conversationId]);
// // controls the height of Bing tone style tabs useEffect(() => {
// useEffect(() => { const timeoutId = setTimeout(() => {
// if (!inputRef.current) { inputRef.current?.focus();
// return; // wait for the ref to be available }, 100);
// }
// const resizeObserver = new ResizeObserver(() => { return () => clearTimeout(timeoutId);
// const newHeight = inputRef.current.clientHeight; }, [isSubmitting]);
// if (newHeight >= 24) {
// // 24 is the default height of the input
// bingStylesRef.current.style.bottom = 15 + newHeight + 'px';
// }
// });
// resizeObserver.observe(inputRef.current);
// return () => resizeObserver.disconnect();
// }, [inputRef]);
const submitMessage = () => { const submitMessage = () => {
ask({ text }); ask({ text });

View file

@ -16,6 +16,7 @@
"e2e:debug": "cross-env PWDEBUG=1 playwright test --config=e2e/playwright.config.js", "e2e:debug": "cross-env PWDEBUG=1 playwright test --config=e2e/playwright.config.js",
"e2e:report": "npx playwright show-report e2e/playwright-report", "e2e:report": "npx playwright show-report e2e/playwright-report",
"e2e:auth": "npx playwright codegen --save-storage=e2e/auth.json http://localhost:3080/", "e2e:auth": "npx playwright codegen --save-storage=e2e/auth.json http://localhost:3080/",
"e2e:test-auth": "npx playwright codegen --load-storage=e2e/auth.json http://localhost:3080/",
"prepare": "husky install", "prepare": "husky install",
"format": "prettier-eslint --write \"{,!(node_modules)/**/}*.{js,jsx,ts,tsx}\"" "format": "prettier-eslint --write \"{,!(node_modules)/**/}*.{js,jsx,ts,tsx}\""
}, },