mirror of
https://github.com/block/goose.git
synced 2026-04-28 03:29:36 +00:00
fix(desktop): start recipe deeplink sessions from the recipe prompt (#8424)
Some checks are pending
Canary / Prepare Version (push) Waiting to run
Canary / build-cli (push) Blocked by required conditions
Canary / Upload Install Script (push) Blocked by required conditions
Canary / bundle-desktop (push) Blocked by required conditions
Canary / bundle-desktop-intel (push) Blocked by required conditions
Canary / bundle-desktop-linux (push) Blocked by required conditions
Canary / bundle-desktop-windows (push) Blocked by required conditions
Canary / Release (push) Blocked by required conditions
CI / changes (push) Waiting to run
CI / Check Rust Code Format (push) Blocked by required conditions
CI / Build and Test Rust Project (push) Blocked by required conditions
CI / Build Rust Project on Windows (push) Waiting to run
CI / Lint Rust Code (push) Blocked by required conditions
CI / Check OpenAPI Schema is Up-to-Date (push) Blocked by required conditions
CI / Test and Lint Electron Desktop App (push) Blocked by required conditions
Live Provider Tests / Smoke Tests (Code Execution) (push) Blocked by required conditions
Live Provider Tests / check-fork (push) Waiting to run
Live Provider Tests / changes (push) Blocked by required conditions
Live Provider Tests / Build Binary (push) Blocked by required conditions
Live Provider Tests / Smoke Tests (push) Blocked by required conditions
Live Provider Tests / Compaction Tests (push) Blocked by required conditions
Live Provider Tests / goose server HTTP integration tests (push) Blocked by required conditions
Publish Docker Image / docker (push) Waiting to run
Scorecard supply-chain security / Scorecard analysis (push) Waiting to run
Some checks are pending
Canary / Prepare Version (push) Waiting to run
Canary / build-cli (push) Blocked by required conditions
Canary / Upload Install Script (push) Blocked by required conditions
Canary / bundle-desktop (push) Blocked by required conditions
Canary / bundle-desktop-intel (push) Blocked by required conditions
Canary / bundle-desktop-linux (push) Blocked by required conditions
Canary / bundle-desktop-windows (push) Blocked by required conditions
Canary / Release (push) Blocked by required conditions
CI / changes (push) Waiting to run
CI / Check Rust Code Format (push) Blocked by required conditions
CI / Build and Test Rust Project (push) Blocked by required conditions
CI / Build Rust Project on Windows (push) Waiting to run
CI / Lint Rust Code (push) Blocked by required conditions
CI / Check OpenAPI Schema is Up-to-Date (push) Blocked by required conditions
CI / Test and Lint Electron Desktop App (push) Blocked by required conditions
Live Provider Tests / Smoke Tests (Code Execution) (push) Blocked by required conditions
Live Provider Tests / check-fork (push) Waiting to run
Live Provider Tests / changes (push) Blocked by required conditions
Live Provider Tests / Build Binary (push) Blocked by required conditions
Live Provider Tests / Smoke Tests (push) Blocked by required conditions
Live Provider Tests / Compaction Tests (push) Blocked by required conditions
Live Provider Tests / goose server HTTP integration tests (push) Blocked by required conditions
Publish Docker Image / docker (push) Waiting to run
Scorecard supply-chain security / Scorecard analysis (push) Waiting to run
Signed-off-by: sunilkumarvalmiki <g.sunilkumarvalmiki@gmail.com> Co-authored-by: Lifei Zhou <lifei@squareup.com>
This commit is contained in:
parent
ed4836b923
commit
b1eff5f7f9
5 changed files with 140 additions and 6 deletions
|
|
@ -6,7 +6,7 @@
|
|||
import React from 'react';
|
||||
import { screen, render, waitFor } from '@testing-library/react';
|
||||
import { vi, describe, it, expect, beforeEach, afterEach } from 'vitest';
|
||||
import { AppInner } from './App';
|
||||
import { AppInner, resolveSessionInitialMessage } from './App';
|
||||
import { IntlTestWrapper } from './i18n/test-utils';
|
||||
|
||||
// Set up globals for jsdom
|
||||
|
|
@ -60,6 +60,7 @@ vi.mock('./sessions', () => ({
|
|||
.fn()
|
||||
.mockResolvedValue({ sessionId: 'test', messages: [], metadata: { description: '' } }),
|
||||
generateSessionId: vi.fn(),
|
||||
createSession: vi.fn(),
|
||||
}));
|
||||
|
||||
// Mock the ConfigContext module
|
||||
|
|
@ -161,7 +162,7 @@ const mockElectron = {
|
|||
|
||||
// Mock appConfig
|
||||
const mockAppConfig = {
|
||||
get: vi.fn((key: string) => {
|
||||
get: vi.fn((key: string): string | null => {
|
||||
if (key === 'GOOSE_WORKING_DIR') return '/test/dir';
|
||||
return null;
|
||||
}),
|
||||
|
|
@ -191,6 +192,10 @@ describe('App Component - Brand New State', () => {
|
|||
vi.clearAllMocks();
|
||||
mockNavigate.mockClear();
|
||||
mockSetSearchParams.mockClear();
|
||||
mockAppConfig.get.mockImplementation((key: string): string | null => {
|
||||
if (key === 'GOOSE_WORKING_DIR') return '/test/dir';
|
||||
return null;
|
||||
});
|
||||
|
||||
// Reset search params
|
||||
mockSearchParams.forEach((_, key) => {
|
||||
|
|
@ -290,4 +295,20 @@ describe('App Component - Brand New State', () => {
|
|||
// App should still initialize without any navigation calls
|
||||
expect(mockNavigate).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should seed recipe sessions with the recipe prompt when no initial message is provided', () => {
|
||||
expect(
|
||||
resolveSessionInitialMessage(
|
||||
{
|
||||
recipe: {
|
||||
prompt: 'Write a release note for the latest change',
|
||||
},
|
||||
},
|
||||
undefined
|
||||
)
|
||||
).toEqual({
|
||||
msg: 'Write a release note for the latest change',
|
||||
images: [],
|
||||
});
|
||||
});
|
||||
});
|
||||
|
|
|
|||
|
|
@ -68,6 +68,16 @@ const HubRouteWrapper = () => {
|
|||
return <Hub setView={setView} />;
|
||||
};
|
||||
|
||||
export function resolveSessionInitialMessage(
|
||||
session: { recipe?: { prompt?: string | null } | null },
|
||||
initialMessage?: UserInput
|
||||
): UserInput | undefined {
|
||||
return (
|
||||
initialMessage ??
|
||||
(session.recipe?.prompt ? { msg: session.recipe.prompt, images: [] } : undefined)
|
||||
);
|
||||
}
|
||||
|
||||
const PairRouteWrapper = ({
|
||||
activeSessions,
|
||||
}: {
|
||||
|
|
@ -105,12 +115,13 @@ const PairRouteWrapper = ({
|
|||
recipeId: recipeIdFromConfig,
|
||||
allExtensions: extensionsList,
|
||||
});
|
||||
const sessionInitialMessage = resolveSessionInitialMessage(newSession, initialMessage);
|
||||
|
||||
window.dispatchEvent(
|
||||
new CustomEvent(AppEvents.ADD_ACTIVE_SESSION, {
|
||||
detail: {
|
||||
sessionId: newSession.id,
|
||||
initialMessage,
|
||||
initialMessage: sessionInitialMessage,
|
||||
},
|
||||
})
|
||||
);
|
||||
|
|
|
|||
|
|
@ -137,12 +137,15 @@ export default function BaseChat({
|
|||
return initialMessage;
|
||||
}, [initialMessage, recipe?.prompt, session?.user_recipe_values]);
|
||||
|
||||
const canAutoSubmit = !recipe || hasNotAcceptedRecipe === false;
|
||||
|
||||
useAutoSubmit({
|
||||
sessionId,
|
||||
session,
|
||||
messages,
|
||||
chatState,
|
||||
initialMessage: resolvedInitialMessage,
|
||||
canAutoSubmit,
|
||||
handleSubmit,
|
||||
});
|
||||
|
||||
|
|
@ -206,7 +209,7 @@ export default function BaseChat({
|
|||
const sessionLoaded = session !== undefined;
|
||||
|
||||
useEffect(() => {
|
||||
if (!recipe) return;
|
||||
if (!recipe || !isActiveSession) return;
|
||||
|
||||
(async () => {
|
||||
const accepted = await window.electron.hasAcceptedRecipeBefore(recipe);
|
||||
|
|
@ -217,7 +220,7 @@ export default function BaseChat({
|
|||
setHasRecipeSecurityWarnings(scanResult.has_security_warnings);
|
||||
}
|
||||
})();
|
||||
}, [recipe]);
|
||||
}, [recipe, isActiveSession]);
|
||||
|
||||
const handleRecipeAccept = async (accept: boolean) => {
|
||||
if (recipe && accept) {
|
||||
|
|
@ -525,7 +528,7 @@ export default function BaseChat({
|
|||
</div>
|
||||
</MainPanelLayout>
|
||||
|
||||
{recipe && (
|
||||
{recipe && isActiveSession && (
|
||||
<RecipeWarningModal
|
||||
isOpen={!!hasNotAcceptedRecipe}
|
||||
onConfirm={() => handleRecipeAccept(true)}
|
||||
|
|
|
|||
92
ui/desktop/src/hooks/useAutoSubmit.test.tsx
Normal file
92
ui/desktop/src/hooks/useAutoSubmit.test.tsx
Normal file
|
|
@ -0,0 +1,92 @@
|
|||
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
||||
import { renderHook } from '@testing-library/react';
|
||||
import { MemoryRouter } from 'react-router-dom';
|
||||
import type { PropsWithChildren } from 'react';
|
||||
import { useAutoSubmit } from './useAutoSubmit';
|
||||
import { ChatState } from '../types/chatState';
|
||||
import type { Session } from '../api';
|
||||
import type { UserInput } from '../types/message';
|
||||
|
||||
function makeSession(overrides: Partial<Session> = {}): Session {
|
||||
return {
|
||||
id: 'sess-1',
|
||||
name: 'untitled',
|
||||
message_count: 0,
|
||||
created_at: new Date().toISOString(),
|
||||
updated_at: new Date().toISOString(),
|
||||
working_dir: '/tmp',
|
||||
extension_data: { active: [], installed: [] },
|
||||
...overrides,
|
||||
} as Session;
|
||||
}
|
||||
|
||||
const initialMessage: UserInput = {
|
||||
msg: 'Run the recipe',
|
||||
images: [],
|
||||
};
|
||||
|
||||
describe('useAutoSubmit', () => {
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
});
|
||||
|
||||
it('does not auto-submit while recipe acceptance is unresolved', () => {
|
||||
const handleSubmit = vi.fn();
|
||||
const dispatchEventSpy = vi.spyOn(window, 'dispatchEvent');
|
||||
|
||||
const wrapper = ({ children }: PropsWithChildren) => (
|
||||
<MemoryRouter initialEntries={['/pair?resumeSessionId=sess-1']}>{children}</MemoryRouter>
|
||||
);
|
||||
|
||||
renderHook(
|
||||
() =>
|
||||
useAutoSubmit({
|
||||
sessionId: 'sess-1',
|
||||
session: makeSession(),
|
||||
messages: [],
|
||||
chatState: ChatState.Idle,
|
||||
initialMessage,
|
||||
canAutoSubmit: false,
|
||||
handleSubmit,
|
||||
}),
|
||||
{ wrapper }
|
||||
);
|
||||
|
||||
expect(handleSubmit).not.toHaveBeenCalled();
|
||||
expect(dispatchEventSpy).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('auto-submits once recipe acceptance is confirmed', () => {
|
||||
const handleSubmit = vi.fn();
|
||||
const dispatchEventSpy = vi.spyOn(window, 'dispatchEvent');
|
||||
|
||||
const wrapper = ({ children }: PropsWithChildren) => (
|
||||
<MemoryRouter initialEntries={['/pair?resumeSessionId=sess-1']}>{children}</MemoryRouter>
|
||||
);
|
||||
|
||||
const { rerender } = renderHook(
|
||||
({ canAutoSubmit }) =>
|
||||
useAutoSubmit({
|
||||
sessionId: 'sess-1',
|
||||
session: makeSession(),
|
||||
messages: [],
|
||||
chatState: ChatState.Idle,
|
||||
initialMessage,
|
||||
canAutoSubmit,
|
||||
handleSubmit,
|
||||
}),
|
||||
{
|
||||
initialProps: { canAutoSubmit: false },
|
||||
wrapper,
|
||||
}
|
||||
);
|
||||
|
||||
expect(handleSubmit).not.toHaveBeenCalled();
|
||||
|
||||
rerender({ canAutoSubmit: true });
|
||||
|
||||
expect(handleSubmit).toHaveBeenCalledTimes(1);
|
||||
expect(handleSubmit).toHaveBeenCalledWith(initialMessage);
|
||||
expect(dispatchEventSpy).toHaveBeenCalledTimes(1);
|
||||
});
|
||||
});
|
||||
|
|
@ -19,6 +19,7 @@ interface UseAutoSubmitProps {
|
|||
messages: Message[];
|
||||
chatState: ChatState;
|
||||
initialMessage: UserInput | undefined;
|
||||
canAutoSubmit?: boolean;
|
||||
handleSubmit: (input: UserInput) => void;
|
||||
}
|
||||
|
||||
|
|
@ -32,6 +33,7 @@ export function useAutoSubmit({
|
|||
messages,
|
||||
chatState,
|
||||
initialMessage,
|
||||
canAutoSubmit = true,
|
||||
handleSubmit,
|
||||
}: UseAutoSubmitProps): UseAutoSubmitReturn {
|
||||
const [searchParams] = useSearchParams();
|
||||
|
|
@ -65,6 +67,10 @@ export function useAutoSubmit({
|
|||
return;
|
||||
}
|
||||
|
||||
if (!canAutoSubmit) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (chatState !== ChatState.Idle) {
|
||||
return;
|
||||
}
|
||||
|
|
@ -107,6 +113,7 @@ export function useAutoSubmit({
|
|||
sessionId,
|
||||
messages.length,
|
||||
chatState,
|
||||
canAutoSubmit,
|
||||
clearInitialMessage,
|
||||
hasUnfilledParameters,
|
||||
]);
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue