From a5e81c33d1cc0c5bd4c733b8b23588b81181dde1 Mon Sep 17 00:00:00 2001 From: alashchev17 Date: Mon, 13 Jan 2025 12:46:08 +0100 Subject: [PATCH] feat: adding stories for tool confirmation & better highlighting of rules and links --- src/__fixtures__/confirmation.ts | 52 +++++++++++++ .../ChatForm/ToolConfirmation.stories.tsx | 75 +++++++++++++++++++ src/components/ChatForm/ToolConfirmation.tsx | 3 +- src/components/Markdown/CodeBlock.tsx | 9 ++- src/components/Markdown/Markdown.tsx | 13 +++- 5 files changed, 145 insertions(+), 7 deletions(-) create mode 100644 src/__fixtures__/confirmation.ts create mode 100644 src/components/ChatForm/ToolConfirmation.stories.tsx diff --git a/src/__fixtures__/confirmation.ts b/src/__fixtures__/confirmation.ts new file mode 100644 index 00000000..0c3edc01 --- /dev/null +++ b/src/__fixtures__/confirmation.ts @@ -0,0 +1,52 @@ +import { ToolConfirmationPauseReason } from "../services/refact"; + +export const CONFIRMATIONAL_PAUSE_REASONS_WITH_PATH: ToolConfirmationPauseReason[] = + [ + { + command: "SELECT *", + rule: "*", + type: "confirmation", + tool_call_id: "1", + integr_config_path: + "\\\\?\\d:\\work\\refact.ai\\refact-lsp\\.refact\\integrations\\postgres.yaml", + }, + ]; +export const CONFIRMATIONAL_PAUSE_REASONS: ToolConfirmationPauseReason[] = [ + { + command: "patch", + rule: "default", + type: "confirmation", + tool_call_id: "1", + integr_config_path: null, + }, +]; + +export const DENIAL_PAUSE_REASONS_WITH_PATH: ToolConfirmationPauseReason[] = [ + { + command: "SELECT *", + rule: "*", + type: "denial", + tool_call_id: "1", + integr_config_path: + "\\\\?\\d:\\work\\refact.ai\\refact-lsp\\.refact\\integrations\\postgres.yaml", + }, +]; + +export const MIXED_PAUSE_REASONS: ToolConfirmationPauseReason[] = [ + { + command: "SELECT *", + rule: "*", + type: "denial", + tool_call_id: "1", + integr_config_path: + "\\\\?\\d:\\work\\refact.ai\\refact-lsp\\.refact\\integrations\\postgres.yaml", + }, + { + command: "DROP *", + rule: "*", + type: "confirmation", + tool_call_id: "1", + integr_config_path: + "\\\\?\\d:\\work\\refact.ai\\refact-lsp\\.refact\\integrations\\postgres.yaml", + }, +]; diff --git a/src/components/ChatForm/ToolConfirmation.stories.tsx b/src/components/ChatForm/ToolConfirmation.stories.tsx new file mode 100644 index 00000000..757c3c04 --- /dev/null +++ b/src/components/ChatForm/ToolConfirmation.stories.tsx @@ -0,0 +1,75 @@ +import React from "react"; +import type { Meta, StoryObj } from "@storybook/react"; +import { ToolConfirmation } from "./ToolConfirmation"; +import { Provider } from "react-redux"; +import { setUpStore } from "../../app/store"; +import { Theme } from "../Theme"; +import { ToolConfirmationPauseReason } from "../../services/refact"; +import { AbortControllerProvider } from "../../contexts/AbortControllers"; +import { + CONFIRMATIONAL_PAUSE_REASONS, + CONFIRMATIONAL_PAUSE_REASONS_WITH_PATH, + DENIAL_PAUSE_REASONS_WITH_PATH, + MIXED_PAUSE_REASONS, +} from "../../__fixtures__/confirmation"; + +const MockedStore: React.FC<{ + pauseReasons: ToolConfirmationPauseReason[]; +}> = ({ pauseReasons }) => { + const store = setUpStore({ + confirmation: { + pauseReasons, + pause: true, + status: { + wasInteracted: false, + confirmationStatus: false, + }, + }, + }); + + return ( + + + + + + + + ); +}; + +const meta: Meta = { + title: "ToolConfirmation", + component: MockedStore, + args: { + pauseReasons: [], + }, +}; + +export default meta; + +type Story = StoryObj; + +export const Default: Story = { + args: { + pauseReasons: CONFIRMATIONAL_PAUSE_REASONS_WITH_PATH, + }, +}; + +export const WithDenial: Story = { + args: { + pauseReasons: DENIAL_PAUSE_REASONS_WITH_PATH, + }, +}; + +export const Patch: Story = { + args: { + pauseReasons: CONFIRMATIONAL_PAUSE_REASONS, + }, +}; + +export const Mixed: Story = { + args: { + pauseReasons: MIXED_PAUSE_REASONS, + }, +}; diff --git a/src/components/ChatForm/ToolConfirmation.tsx b/src/components/ChatForm/ToolConfirmation.tsx index dacc0cb3..9fe9d71b 100644 --- a/src/components/ChatForm/ToolConfirmation.tsx +++ b/src/components/ChatForm/ToolConfirmation.tsx @@ -100,7 +100,7 @@ export const ToolConfirmation: React.FC = ({ >{`${"```bash\n"}${command}${"\n```"}`} ))} - {message.concat("\n\n")} + {message.concat("\n\n")} {maybeIntegrationPath && ( You can modify the ruleset on{" "} @@ -114,6 +114,7 @@ export const ToolConfirmation: React.FC = ({ }), ); }} + color="indigo" > Configuration Page diff --git a/src/components/Markdown/CodeBlock.tsx b/src/components/Markdown/CodeBlock.tsx index 76da4350..7d8f075f 100644 --- a/src/components/Markdown/CodeBlock.tsx +++ b/src/components/Markdown/CodeBlock.tsx @@ -2,7 +2,7 @@ import React, { CSSProperties } from "react"; import SyntaxHighlighter, { type SyntaxHighlighterProps, } from "react-syntax-highlighter"; -import { Code, Text } from "@radix-ui/themes"; +import { Code, CodeProps, Text } from "@radix-ui/themes"; import classNames from "classnames"; import { PreTag, PreTagProps } from "./Pre"; // import "./highlightjs.css"; @@ -19,6 +19,7 @@ export type MarkdownCodeBlockProps = React.JSX.IntrinsicElements["code"] & { node?: Element | undefined; style?: Record | SyntaxHighlighterProps["style"]; wrap?: boolean; + color?: CodeProps["color"]; } & Pick< SyntaxHighlighterProps, "showLineNumbers" | "startingLineNumber" | "useInlineStyles" @@ -31,6 +32,7 @@ const _MarkdownCodeBlock: React.FC = ({ style = hljsStyle, onCopyClick, wrap = false, + color = undefined, }) => { const codeRef = React.useRef(null); const match = /language-(\w+)/.exec(className ?? ""); @@ -76,7 +78,10 @@ const _MarkdownCodeBlock: React.FC = ({ } return ( - + {children} ); diff --git a/src/components/Markdown/Markdown.tsx b/src/components/Markdown/Markdown.tsx index 6820ce2e..c5861c55 100644 --- a/src/components/Markdown/Markdown.tsx +++ b/src/components/Markdown/Markdown.tsx @@ -52,7 +52,11 @@ export type MarkdownProps = Pick< > & Pick< MarkdownCodeBlockProps, - "startingLineNumber" | "showLineNumbers" | "useInlineStyles" | "style" + | "startingLineNumber" + | "showLineNumbers" + | "useInlineStyles" + | "style" + | "color" > & { canHaveInteractiveElements?: boolean; wrap?: boolean; @@ -237,6 +241,7 @@ const _Markdown: React.FC = ({ allowedElements, unwrapDisallowed, canHaveInteractiveElements, + color, ...rest }) => { const components: Partial = useMemo(() => { @@ -251,8 +256,8 @@ const _Markdown: React.FC = ({
    ); }, - code({ style: _style, ...props }) { - return ; + code({ style: _style, color: _color, ...props }) { + return ; }, p({ color: _color, ref: _ref, node: _node, ...props }) { if (canHaveInteractiveElements) { @@ -329,7 +334,7 @@ const _Markdown: React.FC = ({ return ; }, }; - }, [rest, canHaveInteractiveElements]); + }, [rest, canHaveInteractiveElements, color]); return (