Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(autofix): Iterative version history for changes #82223

Draft
wants to merge 2 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@ function AutofixActionSelector<T extends string>({

const Container = styled('div')`
min-height: 40px;
padding: 0 ${space(1)};
`;

const ContentWrapper = styled('div')`
Expand Down
226 changes: 184 additions & 42 deletions static/app/components/events/autofix/autofixChanges.tsx
Original file line number Diff line number Diff line change
@@ -1,21 +1,29 @@
import {Fragment} from 'react';
import React, {Fragment, useEffect, useState} from 'react';
import styled from '@emotion/styled';
import {AnimatePresence, type AnimationProps, motion} from 'framer-motion';

import ClippedBox from 'sentry/components/clippedBox';
import {AutofixDiff} from 'sentry/components/events/autofix/autofixDiff';
import type {
AutofixChangesStep,
AutofixCodebaseChange,
import {SetupAndCreatePRsButton} from 'sentry/components/events/autofix/autofixPrButton';
import {AutofixViewPrButton} from 'sentry/components/events/autofix/autofixViewPrButton';
import {
type AutofixChangesStep,
type AutofixCodebaseChange,
AutofixStatus,
} from 'sentry/components/events/autofix/types';
import {useAutofixData} from 'sentry/components/events/autofix/useAutofix';
import {IconFix} from 'sentry/icons';
import LoadingIndicator from 'sentry/components/loadingIndicator';
import {IconChevron, IconFix} from 'sentry/icons';
import {t} from 'sentry/locale';
import {space} from 'sentry/styles/space';
import testableTransition from 'sentry/utils/testableTransition';
import usePrevious from 'sentry/utils/usePrevious';

type AutofixChangesProps = {
changesVersionIndex: number;
groupId: string;
hasMoreThanOneChangesStep: boolean;
hasStepBelow: boolean;
runId: string;
step: AutofixChangesStep;
};
Expand All @@ -30,12 +38,19 @@ function AutofixRepoChange({
runId: string;
}) {
return (
<Content>
<RepoChangeContent>
<RepoChangesHeader>
<div>
<Title>{change.title}</Title>
<PullRequestTitle>{change.repo_name}</PullRequestTitle>
</div>
{change.pull_request && (
<AutofixViewPrButton
repoName={change.repo_name}
prUrl={change.pull_request?.pr_url}
isPrimary={false}
/>
)}
</RepoChangesHeader>
<AutofixDiff
diff={change.diff}
Expand All @@ -44,7 +59,7 @@ function AutofixRepoChange({
repoId={change.repo_external_id}
editable={!change.pull_request}
/>
</Content>
</RepoChangeContent>
);
}

Expand All @@ -69,12 +84,29 @@ const cardAnimationProps: AnimationProps = {
}),
};

export function AutofixChanges({step, groupId, runId}: AutofixChangesProps) {
export function AutofixChanges({
step,
groupId,
runId,
hasStepBelow,
changesVersionIndex,
hasMoreThanOneChangesStep,
}: AutofixChangesProps) {
const data = useAutofixData({groupId});
const [isExpanded, setIsExpanded] = useState(!hasStepBelow);

const previousHasStepBelow = usePrevious(hasStepBelow);

useEffect(() => {
// When a new step shows up below this one, we want to auto collapse it.
if (previousHasStepBelow && !hasStepBelow) {
setIsExpanded(false);
}
}, [previousHasStepBelow, hasStepBelow]);

if (step.status === 'ERROR' || data?.status === 'ERROR') {
return (
<Content>
<RepoChangeContent>
<PreviewContent>
{data?.error_message ? (
<Fragment>
Expand All @@ -85,38 +117,93 @@ export function AutofixChanges({step, groupId, runId}: AutofixChangesProps) {
<span>{t('Something went wrong.')}</span>
)}
</PreviewContent>
</Content>
</RepoChangeContent>
);
}

if (!step.changes.length) {
if (
step.status === AutofixStatus.COMPLETED &&
Object.keys(step.codebase_changes).length === 0
) {
return (
<Content>
<RepoChangeContent>
<PreviewContent>
<span>{t('Could not find a fix.')}</span>
</PreviewContent>
</Content>
</RepoChangeContent>
);
}

const allChangesHavePullRequests = step.changes.every(change => change.pull_request);
const allChangesHavePullRequests = Object.values(step.codebase_changes).every(
change => change.pull_request
);

const changesText = hasMoreThanOneChangesStep
? hasStepBelow
? t('Changes (version %s)', changesVersionIndex + 1)
: t('Latest Changes (version %s)', changesVersionIndex + 1)
: t('Changes');

const changesAreReady = Object.values(step.codebase_changes).every(
change => change.diff_str
);

return (
<AnimatePresence initial>
<AnimationWrapper key="card" {...cardAnimationProps}>
<ChangesContainer allChangesHavePullRequests={allChangesHavePullRequests}>
<ClippedBox clipHeight={408}>
<HeaderText>
<IconFix size="sm" />
{t('Fixes')}
</HeaderText>
{step.changes.map((change, i) => (
<Fragment key={change.repo_external_id}>
{i > 0 && <Separator />}
<AutofixRepoChange change={change} groupId={groupId} runId={runId} />
</Fragment>
))}
</ClippedBox>
<ChangesContainer
allChangesHavePullRequests={allChangesHavePullRequests}
hasStepBelow={hasStepBelow}
>
<HeaderRow
onClick={() => hasStepBelow && setIsExpanded(!isExpanded)}
role={hasStepBelow ? 'button' : undefined}
aria-expanded={hasStepBelow ? isExpanded : undefined}
expandable={hasStepBelow}
>
<HeaderTextWrapper>
<HeaderText isPreviousChanges={hasStepBelow}>
<IconFix size="sm" />
{changesText}
</HeaderText>
</HeaderTextWrapper>
<HeaderButtonsWrapper>
{changesAreReady ? (
<React.Fragment>
{(!hasStepBelow || isExpanded) && !allChangesHavePullRequests && (
<SetupAndCreatePRsButton
changes={Object.values(step.codebase_changes)}
groupId={groupId}
hasStepBelow={hasStepBelow}
changesStepId={step.id}
isPrimary={false}
/>
)}
{hasStepBelow && (
<CollapseButton
aria-label={
isExpanded ? t('Collapse changes') : t('Expand changes')
}
>
<IconChevron direction={isExpanded ? 'up' : 'down'} size="sm" />
</CollapseButton>
)}
</React.Fragment>
) : (
<StyledLoadingIndicator size={14} mini hideMessage />
)}
</HeaderButtonsWrapper>
</HeaderRow>
{(!hasStepBelow || isExpanded) && changesAreReady && (
<ClippedBox clipHeight={408}>
{Object.values(step.codebase_changes).map((change, i) => (
<Fragment key={change.repo_external_id}>
{i > 0 && <Separator />}
<AutofixRepoChange change={change} groupId={groupId} runId={runId} />
</Fragment>
))}
</ClippedBox>
)}
</ChangesContainer>
</AnimationWrapper>
</AnimatePresence>
Expand All @@ -136,21 +223,24 @@ const AnimationWrapper = styled(motion.div)`

const PrefixText = styled('span')``;

const ChangesContainer = styled('div')<{allChangesHavePullRequests: boolean}>`
border: 2px solid
const ChangesContainer = styled('div')<{
allChangesHavePullRequests: boolean;
hasStepBelow?: boolean;
}>`
border: ${p => (p.hasStepBelow ? 1 : 2)}px solid
${p =>
p.allChangesHavePullRequests
? p.theme.alert.success.border
: p.theme.alert.info.border};
p.hasStepBelow
? p.theme.innerBorder
: p.allChangesHavePullRequests
? p.theme.alert.success.border
: p.theme.alert.info.border};
border-radius: ${p => p.theme.borderRadius};
box-shadow: ${p => p.theme.dropShadowMedium};
padding-left: ${space(2)};
padding-right: ${space(2)};
padding-top: ${space(1)};
overflow: hidden;
`;

const Content = styled('div')`
padding: 0 ${space(1)} ${space(1)} ${space(1)};
const RepoChangeContent = styled('div')`
padding: 0 ${space(2)} ${space(2)} ${space(2)};
`;

const Title = styled('div')`
Expand All @@ -164,9 +254,10 @@ const PullRequestTitle = styled('div')`

const RepoChangesHeader = styled('div')`
padding: ${space(2)} 0;
display: grid;
align-items: center;
grid-template-columns: 1fr auto;
display: flex;
align-items: flex-start;
justify-content: space-between;
gap: ${space(2)};
`;

const Separator = styled('hr')`
Expand All @@ -175,10 +266,61 @@ const Separator = styled('hr')`
margin: ${space(2)} -${space(2)} 0 -${space(2)};
`;

const HeaderText = styled('div')`
const HeaderRow = styled('div')<{expandable?: boolean}>`
display: flex;
align-items: center;
width: 100%;
justify-content: space-between;
gap: ${space(1)};
height: calc((2 * ${space(2)}) + ${p => p.theme.form.sm.height}px);
padding: ${space(2)};
cursor: ${p => (p.expandable ? 'pointer' : 'default')};

&:hover {
background-color: ${p =>
p.expandable ? p.theme.backgroundSecondary : 'transparent'};
}
`;

const HeaderButtonsWrapper = styled('div')`
display: flex;
align-items: center;
gap: ${space(2)};
`;

const StyledLoadingIndicator = styled(LoadingIndicator)`
&& {
margin: 0;
height: 16px;
width: 16px;
}
`;

const HeaderText = styled('div')<{isPreviousChanges?: boolean}>`
font-weight: bold;
font-size: 1.2em;
font-size: ${p => (p.isPreviousChanges ? 1.1 : 1.2)}em;
color: ${p => (p.isPreviousChanges ? p.theme.subText : p.theme.textColor)};
display: flex;
align-items: center;
gap: ${space(1)};
`;

const HeaderTextWrapper = styled('div')`
display: flex;
align-items: center;
gap: ${space(1)};
`;

const CollapseButton = styled('button')`
background: none;
border: none;
cursor: pointer;
padding: 0;
color: ${p => p.theme.subText};
display: flex;
align-items: center;

&:hover {
color: ${p => p.theme.textColor};
}
`;
33 changes: 33 additions & 0 deletions static/app/components/events/autofix/autofixInsightCards.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -412,6 +412,39 @@ export function useUpdateInsightCard({groupId, runId}: {groupId: string; runId:
});
}

export function useSendFeedbackOnChanges({
groupId,
runId,
}: {
groupId: string;
runId: string;
}) {
const api = useApi({persistInFlight: true});
const queryClient = useQueryClient();

return useMutation({
mutationFn: (params: {message: string}) => {
return api.requestPromise(`/issues/${groupId}/autofix/update/`, {
method: 'POST',
data: {
run_id: runId,
payload: {
type: 'continue_with_feedback',
message: params.message,
},
},
});
},
onSuccess: _ => {
queryClient.invalidateQueries({queryKey: makeAutofixQueryKey(groupId)});
addSuccessMessage(t('Thanks, rethinking this...'));
},
onError: () => {
addErrorMessage(t('Something went wrong when sending Autofix your message.'));
},
});
}

function ChainLink({
groupId,
runId,
Expand Down
Loading
Loading