aboutsummaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
authorDamien Urruty <damien.urruty@sonarsource.com>2024-11-08 19:59:25 +0100
committersonartech <sonartech@sonarsource.com>2024-11-19 20:02:54 +0000
commite0c824f58666eee38fa0239bce07aa81c1d30f44 (patch)
tree61271e48937d41fdd532dba159e9a2ad4634ace7 /server
parent12d6a58bf18b8996a52be8c30af3fc03c31a0a57 (diff)
downloadsonarqube-e0c824f58666eee38fa0239bce07aa81c1d30f44.tar.gz
sonarqube-e0c824f58666eee38fa0239bce07aa81c1d30f44.zip
CODEFIX-189 Allow admin to enable AI CodeFix at the project level
Diffstat (limited to 'server')
-rw-r--r--server/sonar-web/src/main/js/api/fix-suggestions.ts16
-rw-r--r--server/sonar-web/src/main/js/api/mocks/ComponentsServiceMock.ts5
-rw-r--r--server/sonar-web/src/main/js/api/mocks/FixSuggestionsServiceMock.ts (renamed from server/sonar-web/src/main/js/api/mocks/FixIssueServiceMock.ts)9
-rw-r--r--server/sonar-web/src/main/js/api/mocks/data/projects.ts1
-rw-r--r--server/sonar-web/src/main/js/apps/issues/test-utils.tsx4
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/AdditionalCategories.tsx14
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/AiCodeFixAdmin.tsx526
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/CodeFixAdmin.tsx298
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/__tests__/AiCodeFixAdmin-it.tsx291
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/__tests__/CodeFixAdmin-it.tsx195
-rw-r--r--server/sonar-web/src/main/js/apps/settings/constants.ts2
-rw-r--r--server/sonar-web/src/main/js/components/rules/IssueSuggestionFileSnippet.tsx1
-rw-r--r--server/sonar-web/src/main/js/helpers/mocks/component.ts12
-rw-r--r--server/sonar-web/src/main/js/queries/fix-suggestions.tsx7
-rw-r--r--server/sonar-web/src/main/js/types/fix-suggestions.ts6
15 files changed, 882 insertions, 505 deletions
diff --git a/server/sonar-web/src/main/js/api/fix-suggestions.ts b/server/sonar-web/src/main/js/api/fix-suggestions.ts
index 04eba994fc5..a337e8be7aa 100644
--- a/server/sonar-web/src/main/js/api/fix-suggestions.ts
+++ b/server/sonar-web/src/main/js/api/fix-suggestions.ts
@@ -19,7 +19,7 @@
*/
import { axiosToCatch } from '../helpers/request';
-import { SuggestedFix } from '../types/fix-suggestions';
+import { AiCodeFixFeatureEnablement, SuggestedFix } from '../types/fix-suggestions';
export interface FixParam {
issueId: string;
@@ -41,6 +41,14 @@ export interface SuggestionServiceStatusCheckResponse {
status: SuggestionServiceStatus;
}
+export interface UpdateFeatureEnablementParams {
+ changes: {
+ disabledProjectKeys: string[];
+ enabledProjectKeys: string[];
+ };
+ enablement: AiCodeFixFeatureEnablement;
+}
+
export function getSuggestions(data: FixParam): Promise<SuggestedFix> {
return axiosToCatch.post<SuggestedFix>('/api/v2/fix-suggestions/ai-suggestions', data);
}
@@ -52,3 +60,9 @@ export function getFixSuggestionsIssues(data: FixParam): Promise<AiIssue> {
export function checkSuggestionServiceStatus(): Promise<SuggestionServiceStatusCheckResponse> {
return axiosToCatch.post(`/api/v2/fix-suggestions/service-status-checks`);
}
+
+export function updateFeatureEnablement(
+ featureEnablementParams: UpdateFeatureEnablementParams,
+): Promise<void> {
+ return axiosToCatch.patch(`/api/v2/fix-suggestions/feature-enablements`, featureEnablementParams);
+}
diff --git a/server/sonar-web/src/main/js/api/mocks/ComponentsServiceMock.ts b/server/sonar-web/src/main/js/api/mocks/ComponentsServiceMock.ts
index 55cf9a0878d..ff3e47932e9 100644
--- a/server/sonar-web/src/main/js/api/mocks/ComponentsServiceMock.ts
+++ b/server/sonar-web/src/main/js/api/mocks/ComponentsServiceMock.ts
@@ -120,6 +120,7 @@ export default class ComponentsServiceMock {
const query = data.filter.split('query=')[1];
return c.key.includes(query) || c.name.includes(query);
}
+ return true;
})
.map((c) => c);
@@ -166,6 +167,10 @@ export default class ComponentsServiceMock {
throw new Error(`Couldn't find source file for key ${key}`);
};
+ registerProject = (project: ComponentRaw) => {
+ this.projects.push(project);
+ };
+
registerComponent = (component: Component, ancestors: Component[] = []) => {
this.components.push({ component, ancestors, children: [] });
};
diff --git a/server/sonar-web/src/main/js/api/mocks/FixIssueServiceMock.ts b/server/sonar-web/src/main/js/api/mocks/FixSuggestionsServiceMock.ts
index 4ea23011cab..f46b2c36ac3 100644
--- a/server/sonar-web/src/main/js/api/mocks/FixIssueServiceMock.ts
+++ b/server/sonar-web/src/main/js/api/mocks/FixSuggestionsServiceMock.ts
@@ -26,6 +26,8 @@ import {
getSuggestions,
SuggestionServiceStatus,
SuggestionServiceStatusCheckResponse,
+ updateFeatureEnablement,
+ UpdateFeatureEnablementParams,
} from '../fix-suggestions';
import { ISSUE_101, ISSUE_1101 } from './data/ids';
@@ -33,7 +35,7 @@ jest.mock('../fix-suggestions');
export type MockSuggestionServiceStatus = SuggestionServiceStatus | 'WTF' | undefined;
-export default class FixIssueServiceMock {
+export default class FixSuggestionsServiceMock {
fixSuggestion = {
id: '70b14d4c-d302-4979-9121-06ac7d563c5c',
issueId: 'AYsVhClEbjXItrbcN71J',
@@ -54,6 +56,7 @@ export default class FixIssueServiceMock {
jest.mocked(getSuggestions).mockImplementation(this.handleGetFixSuggestion);
jest.mocked(getFixSuggestionsIssues).mockImplementation(this.handleGetFixSuggestionsIssues);
jest.mocked(checkSuggestionServiceStatus).mockImplementation(this.handleCheckService);
+ jest.mocked(updateFeatureEnablement).mockImplementation(this.handleUpdateFeatureEnablement);
}
handleGetFixSuggestionsIssues = (data: FixParam) => {
@@ -77,6 +80,10 @@ export default class FixIssueServiceMock {
return Promise.reject({ error: { msg: 'Error' } });
};
+ handleUpdateFeatureEnablement = (_: UpdateFeatureEnablementParams) => {
+ return Promise.resolve();
+ };
+
reply<T>(response: T): Promise<T> {
return new Promise((resolve) => {
setTimeout(() => {
diff --git a/server/sonar-web/src/main/js/api/mocks/data/projects.ts b/server/sonar-web/src/main/js/api/mocks/data/projects.ts
index fbffae15233..edbd6d42c29 100644
--- a/server/sonar-web/src/main/js/api/mocks/data/projects.ts
+++ b/server/sonar-web/src/main/js/api/mocks/data/projects.ts
@@ -133,6 +133,7 @@ export function mockProjects(): ComponentRaw[] {
tags: ['sonarqube'],
visibility: Visibility.Public,
leakPeriodDate: '2023-08-10T12:28:45+0000',
+ isAiCodeFixEnabled: true,
},
{
key: 'org.sonarsource.javascript:javascript',
diff --git a/server/sonar-web/src/main/js/apps/issues/test-utils.tsx b/server/sonar-web/src/main/js/apps/issues/test-utils.tsx
index dc6200c22c5..e0e8347b482 100644
--- a/server/sonar-web/src/main/js/apps/issues/test-utils.tsx
+++ b/server/sonar-web/src/main/js/apps/issues/test-utils.tsx
@@ -24,7 +24,7 @@ import { byPlaceholderText, byRole, byTestId, byText } from '~sonar-aligned/help
import BranchesServiceMock from '../../api/mocks/BranchesServiceMock';
import ComponentsServiceMock from '../../api/mocks/ComponentsServiceMock';
import CveServiceMock from '../../api/mocks/CveServiceMock';
-import FixIssueServiceMock from '../../api/mocks/FixIssueServiceMock';
+import FixSuggestionsServiceMock from '../../api/mocks/FixSuggestionsServiceMock';
import IssuesServiceMock from '../../api/mocks/IssuesServiceMock';
import SettingsServiceMock from '../../api/mocks/SettingsServiceMock';
import SourcesServiceMock from '../../api/mocks/SourcesServiceMock';
@@ -50,7 +50,7 @@ export const cveHandler = new CveServiceMock();
export const componentsHandler = new ComponentsServiceMock();
export const sourcesHandler = new SourcesServiceMock();
export const branchHandler = new BranchesServiceMock();
-export const fixIssueHandler = new FixIssueServiceMock();
+export const fixIssueHandler = new FixSuggestionsServiceMock();
export const settingsHandler = new SettingsServiceMock();
export const ui = {
diff --git a/server/sonar-web/src/main/js/apps/settings/components/AdditionalCategories.tsx b/server/sonar-web/src/main/js/apps/settings/components/AdditionalCategories.tsx
index 015df89393c..ffd88b6893b 100644
--- a/server/sonar-web/src/main/js/apps/settings/components/AdditionalCategories.tsx
+++ b/server/sonar-web/src/main/js/apps/settings/components/AdditionalCategories.tsx
@@ -23,18 +23,18 @@ import { translate } from '../../../helpers/l10n';
import { ExtendedSettingDefinition } from '../../../types/settings';
import { Component } from '../../../types/types';
import {
+ AI_CODE_FIX_CATEGORY,
ALM_INTEGRATION_CATEGORY,
ANALYSIS_SCOPE_CATEGORY,
AUTHENTICATION_CATEGORY,
- CODE_FIX_CATEGORY,
EMAIL_NOTIFICATION_CATEGORY,
LANGUAGES_CATEGORY,
MODE_CATEGORY,
NEW_CODE_PERIOD_CATEGORY,
PULL_REQUEST_DECORATION_BINDING_CATEGORY,
} from '../constants';
+import AiCodeFixAdmin from './AiCodeFixAdmin';
import { AnalysisScope } from './AnalysisScope';
-import CodeFixAdmin from './CodeFixAdmin';
import Languages from './Languages';
import { Mode } from './Mode';
import NewCodeDefinition from './NewCodeDefinition';
@@ -94,9 +94,9 @@ export const ADDITIONAL_CATEGORIES: AdditionalCategory[] = [
displayTab: true,
},
{
- key: CODE_FIX_CATEGORY,
- name: translate('property.category.codefix'),
- renderComponent: getCodeFixComponent,
+ key: AI_CODE_FIX_CATEGORY,
+ name: translate('property.category.aicodefix'),
+ renderComponent: getAiCodeFixComponent,
availableGlobally: true,
availableForProject: false,
displayTab: true,
@@ -152,8 +152,8 @@ function getAlmIntegrationComponent(props: AdditionalCategoryComponentProps) {
return <AlmIntegration {...props} />;
}
-function getCodeFixComponent(props: AdditionalCategoryComponentProps) {
- return <CodeFixAdmin {...props} />;
+function getAiCodeFixComponent(props: AdditionalCategoryComponentProps) {
+ return <AiCodeFixAdmin {...props} />;
}
function getAuthenticationComponent(props: AdditionalCategoryComponentProps) {
diff --git a/server/sonar-web/src/main/js/apps/settings/components/AiCodeFixAdmin.tsx b/server/sonar-web/src/main/js/apps/settings/components/AiCodeFixAdmin.tsx
new file mode 100644
index 00000000000..f5a9526bf24
--- /dev/null
+++ b/server/sonar-web/src/main/js/apps/settings/components/AiCodeFixAdmin.tsx
@@ -0,0 +1,526 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2024 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+
+import styled from '@emotion/styled';
+import {
+ Button,
+ ButtonVariety,
+ Checkbox,
+ Heading,
+ IconCheckCircle,
+ IconError,
+ IconInfo,
+ Link,
+ RadioButtonGroup,
+ Spinner,
+ Text,
+} from '@sonarsource/echoes-react';
+import { MutationStatus } from '@tanstack/react-query';
+import { AxiosError } from 'axios';
+import React, { useEffect } from 'react';
+import { FormattedMessage } from 'react-intl';
+import {
+ BasicSeparator,
+ HighlightedSection,
+ Note,
+ themeColor,
+ UnorderedList,
+} from '~design-system';
+import { throwGlobalError } from '~sonar-aligned/helpers/error';
+import { searchProjects } from '../../../api/components';
+import { SuggestionServiceStatusCheckResponse } from '../../../api/fix-suggestions';
+import withAvailableFeatures, {
+ WithAvailableFeaturesProps,
+} from '../../../app/components/available-features/withAvailableFeatures';
+import DocumentationLink from '../../../components/common/DocumentationLink';
+import SelectList, {
+ SelectListFilter,
+ SelectListSearchParams,
+} from '../../../components/controls/SelectList';
+import { DocLink } from '../../../helpers/doc-links';
+import { translate } from '../../../helpers/l10n';
+import { getAiCodeFixTermsOfServiceUrl } from '../../../helpers/urls';
+import {
+ useCheckServiceMutation,
+ useRemoveCodeSuggestionsCache,
+ useUpdateFeatureEnablementMutation,
+} from '../../../queries/fix-suggestions';
+import { useGetValueQuery } from '../../../queries/settings';
+import { Feature } from '../../../types/features';
+import { AiCodeFixFeatureEnablement } from '../../../types/fix-suggestions';
+import { SettingsKey } from '../../../types/settings';
+import PromotedSection from '../../overview/branches/PromotedSection';
+
+interface Props extends WithAvailableFeaturesProps {}
+
+const AI_CODE_FIX_SETTING_KEY = SettingsKey.CodeSuggestion;
+
+function AiCodeFixAdmin({ hasFeature }: Readonly<Props>) {
+ const { data: aiCodeFixSetting } = useGetValueQuery({
+ key: AI_CODE_FIX_SETTING_KEY,
+ });
+ const removeCodeSuggestionsCache = useRemoveCodeSuggestionsCache();
+
+ const initialAiCodeFixEnablement =
+ (aiCodeFixSetting?.value as AiCodeFixFeatureEnablement) || AiCodeFixFeatureEnablement.disabled;
+
+ const [savedAiCodeFixEnablement, setSavedAiCodeFixEnablement] = React.useState(
+ initialAiCodeFixEnablement,
+ );
+ const [currentAiCodeFixEnablement, setCurrentAiCodeFixEnablement] =
+ React.useState(savedAiCodeFixEnablement);
+ const {
+ mutate: checkService,
+ isIdle,
+ isPending: isServiceCheckPending,
+ status,
+ error,
+ data,
+ } = useCheckServiceMutation();
+
+ const { mutate: updateFeatureEnablement } = useUpdateFeatureEnablementMutation();
+ const [changedProjects, setChangedProjects] = React.useState<Map<string, boolean>>(new Map());
+
+ useEffect(() => {
+ setSavedAiCodeFixEnablement(initialAiCodeFixEnablement);
+ }, [initialAiCodeFixEnablement]);
+
+ useEffect(() => {
+ setCurrentAiCodeFixEnablement(savedAiCodeFixEnablement);
+ }, [savedAiCodeFixEnablement]);
+
+ const [currentSearchResults, setCurrentSearchResults] = React.useState<ProjectSearchResult>();
+ const [currentTabItems, setCurrentTabItems] = React.useState<ProjectItem[]>([]);
+
+ const handleSave = () => {
+ updateFeatureEnablement(
+ {
+ enablement: currentAiCodeFixEnablement,
+ changes: {
+ enabledProjectKeys: [...changedProjects]
+ .filter(([_, enabled]) => enabled)
+ .map(([project]) => project),
+ disabledProjectKeys: [...changedProjects]
+ .filter(([_, enabled]) => !enabled)
+ .map(([project]) => project),
+ },
+ },
+ {
+ onSuccess: () => {
+ removeCodeSuggestionsCache();
+ const savedChanges = changedProjects;
+ setChangedProjects(new Map());
+ setSavedAiCodeFixEnablement(currentAiCodeFixEnablement);
+ if (currentSearchResults) {
+ // some items might not be in the right tab if they were toggled just before saving, we need to refresh the view
+ updateItemsWithSearchResult(currentSearchResults, savedChanges);
+ }
+ },
+ },
+ );
+ };
+
+ const handleCancel = () => {
+ setCurrentAiCodeFixEnablement(savedAiCodeFixEnablement);
+ setChangedProjects(new Map());
+ if (currentSearchResults) {
+ // some items might have moved to another tab than the current one, we need to refresh the view
+ updateItemsWithSearchResult(currentSearchResults, new Map());
+ }
+ };
+
+ if (!hasFeature(Feature.FixSuggestions)) {
+ return null;
+ }
+
+ const renderProjectElement = (projectKey: string): React.ReactNode => {
+ const project = currentTabItems.find((project) => project.key === projectKey);
+ return (
+ <div>
+ {project === undefined ? (
+ projectKey
+ ) : (
+ <>
+ {project.name}
+ <br />
+ <Note>{project.key}</Note>
+ </>
+ )}
+ </div>
+ );
+ };
+
+ const onProjectSelected = (projectKey: string) => {
+ const newChangedProjects = new Map(changedProjects);
+ newChangedProjects.set(projectKey, true);
+ setChangedProjects(newChangedProjects);
+ const project = currentTabItems.find((project) => project.key === projectKey);
+ if (project) {
+ project.selected = true;
+ setCurrentTabItems([...currentTabItems]);
+ }
+ return Promise.resolve();
+ };
+
+ const onProjectUnselected = (projectKey: string) => {
+ const newChangedProjects = new Map(changedProjects);
+ newChangedProjects.set(projectKey, false);
+ setChangedProjects(newChangedProjects);
+ const project = currentTabItems.find((project) => project.key === projectKey);
+ if (project) {
+ project.selected = false;
+ setCurrentTabItems([...currentTabItems]);
+ }
+ return Promise.resolve();
+ };
+
+ const onSearch = (searchParams: SelectListSearchParams) => {
+ searchProjects({
+ p: searchParams.page,
+ filter: searchParams.query !== '' ? `query=${searchParams.query}` : undefined,
+ })
+ .then((response) => {
+ const searchResults = {
+ filter: searchParams.filter,
+ projects: response.components.map((project) => {
+ return {
+ key: project.key,
+ name: project.name,
+ isAiCodeFixEnabled: project.isAiCodeFixEnabled === true,
+ };
+ }),
+ totalCount: response.paging.total,
+ };
+ setCurrentSearchResults(searchResults);
+ updateItemsWithSearchResult(searchResults, changedProjects);
+ })
+ .catch(throwGlobalError);
+ };
+
+ const updateItemsWithSearchResult = (
+ searchResult: ProjectSearchResult,
+ changedProjects: Map<string, boolean>,
+ ) => {
+ const { filter } = searchResult;
+ setCurrentTabItems(
+ searchResult.projects
+ .filter(
+ (project) =>
+ filter === SelectListFilter.All ||
+ (filter === SelectListFilter.Selected &&
+ (changedProjects.has(project.key)
+ ? changedProjects.get(project.key) === true
+ : project.isAiCodeFixEnabled)) ||
+ (filter === SelectListFilter.Unselected &&
+ (changedProjects.has(project.key)
+ ? !changedProjects.get(project.key)
+ : !project.isAiCodeFixEnabled)),
+ )
+ .map((project) => {
+ return {
+ key: project.key,
+ name: project.name,
+ selected: changedProjects.has(project.key)
+ ? changedProjects.get(project.key) === true
+ : project.isAiCodeFixEnabled,
+ };
+ }),
+ );
+ };
+
+ return (
+ <div className="sw-flex">
+ <div className="sw-flex-grow sw-p-6">
+ <Heading as="h2" hasMarginBottom>
+ {translate('property.aicodefix.admin.title')}
+ </Heading>
+ <PromotedSection
+ content={
+ <>
+ <p>{translate('property.aicodefix.admin.promoted_section.content1')}</p>
+ <p className="sw-mt-2">
+ {translate('property.aicodefix.admin.promoted_section.content2')}
+ </p>
+ </>
+ }
+ title={translate('property.aicodefix.admin.promoted_section.title')}
+ />
+ <p>{translate('property.aicodefix.admin.description')}</p>
+ <Checkbox
+ className="sw-my-6"
+ label={translate('property.aicodefix.admin.checkbox.label')}
+ checked={currentAiCodeFixEnablement !== AiCodeFixFeatureEnablement.disabled}
+ onCheck={() =>
+ setCurrentAiCodeFixEnablement(
+ currentAiCodeFixEnablement === AiCodeFixFeatureEnablement.disabled
+ ? AiCodeFixFeatureEnablement.allProjects
+ : AiCodeFixFeatureEnablement.disabled,
+ )
+ }
+ helpText={
+ <FormattedMessage
+ id="property.aicodefix.admin.terms"
+ defaultMessage={translate('property.aicodefix.admin.acceptTerm.label')}
+ values={{
+ terms: (
+ <Link shouldOpenInNewTab to={getAiCodeFixTermsOfServiceUrl()}>
+ {translate('property.aicodefix.admin.acceptTerm.terms')}
+ </Link>
+ ),
+ }}
+ />
+ }
+ />
+ <div className="sw-ml-6">
+ {currentAiCodeFixEnablement !== AiCodeFixFeatureEnablement.disabled && (
+ <RadioButtonGroup
+ label={translate('property.aicodefix.admin.enable.title')}
+ id="ai-code-fix-enablement"
+ isRequired
+ options={[
+ {
+ helpText: translate('property.aicodefix.admin.enable.all.projects.help'),
+ label: translate('property.aicodefix.admin.enable.all.projects.label'),
+ value: AiCodeFixFeatureEnablement.allProjects,
+ },
+ {
+ helpText: translate('property.aicodefix.admin.enable.some.projects.help'),
+ label: translate('property.aicodefix.admin.enable.some.projects.label'),
+ value: AiCodeFixFeatureEnablement.someProjects,
+ },
+ ]}
+ value={currentAiCodeFixEnablement}
+ onChange={(enablement: AiCodeFixFeatureEnablement) =>
+ setCurrentAiCodeFixEnablement(enablement)
+ }
+ />
+ )}
+ {currentAiCodeFixEnablement === AiCodeFixFeatureEnablement.someProjects && (
+ <div className="sw-ml-6">
+ <div className="sw-flex sw-mb-6">
+ <IconInfo className="sw-mr-1" color="echoes-color-icon-info" />
+ <Text>{translate('property.aicodefix.admin.enable.some.projects.note')}</Text>
+ </div>
+ <SelectList
+ loading={false}
+ elements={currentTabItems.map((project) => project.key)}
+ elementsTotalCount={currentSearchResults?.totalCount}
+ labelAll={translate('all')}
+ labelSelected={translate('selected')}
+ labelUnselected={translate('unselected')}
+ needToReload={false}
+ onSearch={onSearch}
+ onSelect={onProjectSelected}
+ onUnselect={onProjectUnselected}
+ renderElement={renderProjectElement}
+ selectedElements={currentTabItems.filter((p) => p.selected).map((u) => u.key)}
+ withPaging
+ />
+ </div>
+ )}
+ </div>
+ <div>
+ <div className="sw-mt-6">
+ <Button
+ variety={ButtonVariety.Primary}
+ isDisabled={
+ currentAiCodeFixEnablement === savedAiCodeFixEnablement &&
+ (currentAiCodeFixEnablement !== AiCodeFixFeatureEnablement.someProjects ||
+ changedProjects.size === 0)
+ }
+ onClick={() => {
+ handleSave();
+ }}
+ >
+ {translate('save')}
+ </Button>
+ <Button className="sw-ml-3" variety={ButtonVariety.Default} onClick={handleCancel}>
+ {translate('cancel')}
+ </Button>
+ </div>
+ </div>
+ </div>
+ <div className="sw-flex-col sw-w-abs-600 sw-p-6">
+ <HighlightedSection className="sw-items-start">
+ <Heading as="h3" hasMarginBottom>
+ {translate('property.aicodefix.admin.serviceCheck.title')}
+ </Heading>
+ <p>{translate('property.aicodefix.admin.serviceCheck.description1')}</p>
+ <DocumentationLink to={DocLink.AiCodeFixEnabling}>
+ {translate('property.aicodefix.admin.serviceCheck.learnMore')}
+ </DocumentationLink>
+ <p>{translate('property.aicodefix.admin.serviceCheck.description2')}</p>
+ <Button
+ className="sw-mt-4"
+ variety={ButtonVariety.Default}
+ onClick={() => checkService()}
+ isDisabled={isServiceCheckPending}
+ >
+ {translate('property.aicodefix.admin.serviceCheck.action')}
+ </Button>
+ {!isIdle && (
+ <div>
+ <BasicSeparator className="sw-my-4" />
+ <ServiceCheckResultView data={data} error={error} status={status} />
+ </div>
+ )}
+ </HighlightedSection>
+ </div>
+ </div>
+ );
+}
+
+interface ProjectSearchResult {
+ filter: SelectListFilter;
+ projects: RemoteProject[];
+ totalCount: number;
+}
+
+interface RemoteProject {
+ isAiCodeFixEnabled: boolean;
+ key: string;
+ name: string;
+}
+
+interface ProjectItem {
+ key: string;
+ name: string;
+ selected: boolean;
+}
+
+interface ServiceCheckResultViewProps {
+ data: SuggestionServiceStatusCheckResponse | undefined;
+ error: AxiosError | null;
+ status: MutationStatus;
+}
+
+function ServiceCheckResultView({ data, error, status }: Readonly<ServiceCheckResultViewProps>) {
+ switch (status) {
+ case 'pending':
+ return <Spinner label={translate('property.aicodefix.admin.serviceCheck.spinner.label')} />;
+ case 'error':
+ return (
+ <ErrorMessage
+ text={`${translate('property.aicodefix.admin.serviceCheck.result.requestError')} ${error?.status ?? 'No status'}`}
+ />
+ );
+ case 'success':
+ return ServiceCheckValidResponseView(data);
+ }
+ // normally unreachable
+ throw Error(`Unexpected response from the service status check, received ${status}`);
+}
+
+function ServiceCheckValidResponseView(data: SuggestionServiceStatusCheckResponse | undefined) {
+ switch (data?.status) {
+ case 'SUCCESS':
+ return (
+ <SuccessMessage text={translate('property.aicodefix.admin.serviceCheck.result.success')} />
+ );
+ case 'TIMEOUT':
+ case 'CONNECTION_ERROR':
+ return (
+ <div className="sw-flex">
+ <IconError className="sw-mr-1" color="echoes-color-icon-danger" />
+ <div className="sw-flex-col">
+ <ErrorLabel
+ text={translate('property.aicodefix.admin.serviceCheck.result.unresponsive.message')}
+ />
+ <p className="sw-mt-4">
+ <ErrorLabel
+ text={translate(
+ 'property.aicodefix.admin.serviceCheck.result.unresponsive.causes.title',
+ )}
+ />
+ </p>
+ <UnorderedList className="sw-ml-8" ticks>
+ <ErrorListItem className="sw-mb-2">
+ <ErrorLabel
+ text={translate(
+ 'property.aicodefix.admin.serviceCheck.result.unresponsive.causes.1',
+ )}
+ />
+ </ErrorListItem>
+ <ErrorListItem>
+ <ErrorLabel
+ text={translate(
+ 'property.aicodefix.admin.serviceCheck.result.unresponsive.causes.2',
+ )}
+ />
+ </ErrorListItem>
+ </UnorderedList>
+ </div>
+ </div>
+ );
+ case 'UNAUTHORIZED':
+ return (
+ <ErrorMessage
+ text={translate('property.aicodefix.admin.serviceCheck.result.unauthorized')}
+ />
+ );
+ case 'SERVICE_ERROR':
+ return (
+ <ErrorMessage
+ text={translate('property.aicodefix.admin.serviceCheck.result.serviceError')}
+ />
+ );
+ default:
+ return (
+ <ErrorMessage
+ text={`${translate('property.aicodefix.admin.serviceCheck.result.unknown')} ${data?.status ?? 'no status returned from the service'}`}
+ />
+ );
+ }
+}
+
+function ErrorMessage({ text }: Readonly<TextProps>) {
+ return (
+ <div className="sw-flex">
+ <IconError className="sw-mr-1" color="echoes-color-icon-danger" />
+ <ErrorLabel text={text} />
+ </div>
+ );
+}
+
+function ErrorLabel({ text }: Readonly<TextProps>) {
+ return <Text colorOverride="echoes-color-text-danger">{text}</Text>;
+}
+
+function SuccessMessage({ text }: Readonly<TextProps>) {
+ return (
+ <div className="sw-flex">
+ <IconCheckCircle className="sw-mr-1" color="echoes-color-icon-success" />
+ <Text colorOverride="echoes-color-text-success">{text}</Text>
+ </div>
+ );
+}
+
+const ErrorListItem = styled.li`
+ ::marker {
+ color: ${themeColor('errorText')};
+ }
+`;
+
+interface TextProps {
+ /** The text to display inside the component */
+ text: string;
+}
+
+export default withAvailableFeatures(AiCodeFixAdmin);
diff --git a/server/sonar-web/src/main/js/apps/settings/components/CodeFixAdmin.tsx b/server/sonar-web/src/main/js/apps/settings/components/CodeFixAdmin.tsx
deleted file mode 100644
index 486aeebeeb5..00000000000
--- a/server/sonar-web/src/main/js/apps/settings/components/CodeFixAdmin.tsx
+++ /dev/null
@@ -1,298 +0,0 @@
-/*
- * SonarQube
- * Copyright (C) 2009-2024 SonarSource SA
- * mailto:info AT sonarsource DOT com
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 3 of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public License
- * along with this program; if not, write to the Free Software Foundation,
- * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
-
-import styled from '@emotion/styled';
-import {
- Button,
- ButtonVariety,
- Checkbox,
- Heading,
- IconCheckCircle,
- IconError,
- Link,
- Spinner,
- Text
-} from '@sonarsource/echoes-react';
-import { MutationStatus } from '@tanstack/react-query';
-import { AxiosError } from 'axios';
-import React, { useEffect } from 'react';
-import { FormattedMessage } from 'react-intl';
-import { BasicSeparator, HighlightedSection, themeColor, UnorderedList } from '~design-system';
-import { SuggestionServiceStatusCheckResponse } from '../../../api/fix-suggestions';
-import withAvailableFeatures, {
- WithAvailableFeaturesProps
-} from '../../../app/components/available-features/withAvailableFeatures';
-import DocumentationLink from '../../../components/common/DocumentationLink';
-import { DocLink } from '../../../helpers/doc-links';
-import { translate } from '../../../helpers/l10n';
-import { getAiCodeFixTermsOfServiceUrl } from '../../../helpers/urls';
-import { useCheckServiceMutation, useRemoveCodeSuggestionsCache } from '../../../queries/fix-suggestions';
-import { useGetValueQuery, useSaveSimpleValueMutation } from '../../../queries/settings';
-import { Feature } from '../../../types/features';
-import { SettingsKey } from '../../../types/settings';
-import PromotedSection from '../../overview/branches/PromotedSection';
-
-interface Props extends WithAvailableFeaturesProps {}
-
-const CODE_FIX_SETTING_KEY = SettingsKey.CodeSuggestion;
-
-function CodeFixAdmin({ hasFeature }: Readonly<Props>) {
- const { data: codeFixSetting } = useGetValueQuery({
- key: CODE_FIX_SETTING_KEY,
- });
-
- const removeCodeSuggestionsCache = useRemoveCodeSuggestionsCache();
-
- const { mutate: saveSetting } = useSaveSimpleValueMutation();
-
- const isCodeFixEnabled = codeFixSetting?.value === 'true';
-
- const [enableCodeFix, setEnableCodeFix] = React.useState(isCodeFixEnabled);
- const {
- mutate: checkService,
- isIdle,
- isPending,
- status,
- error,
- data,
- } = useCheckServiceMutation();
- const isValueChanged = enableCodeFix !== isCodeFixEnabled;
-
- useEffect(() => {
- setEnableCodeFix(isCodeFixEnabled);
- }, [isCodeFixEnabled]);
-
- const handleSave = () => {
- saveSetting(
- { key: CODE_FIX_SETTING_KEY, value: enableCodeFix ? 'true' : 'false' },
- {
- onSuccess: removeCodeSuggestionsCache,
- },
- );
- };
-
- const handleCancel = () => {
- setEnableCodeFix(isCodeFixEnabled);
- };
-
- if (!hasFeature(Feature.FixSuggestions)) {
- return null;
- }
-
- return (
- <div className="sw-flex">
- <div className="sw-flex-grow sw-p-6">
- <Heading as="h2" hasMarginBottom>
- {translate('property.codefix.admin.title')}
- </Heading>
- <PromotedSection
- content={
- <>
- <p>{translate('property.codefix.admin.promoted_section.content1')}</p>
- <p className="sw-mt-2">
- {translate('property.codefix.admin.promoted_section.content2')}
- </p>
- </>
- }
- title={translate('property.codefix.admin.promoted_section.title')}
- />
- <p>{translate('property.codefix.admin.description')}</p>
- <Checkbox
- className="sw-mt-6"
- label={translate('property.codefix.admin.checkbox.label')}
- checked={Boolean(enableCodeFix)}
- onCheck={() => setEnableCodeFix(!enableCodeFix)}
- helpText={
- <FormattedMessage
- id="property.codefix.admin.terms"
- defaultMessage={translate('property.codefix.admin.acceptTerm.label')}
- values={{
- terms: (
- <Link shouldOpenInNewTab to={getAiCodeFixTermsOfServiceUrl()}>
- {translate('property.codefix.admin.acceptTerm.terms')}
- </Link>
- ),
- }}
- />
- }
- />
- <div>
- <BasicSeparator className="sw-mt-6" />
- <div className="sw-mt-6">
- <Button
- variety={ButtonVariety.Primary}
- isDisabled={!isValueChanged}
- onClick={() => {
- handleSave();
- }}
- >
- {translate('save')}
- </Button>
- <Button className="sw-ml-3" variety={ButtonVariety.Default} onClick={handleCancel}>
- {translate('cancel')}
- </Button>
- </div>
- </div>
- </div>
- <div className="sw-flex-col sw-w-abs-600 sw-p-6">
- <HighlightedSection className="sw-items-start">
- <Heading as="h3" hasMarginBottom>
- {translate('property.codefix.admin.serviceCheck.title')}
- </Heading>
- <p>{translate('property.codefix.admin.serviceCheck.description1')}</p>
- <DocumentationLink to={DocLink.AiCodeFixEnabling}>
- {translate('property.codefix.admin.serviceCheck.learnMore')}
- </DocumentationLink>
- <p>{translate('property.codefix.admin.serviceCheck.description2')}</p>
- <Button
- className="sw-mt-4"
- variety={ButtonVariety.Default}
- onClick={() => checkService()}
- isDisabled={isPending}
- >
- {translate('property.codefix.admin.serviceCheck.action')}
- </Button>
- {!isIdle && (
- <div>
- <BasicSeparator className="sw-my-4" />
- <ServiceCheckResultView data={data} error={error} status={status} />
- </div>
- )}
- </HighlightedSection>
- </div>
- </div>
- );
-}
-
-interface ServiceCheckResultViewProps {
- data: SuggestionServiceStatusCheckResponse | undefined;
- error: AxiosError | null;
- status: MutationStatus;
-}
-
-function ServiceCheckResultView({ data, error, status }: Readonly<ServiceCheckResultViewProps>) {
- switch (status) {
- case 'pending':
- return <Spinner label={translate('property.codefix.admin.serviceCheck.spinner.label')} />;
- case 'error':
- return (
- <ErrorMessage
- text={`${translate('property.codefix.admin.serviceCheck.result.requestError')} ${error?.status ?? 'No status'}`}
- />
- );
- case 'success':
- return ServiceCheckValidResponseView(data);
- }
- // normally unreachable
- throw Error(`Unexpected response from the service status check, received ${status}`);
-}
-
-function ServiceCheckValidResponseView(data: SuggestionServiceStatusCheckResponse | undefined) {
- switch (data?.status) {
- case 'SUCCESS':
- return (
- <SuccessMessage text={translate('property.codefix.admin.serviceCheck.result.success')} />
- );
- case 'TIMEOUT':
- case 'CONNECTION_ERROR':
- return (
- <div className="sw-flex">
- <IconError className="sw-mr-1" color="echoes-color-icon-danger" />
- <div className="sw-flex-col">
- <ErrorLabel
- text={translate('property.codefix.admin.serviceCheck.result.unresponsive.message')}
- />
- <p className="sw-mt-4">
- <ErrorLabel
- text={translate(
- 'property.codefix.admin.serviceCheck.result.unresponsive.causes.title',
- )}
- />
- </p>
- <UnorderedList className="sw-ml-8" ticks>
- <ErrorListItem className="sw-mb-2">
- <ErrorLabel
- text={translate(
- 'property.codefix.admin.serviceCheck.result.unresponsive.causes.1',
- )}
- />
- </ErrorListItem>
- <ErrorListItem>
- <ErrorLabel
- text={translate(
- 'property.codefix.admin.serviceCheck.result.unresponsive.causes.2',
- )}
- />
- </ErrorListItem>
- </UnorderedList>
- </div>
- </div>
- );
- case 'UNAUTHORIZED':
- return (
- <ErrorMessage text={translate('property.codefix.admin.serviceCheck.result.unauthorized')} />
- );
- case 'SERVICE_ERROR':
- return (
- <ErrorMessage text={translate('property.codefix.admin.serviceCheck.result.serviceError')} />
- );
- default:
- return (
- <ErrorMessage
- text={`${translate('property.codefix.admin.serviceCheck.result.unknown')} ${data?.status ?? 'no status returned from the service'}`}
- />
- );
- }
-}
-
-function ErrorMessage({ text }: Readonly<TextProps>) {
- return (
- <div className="sw-flex">
- <IconError className="sw-mr-1" color="echoes-color-icon-danger" />
- <ErrorLabel text={text} />
- </div>
- );
-}
-
-function ErrorLabel({ text }: Readonly<TextProps>) {
- return <Text colorOverride="echoes-color-text-danger">{text}</Text>;
-}
-
-function SuccessMessage({ text }: Readonly<TextProps>) {
- return (
- <div className="sw-flex">
- <IconCheckCircle className="sw-mr-1" color="echoes-color-icon-success" />
- <Text colorOverride="echoes-color-text-success">{text}</Text>
- </div>
- );
-}
-
-const ErrorListItem = styled.li`
- ::marker {
- color: ${themeColor('errorText')};
- }
-`;
-
-interface TextProps {
- /** The text to display inside the component */
- text: string;
-}
-
-export default withAvailableFeatures(CodeFixAdmin);
diff --git a/server/sonar-web/src/main/js/apps/settings/components/__tests__/AiCodeFixAdmin-it.tsx b/server/sonar-web/src/main/js/apps/settings/components/__tests__/AiCodeFixAdmin-it.tsx
new file mode 100644
index 00000000000..e5c576631da
--- /dev/null
+++ b/server/sonar-web/src/main/js/apps/settings/components/__tests__/AiCodeFixAdmin-it.tsx
@@ -0,0 +1,291 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2024 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+
+import { screen, waitFor } from '@testing-library/react';
+import userEvent from '@testing-library/user-event';
+import { uniq } from 'lodash';
+import { byRole, byText } from '~sonar-aligned/helpers/testSelector';
+import ComponentsServiceMock from '../../../../api/mocks/ComponentsServiceMock';
+import FixSuggestionsServiceMock from '../../../../api/mocks/FixSuggestionsServiceMock';
+import SettingsServiceMock, {
+ DEFAULT_DEFINITIONS_MOCK,
+} from '../../../../api/mocks/SettingsServiceMock';
+import { AvailableFeaturesContext } from '../../../../app/components/available-features/AvailableFeaturesContext';
+import { mockComponent, mockComponentRaw } from '../../../../helpers/mocks/component';
+import { definitions } from '../../../../helpers/mocks/definitions-list';
+import { renderComponent } from '../../../../helpers/testReactTestingUtils';
+import { Feature } from '../../../../types/features';
+import { AdditionalCategoryComponentProps } from '../AdditionalCategories';
+import AiCodeFixAdmin from '../AiCodeFixAdmin';
+
+let settingServiceMock: SettingsServiceMock;
+let componentsServiceMock: ComponentsServiceMock;
+let fixIssueServiceMock: FixSuggestionsServiceMock;
+
+beforeAll(() => {
+ settingServiceMock = new SettingsServiceMock();
+ settingServiceMock.setDefinitions(definitions);
+ componentsServiceMock = new ComponentsServiceMock();
+ fixIssueServiceMock = new FixSuggestionsServiceMock();
+});
+
+afterEach(() => {
+ settingServiceMock.reset();
+});
+
+const ui = {
+ codeFixTitle: byRole('heading', { name: 'property.aicodefix.admin.title' }),
+ enableAiCodeFixCheckbox: byRole('checkbox', {
+ name: 'property.aicodefix.admin.checkbox.label property.aicodefix.admin.terms property.aicodefix.admin.acceptTerm.terms open_in_new_tab',
+ }),
+ saveButton: byRole('button', { name: 'save' }),
+ cancelButton: byRole('button', { name: 'cancel' }),
+ checkServiceStatusButton: byRole('button', {
+ name: 'property.aicodefix.admin.serviceCheck.action',
+ }),
+ allProjectsEnabledRadio: byRole('radio', {
+ name: 'property.aicodefix.admin.enable.all.projects.label',
+ }),
+ someProjectsEnabledRadio: byRole('radio', {
+ name: 'property.aicodefix.admin.enable.some.projects.label',
+ }),
+ selectedTab: byRole('radio', { name: 'selected' }),
+ unselectedTab: byRole('radio', { name: 'unselected' }),
+ allTab: byRole('radio', { name: 'all' }),
+};
+
+it('should by default propose enabling for all projects when enabling the feature', async () => {
+ settingServiceMock.set('sonar.ai.suggestions.enabled', 'DISABLED');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ expect(await ui.codeFixTitle.find()).toBeInTheDocument();
+ expect(ui.enableAiCodeFixCheckbox.get()).not.toBeChecked();
+
+ await user.click(ui.enableAiCodeFixCheckbox.get());
+ expect(ui.allProjectsEnabledRadio.get()).toBeEnabled();
+});
+
+it('should be able to enable the code fix feature for all projects', async () => {
+ settingServiceMock.set('sonar.ai.suggestions.enabled', 'DISABLED');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ expect(await ui.codeFixTitle.find()).toBeInTheDocument();
+ expect(ui.enableAiCodeFixCheckbox.get()).not.toBeChecked();
+
+ await user.click(ui.enableAiCodeFixCheckbox.get());
+ expect(ui.allProjectsEnabledRadio.get()).toBeEnabled();
+ expect(ui.saveButton.get()).toBeEnabled();
+
+ await user.click(ui.saveButton.get());
+ expect(ui.enableAiCodeFixCheckbox.get()).toBeChecked();
+ await waitFor(() => {
+ expect(ui.saveButton.get()).toBeDisabled();
+ });
+});
+
+it('should be able to enable the code fix feature for some projects', async () => {
+ settingServiceMock.set('sonar.ai.suggestions.enabled', 'DISABLED');
+ const project = mockComponentRaw({ isAiCodeFixEnabled: false });
+ componentsServiceMock.registerProject(project);
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ expect(ui.enableAiCodeFixCheckbox.get()).not.toBeChecked();
+
+ await user.click(ui.enableAiCodeFixCheckbox.get());
+ expect(ui.someProjectsEnabledRadio.get()).toBeEnabled();
+ await user.click(ui.someProjectsEnabledRadio.get());
+
+ expect(ui.selectedTab.get()).toBeVisible();
+ expect(await ui.unselectedTab.find()).toBeVisible();
+ expect(await ui.allTab.find()).toBeVisible();
+ await user.click(ui.unselectedTab.get());
+ const projectCheckBox = byText(project.name);
+ await waitFor(() => {
+ expect(projectCheckBox.get()).toBeVisible();
+ });
+ await user.click(projectCheckBox.get());
+
+ await user.click(ui.saveButton.get());
+ expect(ui.enableAiCodeFixCheckbox.get()).toBeChecked();
+ await waitFor(() => {
+ expect(ui.saveButton.get()).toBeDisabled();
+ });
+});
+
+it('should be able to disable the feature for a single project', async () => {
+ settingServiceMock.set('sonar.ai.suggestions.enabled', 'ENABLED_FOR_SOME_PROJECTS');
+ const project = mockComponentRaw({ isAiCodeFixEnabled: true });
+ componentsServiceMock.registerProject(project);
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await waitFor(() => {
+ expect(ui.enableAiCodeFixCheckbox.get()).toBeChecked();
+ });
+ expect(ui.someProjectsEnabledRadio.get()).toBeEnabled();
+
+ // this project is by default registered by the mock
+ const projectName = 'sonar-plugin-api';
+ const projectCheckBox = byText(projectName);
+ expect(await projectCheckBox.find()).toBeInTheDocument();
+ await user.click(projectCheckBox.get());
+
+ await user.click(ui.saveButton.get());
+ expect(ui.enableAiCodeFixCheckbox.get()).toBeChecked();
+ await waitFor(() => {
+ expect(ui.saveButton.get()).toBeDisabled();
+ });
+});
+
+it('should be able to disable the code fix feature', async () => {
+ settingServiceMock.set('sonar.ai.suggestions.enabled', 'ENABLED_FOR_ALL_PROJECTS');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await waitFor(() => {
+ expect(ui.enableAiCodeFixCheckbox.get()).toBeChecked();
+ });
+
+ await user.click(ui.enableAiCodeFixCheckbox.get());
+ expect(await ui.saveButton.find()).toBeInTheDocument();
+ await user.click(await ui.saveButton.find());
+ expect(ui.enableAiCodeFixCheckbox.get()).not.toBeChecked();
+});
+
+it('should be able to reset the form when canceling', async () => {
+ settingServiceMock.set('sonar.ai.suggestions.enabled', 'ENABLED_FOR_ALL_PROJECTS');
+ componentsServiceMock.registerComponent(mockComponent());
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await waitFor(() => {
+ expect(ui.enableAiCodeFixCheckbox.get()).toBeChecked();
+ });
+
+ await user.click(ui.enableAiCodeFixCheckbox.get());
+ expect(ui.enableAiCodeFixCheckbox.get()).not.toBeChecked();
+ expect(await ui.cancelButton.find()).toBeInTheDocument();
+ await user.click(await ui.cancelButton.find());
+ expect(ui.enableAiCodeFixCheckbox.get()).toBeChecked();
+});
+
+it('should display a success message when the service status can be successfully checked', async () => {
+ fixIssueServiceMock.setServiceStatus('SUCCESS');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await user.click(ui.checkServiceStatusButton.get());
+
+ expect(
+ await screen.findByText('property.aicodefix.admin.serviceCheck.result.success'),
+ ).toBeInTheDocument();
+});
+
+it('should display an error message when the service is not responsive', async () => {
+ fixIssueServiceMock.setServiceStatus('TIMEOUT');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await user.click(ui.checkServiceStatusButton.get());
+
+ expect(
+ await screen.findByText('property.aicodefix.admin.serviceCheck.result.unresponsive.message'),
+ ).toBeInTheDocument();
+});
+
+it('should display an error message when there is a connection error with the service', async () => {
+ fixIssueServiceMock.setServiceStatus('CONNECTION_ERROR');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await user.click(ui.checkServiceStatusButton.get());
+
+ expect(
+ await screen.findByText('property.aicodefix.admin.serviceCheck.result.unresponsive.message'),
+ ).toBeInTheDocument();
+});
+
+it('should display an error message when the current instance is unauthorized', async () => {
+ fixIssueServiceMock.setServiceStatus('UNAUTHORIZED');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await user.click(ui.checkServiceStatusButton.get());
+
+ expect(
+ await screen.findByText('property.aicodefix.admin.serviceCheck.result.unauthorized'),
+ ).toBeInTheDocument();
+});
+
+it('should display an error message when an error happens at service level', async () => {
+ fixIssueServiceMock.setServiceStatus('SERVICE_ERROR');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await user.click(ui.checkServiceStatusButton.get());
+
+ expect(
+ await screen.findByText('property.aicodefix.admin.serviceCheck.result.serviceError'),
+ ).toBeInTheDocument();
+});
+
+it('should display an error message when the service answers with an unknown status', async () => {
+ fixIssueServiceMock.setServiceStatus('WTF');
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await user.click(ui.checkServiceStatusButton.get());
+
+ expect(
+ await screen.findByText('property.aicodefix.admin.serviceCheck.result.unknown WTF'),
+ ).toBeInTheDocument();
+});
+
+it('should display an error message when the backend answers with an error', async () => {
+ fixIssueServiceMock.setServiceStatus(undefined);
+ const user = userEvent.setup();
+ renderCodeFixAdmin();
+
+ await user.click(ui.checkServiceStatusButton.get());
+
+ expect(
+ await screen.findByText('property.aicodefix.admin.serviceCheck.result.requestError No status'),
+ ).toBeInTheDocument();
+});
+
+function renderCodeFixAdmin(
+ overrides: Partial<AdditionalCategoryComponentProps> = {},
+ features?: Feature[],
+) {
+ const props = {
+ definitions: DEFAULT_DEFINITIONS_MOCK,
+ categories: uniq(DEFAULT_DEFINITIONS_MOCK.map((d) => d.category)),
+ selectedCategory: 'general',
+ ...overrides,
+ };
+ return renderComponent(
+ <AvailableFeaturesContext.Provider value={features ?? [Feature.FixSuggestions]}>
+ <AiCodeFixAdmin {...props} />
+ </AvailableFeaturesContext.Provider>,
+ );
+}
diff --git a/server/sonar-web/src/main/js/apps/settings/components/__tests__/CodeFixAdmin-it.tsx b/server/sonar-web/src/main/js/apps/settings/components/__tests__/CodeFixAdmin-it.tsx
deleted file mode 100644
index a1f1ca87a23..00000000000
--- a/server/sonar-web/src/main/js/apps/settings/components/__tests__/CodeFixAdmin-it.tsx
+++ /dev/null
@@ -1,195 +0,0 @@
-/*
- * SonarQube
- * Copyright (C) 2009-2024 SonarSource SA
- * mailto:info AT sonarsource DOT com
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 3 of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public License
- * along with this program; if not, write to the Free Software Foundation,
- * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
-
-import { screen, waitFor } from '@testing-library/react';
-import userEvent from '@testing-library/user-event';
-import { uniq } from 'lodash';
-import { byRole } from '~sonar-aligned/helpers/testSelector';
-import FixIssueServiceMock from '../../../../api/mocks/FixIssueServiceMock';
-import SettingsServiceMock, {
- DEFAULT_DEFINITIONS_MOCK,
-} from '../../../../api/mocks/SettingsServiceMock';
-import { AvailableFeaturesContext } from '../../../../app/components/available-features/AvailableFeaturesContext';
-import { mockComponent } from '../../../../helpers/mocks/component';
-import { definitions } from '../../../../helpers/mocks/definitions-list';
-import { renderComponent } from '../../../../helpers/testReactTestingUtils';
-import { Feature } from '../../../../types/features';
-import { AdditionalCategoryComponentProps } from '../AdditionalCategories';
-import CodeFixAdmin from '../CodeFixAdmin';
-
-let settingServiceMock: SettingsServiceMock;
-let fixIssueServiceMock: FixIssueServiceMock;
-
-beforeAll(() => {
- settingServiceMock = new SettingsServiceMock();
- settingServiceMock.setDefinitions(definitions);
- fixIssueServiceMock = new FixIssueServiceMock();
-});
-
-afterEach(() => {
- settingServiceMock.reset();
-});
-
-const ui = {
- codeFixTitle: byRole('heading', { name: 'property.codefix.admin.title' }),
- changeCodeFixCheckbox: byRole('checkbox', { name: 'property.codefix.admin.checkbox.label' }),
- acceptTermCheckbox: byRole('checkbox', {
- name: 'property.codefix.admin.terms property.codefix.admin.acceptTerm.terms',
- }),
- saveButton: byRole('button', { name: 'save' }),
- checkServiceStatusButton: byRole('button', {
- name: 'property.codefix.admin.serviceCheck.action',
- }),
-};
-
-it('should be able to enable the code fix feature', async () => {
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- expect(await ui.codeFixTitle.find()).toBeInTheDocument();
- expect(ui.changeCodeFixCheckbox.get()).not.toBeChecked();
-
- await user.click(ui.changeCodeFixCheckbox.get());
- expect(ui.acceptTermCheckbox.get()).toBeInTheDocument();
- expect(ui.saveButton.get()).toBeDisabled();
-
- await user.click(ui.acceptTermCheckbox.get());
- expect(ui.saveButton.get()).toBeEnabled();
-
- await user.click(ui.saveButton.get());
- expect(ui.changeCodeFixCheckbox.get()).toBeChecked();
-});
-
-it('should be able to disable the code fix feature', async () => {
- settingServiceMock.set('sonar.ai.suggestions.enabled', 'true');
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- await waitFor(() => {
- expect(ui.changeCodeFixCheckbox.get()).toBeChecked();
- });
-
- await user.click(ui.changeCodeFixCheckbox.get());
- expect(await ui.saveButton.find()).toBeInTheDocument();
- await user.click(await ui.saveButton.find());
- expect(ui.changeCodeFixCheckbox.get()).not.toBeChecked();
-});
-
-it('should display a success message when the service status can be successfully checked', async () => {
- fixIssueServiceMock.setServiceStatus('SUCCESS');
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- await user.click(ui.checkServiceStatusButton.get());
-
- expect(
- await screen.findByText('property.codefix.admin.serviceCheck.result.success'),
- ).toBeInTheDocument();
-});
-
-it('should display an error message when the service is not responsive', async () => {
- fixIssueServiceMock.setServiceStatus('TIMEOUT');
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- await user.click(ui.checkServiceStatusButton.get());
-
- expect(
- await screen.findByText('property.codefix.admin.serviceCheck.result.unresponsive.message'),
- ).toBeInTheDocument();
-});
-
-it('should display an error message when there is a connection error with the service', async () => {
- fixIssueServiceMock.setServiceStatus('CONNECTION_ERROR');
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- await user.click(ui.checkServiceStatusButton.get());
-
- expect(
- await screen.findByText('property.codefix.admin.serviceCheck.result.unresponsive.message'),
- ).toBeInTheDocument();
-});
-
-it('should display an error message when the current instance is unauthorized', async () => {
- fixIssueServiceMock.setServiceStatus('UNAUTHORIZED');
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- await user.click(ui.checkServiceStatusButton.get());
-
- expect(
- await screen.findByText('property.codefix.admin.serviceCheck.result.unauthorized'),
- ).toBeInTheDocument();
-});
-
-it('should display an error message when an error happens at service level', async () => {
- fixIssueServiceMock.setServiceStatus('SERVICE_ERROR');
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- await user.click(ui.checkServiceStatusButton.get());
-
- expect(
- await screen.findByText('property.codefix.admin.serviceCheck.result.serviceError'),
- ).toBeInTheDocument();
-});
-
-it('should display an error message when the service answers with an unknown status', async () => {
- fixIssueServiceMock.setServiceStatus('WTF');
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- await user.click(ui.checkServiceStatusButton.get());
-
- expect(
- await screen.findByText('property.codefix.admin.serviceCheck.result.unknown WTF'),
- ).toBeInTheDocument();
-});
-
-it('should display an error message when the backend answers with an error', async () => {
- fixIssueServiceMock.setServiceStatus(undefined);
- const user = userEvent.setup();
- renderCodeFixAdmin();
-
- await user.click(ui.checkServiceStatusButton.get());
-
- expect(
- await screen.findByText('property.codefix.admin.serviceCheck.result.requestError No status'),
- ).toBeInTheDocument();
-});
-
-function renderCodeFixAdmin(
- overrides: Partial<AdditionalCategoryComponentProps> = {},
- features?: Feature[],
-) {
- const props = {
- definitions: DEFAULT_DEFINITIONS_MOCK,
- categories: uniq(DEFAULT_DEFINITIONS_MOCK.map((d) => d.category)),
- selectedCategory: 'general',
- component: mockComponent(),
- ...overrides,
- };
- return renderComponent(
- <AvailableFeaturesContext.Provider value={features ?? [Feature.FixSuggestions]}>
- <CodeFixAdmin {...props} />
- </AvailableFeaturesContext.Provider>,
- );
-}
diff --git a/server/sonar-web/src/main/js/apps/settings/constants.ts b/server/sonar-web/src/main/js/apps/settings/constants.ts
index 12c0e96a1cf..d21ab6b25f2 100644
--- a/server/sonar-web/src/main/js/apps/settings/constants.ts
+++ b/server/sonar-web/src/main/js/apps/settings/constants.ts
@@ -23,7 +23,7 @@ import { ExtendedSettingDefinition } from '../../types/settings';
import { Dict } from '../../types/types';
export const ALM_INTEGRATION_CATEGORY = 'almintegration';
-export const CODE_FIX_CATEGORY = 'codefix';
+export const AI_CODE_FIX_CATEGORY = 'ai_codefix';
export const AUTHENTICATION_CATEGORY = 'authentication';
export const ANALYSIS_SCOPE_CATEGORY = 'exclusions';
export const LANGUAGES_CATEGORY = 'languages';
diff --git a/server/sonar-web/src/main/js/components/rules/IssueSuggestionFileSnippet.tsx b/server/sonar-web/src/main/js/components/rules/IssueSuggestionFileSnippet.tsx
index 397fa877005..082999567ec 100644
--- a/server/sonar-web/src/main/js/components/rules/IssueSuggestionFileSnippet.tsx
+++ b/server/sonar-web/src/main/js/components/rules/IssueSuggestionFileSnippet.tsx
@@ -48,6 +48,7 @@ interface Props {
issue: Issue;
language?: string;
}
+
const EXPAND_SIZE = 10;
const BUFFER_CODE = 3;
diff --git a/server/sonar-web/src/main/js/helpers/mocks/component.ts b/server/sonar-web/src/main/js/helpers/mocks/component.ts
index dcef4629d10..fe93701c862 100644
--- a/server/sonar-web/src/main/js/helpers/mocks/component.ts
+++ b/server/sonar-web/src/main/js/helpers/mocks/component.ts
@@ -20,6 +20,7 @@
import { ComponentQualifier, Visibility } from '~sonar-aligned/types/component';
import { MetricKey } from '~sonar-aligned/types/metrics';
+import { ComponentRaw } from '../../api/components';
import { TreeComponent } from '../../types/component';
import { Component, ComponentMeasure, ComponentMeasureEnhanced } from '../../types/types';
import { mockMeasureEnhanced } from '../testMocks';
@@ -44,6 +45,17 @@ export function mockComponent(overrides: Partial<Component> = {}): Component {
};
}
+export function mockComponentRaw(overrides: Partial<ComponentRaw> = {}): ComponentRaw {
+ return {
+ key: 'my-project',
+ name: 'MyProject',
+ qualifier: ComponentQualifier.Project,
+ tags: [],
+ ...overrides,
+ visibility: Visibility.Public,
+ };
+}
+
export function mockTreeComponent(overrides: Partial<TreeComponent>): TreeComponent {
return {
key: 'my-key',
diff --git a/server/sonar-web/src/main/js/queries/fix-suggestions.tsx b/server/sonar-web/src/main/js/queries/fix-suggestions.tsx
index fc70ff12e2c..8edcb03f044 100644
--- a/server/sonar-web/src/main/js/queries/fix-suggestions.tsx
+++ b/server/sonar-web/src/main/js/queries/fix-suggestions.tsx
@@ -27,6 +27,7 @@ import {
getFixSuggestionsIssues,
getSuggestions,
SuggestionServiceStatusCheckResponse,
+ updateFeatureEnablement,
} from '../api/fix-suggestions';
import { useAvailableFeatures } from '../app/components/available-features/withAvailableFeatures';
import { CurrentUserContext } from '../app/components/current-user/CurrentUserContext';
@@ -183,6 +184,12 @@ export function withUseGetFixSuggestionsIssues<P extends { issue: Issue }>(
};
}
+export function useUpdateFeatureEnablementMutation() {
+ return useMutation({
+ mutationFn: updateFeatureEnablement,
+ });
+}
+
export function useCheckServiceMutation() {
return useMutation<SuggestionServiceStatusCheckResponse, AxiosError>({
mutationFn: checkSuggestionServiceStatus,
diff --git a/server/sonar-web/src/main/js/types/fix-suggestions.ts b/server/sonar-web/src/main/js/types/fix-suggestions.ts
index f88606353a1..35c52db4448 100644
--- a/server/sonar-web/src/main/js/types/fix-suggestions.ts
+++ b/server/sonar-web/src/main/js/types/fix-suggestions.ts
@@ -18,6 +18,12 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
+export enum AiCodeFixFeatureEnablement {
+ disabled = 'DISABLED',
+ allProjects = 'ENABLED_FOR_ALL_PROJECTS',
+ someProjects = 'ENABLED_FOR_SOME_PROJECTS',
+}
+
interface SuggestedChange {
endLine: number;
newCode: string;