aboutsummaryrefslogtreecommitdiffstats
path: root/server/sonar-web/src/main
diff options
context:
space:
mode:
Diffstat (limited to 'server/sonar-web/src/main')
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/SettingsApp.tsx9
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/authentication/Authentication.tsx17
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/authentication/GithubAuthenticationTab.tsx10
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/authentication/SamlAuthenticationTab.tsx25
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useGithubConfiguration.ts47
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useSamlConfiguration.ts20
-rw-r--r--server/sonar-web/src/main/js/apps/settings/components/authentication/queries/IdentityProvider.ts86
-rw-r--r--server/sonar-web/src/main/js/helpers/testReactTestingUtils.tsx22
8 files changed, 140 insertions, 96 deletions
diff --git a/server/sonar-web/src/main/js/apps/settings/components/SettingsApp.tsx b/server/sonar-web/src/main/js/apps/settings/components/SettingsApp.tsx
index e05ce398856..f10b4a70348 100644
--- a/server/sonar-web/src/main/js/apps/settings/components/SettingsApp.tsx
+++ b/server/sonar-web/src/main/js/apps/settings/components/SettingsApp.tsx
@@ -17,6 +17,7 @@
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
+import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
import * as React from 'react';
import { getDefinitions } from '../../../api/settings';
import withComponentContext from '../../../app/components/componentContext/withComponentContext';
@@ -31,6 +32,8 @@ import { Component } from '../../../types/types';
import '../styles.css';
import SettingsAppRenderer from './SettingsAppRenderer';
+const queryClient = new QueryClient();
+
interface Props {
component?: Component;
}
@@ -77,7 +80,11 @@ class SettingsApp extends React.PureComponent<Props, State> {
render() {
const { component } = this.props;
- return <SettingsAppRenderer component={component} {...this.state} />;
+ return (
+ <QueryClientProvider client={queryClient}>
+ <SettingsAppRenderer component={component} {...this.state} />
+ </QueryClientProvider>
+ );
}
}
diff --git a/server/sonar-web/src/main/js/apps/settings/components/authentication/Authentication.tsx b/server/sonar-web/src/main/js/apps/settings/components/authentication/Authentication.tsx
index 96d3160caef..7d9b411e7ae 100644
--- a/server/sonar-web/src/main/js/apps/settings/components/authentication/Authentication.tsx
+++ b/server/sonar-web/src/main/js/apps/settings/components/authentication/Authentication.tsx
@@ -19,10 +19,8 @@
*/
import classNames from 'classnames';
import * as React from 'react';
-import { useCallback, useEffect, useState } from 'react';
import { FormattedMessage } from 'react-intl';
import { useSearchParams } from 'react-router-dom';
-import { getSystemInfo } from '../../../../api/system';
import withAvailableFeatures, {
WithAvailableFeaturesProps,
} from '../../../../app/components/available-features/withAvailableFeatures';
@@ -37,7 +35,6 @@ import { searchParamsToQuery } from '../../../../helpers/urls';
import { AlmKeys } from '../../../../types/alm-settings';
import { Feature } from '../../../../types/features';
import { ExtendedSettingDefinition } from '../../../../types/settings';
-import { SysInfoCluster } from '../../../../types/types';
import { AUTHENTICATION_CATEGORY } from '../../constants';
import CategoryDefinitionsList from '../CategoryDefinitionsList';
import GithubAuthenticationTab from './GithubAuthenticationTab';
@@ -78,16 +75,6 @@ export function Authentication(props: Props & WithAvailableFeaturesProps) {
const { definitions } = props;
const [query, setSearchParams] = useSearchParams();
- const [provider, setProvider] = useState<string>();
-
- const loadProvider = useCallback(async () => {
- const info = (await getSystemInfo()) as SysInfoCluster;
- setProvider(info.System['External Users and Groups Provisioning']);
- }, []);
-
- useEffect(() => {
- loadProvider();
- }, []);
const currentTab = (query.get('tab') || SAML) as AuthenticationTabs;
@@ -182,16 +169,12 @@ export function Authentication(props: Props & WithAvailableFeaturesProps) {
{tab.key === SAML && (
<SamlAuthenticationTab
definitions={definitions.filter((def) => def.subCategory === SAML)}
- provider={provider}
- onReload={() => loadProvider()}
/>
)}
{tab.key === AlmKeys.GitHub && (
<GithubAuthenticationTab
definitions={definitions.filter((def) => def.subCategory === AlmKeys.GitHub)}
- provider={provider}
- onReload={() => loadProvider()}
/>
)}
diff --git a/server/sonar-web/src/main/js/apps/settings/components/authentication/GithubAuthenticationTab.tsx b/server/sonar-web/src/main/js/apps/settings/components/authentication/GithubAuthenticationTab.tsx
index ca32c00991b..44f704a755b 100644
--- a/server/sonar-web/src/main/js/apps/settings/components/authentication/GithubAuthenticationTab.tsx
+++ b/server/sonar-web/src/main/js/apps/settings/components/authentication/GithubAuthenticationTab.tsx
@@ -36,11 +36,10 @@ import { DOCUMENTATION_LINK_SUFFIXES } from './Authentication';
import AuthenticationFormField from './AuthenticationFormField';
import ConfigurationForm from './ConfigurationForm';
import useGithubConfiguration, { GITHUB_JIT_FIELDS } from './hook/useGithubConfiguration';
+import { useIdentityProvierQuery } from './queries/IdentityProvider';
interface GithubAuthenticationProps {
definitions: ExtendedSettingDefinition[];
- provider: string | undefined;
- onReload: () => void;
}
const GITHUB_EXCLUDED_FIELD = [
@@ -51,7 +50,8 @@ const GITHUB_EXCLUDED_FIELD = [
];
export default function GithubAuthenticationTab(props: GithubAuthenticationProps) {
- const { definitions, provider } = props;
+ const { definitions } = props;
+ const { data } = useIdentityProvierQuery();
const [showEditModal, setShowEditModal] = useState(false);
const [showConfirmProvisioningModal, setShowConfirmProvisioningModal] = useState(false);
@@ -76,9 +76,9 @@ export default function GithubAuthenticationTab(props: GithubAuthenticationProps
changeProvisioning,
toggleEnable,
hasLegacyConfiguration,
- } = useGithubConfiguration(definitions, props.onReload);
+ } = useGithubConfiguration(definitions);
- const hasDifferentProvider = provider !== undefined && provider !== Provider.Github;
+ const hasDifferentProvider = data?.provider !== undefined && data.provider !== Provider.Github;
const handleCreateConfiguration = () => {
setShowEditModal(true);
diff --git a/server/sonar-web/src/main/js/apps/settings/components/authentication/SamlAuthenticationTab.tsx b/server/sonar-web/src/main/js/apps/settings/components/authentication/SamlAuthenticationTab.tsx
index 44722f68f37..092ae89ede2 100644
--- a/server/sonar-web/src/main/js/apps/settings/components/authentication/SamlAuthenticationTab.tsx
+++ b/server/sonar-web/src/main/js/apps/settings/components/authentication/SamlAuthenticationTab.tsx
@@ -20,12 +20,7 @@
import { isEmpty } from 'lodash';
import React from 'react';
import { FormattedMessage } from 'react-intl';
-import {
- activateScim,
- deactivateScim,
- resetSettingValue,
- setSettingValue,
-} from '../../../../api/settings';
+import { resetSettingValue, setSettingValue } from '../../../../api/settings';
import DocLink from '../../../../components/common/DocLink';
import Link from '../../../../components/common/Link';
import ConfirmModal from '../../../../components/controls/ConfirmModal';
@@ -47,11 +42,10 @@ import useSamlConfiguration, {
SAML_GROUP_NAME,
SAML_SCIM_DEPRECATED,
} from './hook/useSamlConfiguration';
+import { useIdentityProvierQuery, useToggleScimMutation } from './queries/IdentityProvider';
interface SamlAuthenticationProps {
definitions: ExtendedSettingDefinition[];
- provider: string | undefined;
- onReload: () => void;
}
export const SAML = 'saml';
@@ -60,7 +54,7 @@ const CONFIG_TEST_PATH = '/saml/validation_init';
const SAML_EXCLUDED_FIELD = [SAML_ENABLED_FIELD, SAML_GROUP_NAME, SAML_SCIM_DEPRECATED];
export default function SamlAuthenticationTab(props: SamlAuthenticationProps) {
- const { definitions, provider, onReload } = props;
+ const { definitions } = props;
const [showEditModal, setShowEditModal] = React.useState(false);
const [showConfirmProvisioningModal, setShowConfirmProvisioningModal] = React.useState(false);
const {
@@ -81,9 +75,12 @@ export default function SamlAuthenticationTab(props: SamlAuthenticationProps) {
setNewGroupSetting,
reload,
deleteConfiguration,
- } = useSamlConfiguration(definitions, onReload);
+ } = useSamlConfiguration(definitions);
+ const toggleScim = useToggleScimMutation();
- const hasDifferentProvider = provider !== undefined && provider !== Provider.Scim;
+ const { data } = useIdentityProvierQuery();
+
+ const hasDifferentProvider = data?.provider !== undefined && data.provider !== Provider.Scim;
const handleCreateConfiguration = () => {
setShowEditModal(true);
@@ -111,10 +108,8 @@ export default function SamlAuthenticationTab(props: SamlAuthenticationProps) {
};
const handleConfirmChangeProvisioning = async () => {
- if (newScimStatus) {
- await activateScim();
- } else {
- await deactivateScim();
+ await toggleScim.mutateAsync(!!newScimStatus);
+ if (!newScimStatus) {
await handleSaveGroup();
}
await reload();
diff --git a/server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useGithubConfiguration.ts b/server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useGithubConfiguration.ts
index dea619fd54a..8c9c255467b 100644
--- a/server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useGithubConfiguration.ts
+++ b/server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useGithubConfiguration.ts
@@ -18,17 +18,15 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
import { isEmpty, some } from 'lodash';
-import { useCallback, useContext, useEffect, useState } from 'react';
-import {
- activateGithubProvisioning,
- deactivateGithubProvisioning,
- fetchIsGithubProvisioningEnabled,
- resetSettingValue,
- setSettingValue,
-} from '../../../../../api/settings';
+import { useCallback, useContext, useState } from 'react';
+import { resetSettingValue, setSettingValue } from '../../../../../api/settings';
import { AvailableFeaturesContext } from '../../../../../app/components/available-features/AvailableFeaturesContext';
import { Feature } from '../../../../../types/features';
import { ExtendedSettingDefinition } from '../../../../../types/settings';
+import {
+ useGithubStatusQuery,
+ useToggleGithubProvisioningMutation,
+} from '../queries/IdentityProvider';
import useConfiguration from './useConfiguration';
export const GITHUB_ENABLED_FIELD = 'sonar.auth.github.enabled';
@@ -52,16 +50,15 @@ export interface SamlSettingValue {
definition: ExtendedSettingDefinition;
}
-export default function useGithubConfiguration(
- definitions: ExtendedSettingDefinition[],
- onReload: () => void
-) {
+export default function useGithubConfiguration(definitions: ExtendedSettingDefinition[]) {
const config = useConfiguration(definitions, OPTIONAL_FIELDS);
const { values, isValueChange, setNewValue, reload: reloadConfig } = config;
+
const hasGithubProvisioning = useContext(AvailableFeaturesContext).includes(
Feature.GithubProvisioning
);
- const [githubProvisioningStatus, setGithubProvisioningStatus] = useState(false);
+ const { data: githubProvisioningStatus } = useGithubStatusQuery();
+ const toggleGithubProvisioning = useToggleGithubProvisioningMutation();
const [newGithubProvisioningStatus, setNewGithubProvisioningStatus] = useState<boolean>();
const hasGithubProvisioningConfigChange =
some(GITHUB_JIT_FIELDS, isValueChange) ||
@@ -72,14 +69,6 @@ export default function useGithubConfiguration(
GITHUB_JIT_FIELDS.forEach((s) => setNewValue(s));
};
- useEffect(() => {
- (async () => {
- if (hasGithubProvisioning) {
- setGithubProvisioningStatus(await fetchIsGithubProvisioningEnabled());
- }
- })();
- }, [hasGithubProvisioning]);
-
const enabled = values[GITHUB_ENABLED_FIELD]?.value === 'true';
const appId = values[GITHUB_APP_ID_FIELD]?.value as string;
const url = values[GITHUB_API_URL_FIELD]?.value;
@@ -87,20 +76,13 @@ export default function useGithubConfiguration(
const reload = useCallback(async () => {
await reloadConfig();
- setGithubProvisioningStatus(await fetchIsGithubProvisioningEnabled());
- onReload();
- }, [reloadConfig, onReload]);
+ }, [reloadConfig]);
const changeProvisioning = async () => {
- if (newGithubProvisioningStatus && newGithubProvisioningStatus !== githubProvisioningStatus) {
- await activateGithubProvisioning();
- await reload();
- } else {
- if (newGithubProvisioningStatus !== githubProvisioningStatus) {
- await deactivateGithubProvisioning();
- }
- await saveGroup();
+ if (newGithubProvisioningStatus !== githubProvisioningStatus) {
+ await toggleGithubProvisioning.mutateAsync(!!newGithubProvisioningStatus);
}
+ await saveGroup();
};
const saveGroup = async () => {
@@ -134,7 +116,6 @@ export default function useGithubConfiguration(
enabled,
appId,
hasGithubProvisioning,
- setGithubProvisioningStatus,
githubProvisioningStatus,
newGithubProvisioningStatus,
setNewGithubProvisioningStatus,
diff --git a/server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useSamlConfiguration.ts b/server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useSamlConfiguration.ts
index 034c8ee5cfa..aed3fa84031 100644
--- a/server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useSamlConfiguration.ts
+++ b/server/sonar-web/src/main/js/apps/settings/components/authentication/hook/useSamlConfiguration.ts
@@ -18,10 +18,10 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
import React from 'react';
-import { fetchIsScimEnabled } from '../../../../../api/settings';
import { AvailableFeaturesContext } from '../../../../../app/components/available-features/AvailableFeaturesContext';
import { Feature } from '../../../../../types/features';
import { ExtendedSettingDefinition } from '../../../../../types/settings';
+import { useScimStatusQuery } from '../queries/IdentityProvider';
import useConfiguration from './useConfiguration';
export const SAML_ENABLED_FIELD = 'sonar.auth.saml.enabled';
@@ -39,23 +39,13 @@ const OPTIONAL_FIELDS = [
SAML_SCIM_DEPRECATED,
];
-export default function useSamlConfiguration(
- definitions: ExtendedSettingDefinition[],
- onReload: () => void
-) {
- const [scimStatus, setScimStatus] = React.useState<boolean>(false);
+export default function useSamlConfiguration(definitions: ExtendedSettingDefinition[]) {
const [newScimStatus, setNewScimStatus] = React.useState<boolean>();
const hasScim = React.useContext(AvailableFeaturesContext).includes(Feature.Scim);
const config = useConfiguration(definitions, OPTIONAL_FIELDS);
const { reload: reloadConfig, values, setNewValue, isValueChange } = config;
- React.useEffect(() => {
- (async () => {
- if (hasScim) {
- setScimStatus(await fetchIsScimEnabled());
- }
- })();
- }, [hasScim]);
+ const { data: scimStatus } = useScimStatusQuery();
const name = values[SAML_PROVIDER_NAME]?.value;
const url = values[SAML_LOGIN_URL]?.value;
@@ -71,9 +61,7 @@ export default function useSamlConfiguration(
const reload = React.useCallback(async () => {
await reloadConfig();
- setScimStatus(await fetchIsScimEnabled());
- onReload();
- }, [reloadConfig, onReload]);
+ }, [reloadConfig]);
return {
...config,
diff --git a/server/sonar-web/src/main/js/apps/settings/components/authentication/queries/IdentityProvider.ts b/server/sonar-web/src/main/js/apps/settings/components/authentication/queries/IdentityProvider.ts
new file mode 100644
index 00000000000..9a0687c630f
--- /dev/null
+++ b/server/sonar-web/src/main/js/apps/settings/components/authentication/queries/IdentityProvider.ts
@@ -0,0 +1,86 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2023 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 { useMutation, useQuery, useQueryClient } from '@tanstack/react-query';
+import { useContext } from 'react';
+import {
+ activateGithubProvisioning,
+ activateScim,
+ deactivateGithubProvisioning,
+ deactivateScim,
+ fetchIsGithubProvisioningEnabled,
+ fetchIsScimEnabled,
+} from '../../../../../api/settings';
+import { getSystemInfo } from '../../../../../api/system';
+import { AvailableFeaturesContext } from '../../../../../app/components/available-features/AvailableFeaturesContext';
+import { Feature } from '../../../../../types/features';
+import { SysInfoCluster } from '../../../../../types/types';
+
+export function useIdentityProvierQuery() {
+ return useQuery(['identity_provider'], async () => {
+ const info = (await getSystemInfo()) as SysInfoCluster;
+ return { provider: info.System['External Users and Groups Provisioning'] };
+ });
+}
+
+export function useScimStatusQuery() {
+ const hasScim = useContext(AvailableFeaturesContext).includes(Feature.Scim);
+
+ return useQuery(['identity_provider', 'scim_status'], () => {
+ if (!hasScim) {
+ return false;
+ }
+ return fetchIsScimEnabled();
+ });
+}
+
+export function useGithubStatusQuery() {
+ const hasGithubProvisioning = useContext(AvailableFeaturesContext).includes(
+ Feature.GithubProvisioning
+ );
+
+ return useQuery(['identity_provider', 'github_status'], () => {
+ if (!hasGithubProvisioning) {
+ return false;
+ }
+ return fetchIsGithubProvisioningEnabled();
+ });
+}
+
+export function useToggleScimMutation() {
+ const client = useQueryClient();
+ return useMutation({
+ mutationFn: (activate: boolean) => (activate ? activateScim() : deactivateScim()),
+ onSuccess: () => {
+ client.invalidateQueries({ queryKey: ['identity_provider'] });
+ },
+ });
+}
+
+export function useToggleGithubProvisioningMutation() {
+ const client = useQueryClient();
+ return useMutation({
+ mutationFn: (activate: boolean) =>
+ activate ? activateGithubProvisioning() : deactivateGithubProvisioning(),
+ onSuccess: () => {
+ client.invalidateQueries({ queryKey: ['identity_provider'] });
+ },
+ });
+}
diff --git a/server/sonar-web/src/main/js/helpers/testReactTestingUtils.tsx b/server/sonar-web/src/main/js/helpers/testReactTestingUtils.tsx
index 1d0ffcf5f5c..765b22ef21c 100644
--- a/server/sonar-web/src/main/js/helpers/testReactTestingUtils.tsx
+++ b/server/sonar-web/src/main/js/helpers/testReactTestingUtils.tsx
@@ -17,6 +17,7 @@
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
+import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
import { fireEvent, Matcher, render, RenderResult, screen, within } from '@testing-library/react';
import { UserEvent } from '@testing-library/user-event/dist/types/setup/setup';
import { omit } from 'lodash';
@@ -43,6 +44,7 @@ import { mockComponent } from './mocks/component';
import { DEFAULT_METRICS } from './mocks/metrics';
import { mockAppState, mockCurrentUser } from './testMocks';
+const queryClient = new QueryClient();
export interface RenderContext {
metrics?: Dict<Metric>;
appState?: AppState;
@@ -98,15 +100,17 @@ export function renderComponent(
function Wrapper({ children }: { children: React.ReactElement }) {
return (
<IntlProvider defaultLocale="en" locale="en">
- <HelmetProvider>
- <AppStateContextProvider appState={appState}>
- <MemoryRouter initialEntries={[pathname]}>
- <Routes>
- <Route path="*" element={children} />
- </Routes>
- </MemoryRouter>
- </AppStateContextProvider>
- </HelmetProvider>
+ <QueryClientProvider client={queryClient}>
+ <HelmetProvider>
+ <AppStateContextProvider appState={appState}>
+ <MemoryRouter initialEntries={[pathname]}>
+ <Routes>
+ <Route path="*" element={children} />
+ </Routes>
+ </MemoryRouter>
+ </AppStateContextProvider>
+ </HelmetProvider>
+ </QueryClientProvider>
</IntlProvider>
);
}