aboutsummaryrefslogtreecommitdiffstats
path: root/server/sonar-web/src/main
diff options
context:
space:
mode:
authorJeremy <47277647+jeremy-davis-sonarsource@users.noreply.github.com>2019-03-04 15:19:47 +0100
committersonartech <sonartech@sonarsource.com>2019-03-06 11:30:44 +0100
commitb41f2655daa7d58a272f0f98ac1ec16ff10f6e24 (patch)
tree6c4a2db000811c72ac74a6069a139e91de887313 /server/sonar-web/src/main
parent052e3b08498bb3c42aaabaaeea2325dfb03e2bac (diff)
downloadsonarqube-b41f2655daa7d58a272f0f98ac1ec16ff10f6e24.tar.gz
sonarqube-b41f2655daa7d58a272f0f98ac1ec16ff10f6e24.zip
SONARCLOUD-343 improve existing org alm binding
Diffstat (limited to 'server/sonar-web/src/main')
-rw-r--r--server/sonar-web/src/main/js/app/styles/components/boxed-group.css6
-rw-r--r--server/sonar-web/src/main/js/apps/create/organization/CreateOrganization.tsx76
-rw-r--r--server/sonar-web/src/main/js/apps/create/organization/__tests__/CreateOrganization-test.tsx53
-rw-r--r--server/sonar-web/src/main/js/apps/create/organization/utils.ts4
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/components/OrganizationBind.tsx126
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/components/OrganizationDelete.tsx95
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/components/OrganizationEdit.tsx21
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationBind-test.tsx72
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationEdit-test.tsx7
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationBind-test.tsx.snap23
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationDelete-test.tsx.snap222
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationEdit-test.tsx.snap27
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigation.tsx8
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationAdministration.tsx20
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeader.tsx63
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeaderContainer.tsx32
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/OrganizationNavigationHeader-test.tsx40
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigation-test.tsx.snap8
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationAdministration-test.tsx.snap20
-rw-r--r--server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationHeader-test.tsx.snap77
-rw-r--r--server/sonar-web/src/main/js/components/nav/ContextNavBar.css3
-rw-r--r--server/sonar-web/src/main/js/helpers/__tests__/almIntegrations-test.ts17
-rw-r--r--server/sonar-web/src/main/js/helpers/almIntegrations.ts6
23 files changed, 719 insertions, 307 deletions
diff --git a/server/sonar-web/src/main/js/app/styles/components/boxed-group.css b/server/sonar-web/src/main/js/app/styles/components/boxed-group.css
index 6912e432353..119c9a9cf40 100644
--- a/server/sonar-web/src/main/js/app/styles/components/boxed-group.css
+++ b/server/sonar-web/src/main/js/app/styles/components/boxed-group.css
@@ -35,6 +35,12 @@
padding: calc(2 * var(--gridSize)) 20px 0;
}
+.boxed-group > h2.boxed-title {
+ padding: 0 0 8px;
+ margin: 0;
+ font-weight: 600;
+}
+
.boxed-group hr {
height: 0;
border-top: 1px solid var(--gray94);
diff --git a/server/sonar-web/src/main/js/apps/create/organization/CreateOrganization.tsx b/server/sonar-web/src/main/js/apps/create/organization/CreateOrganization.tsx
index 0e441863c12..4d28a6aed92 100644
--- a/server/sonar-web/src/main/js/apps/create/organization/CreateOrganization.tsx
+++ b/server/sonar-web/src/main/js/apps/create/organization/CreateOrganization.tsx
@@ -31,7 +31,9 @@ import {
serializeQuery,
Query,
ORGANIZATION_IMPORT_BINDING_IN_PROGRESS_TIMESTAMP,
- Step
+ Step,
+ BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP,
+ BIND_ORGANIZATION_KEY
} from './utils';
import AlmApplicationInstalling from './AlmApplicationInstalling';
import AutoOrganizationCreate from './AutoOrganizationCreate';
@@ -47,7 +49,8 @@ import {
getAlmAppInfo,
getAlmOrganization,
GetAlmOrganizationResponse,
- listUnboundApplications
+ listUnboundApplications,
+ bindAlmOrganization
} from '../../../api/alm-integration';
import { getSubscriptionPlans } from '../../../api/billing';
import * as api from '../../../api/organizations';
@@ -62,6 +65,7 @@ import { get, remove } from '../../../helpers/storage';
import { slugify } from '../../../helpers/strings';
import { getOrganizationUrl } from '../../../helpers/urls';
import { skipOnboarding } from '../../../store/users';
+import addGlobalSuccessMessage from '../../../app/utils/addGlobalSuccessMessage';
import '../../tutorials/styles.css'; // TODO remove me
interface Props {
@@ -82,6 +86,7 @@ interface State {
almOrganization?: T.AlmOrganization;
almOrgLoading: boolean;
almUnboundApplications: T.AlmUnboundApplication[];
+ bindingExistingOrg: boolean;
boundOrganization?: T.OrganizationBase;
loading: boolean;
organization?: T.Organization;
@@ -102,6 +107,7 @@ export class CreateOrganization extends React.PureComponent<Props & WithRouterPr
state: State = {
almOrgLoading: false,
almUnboundApplications: [],
+ bindingExistingOrg: false,
loading: true,
step: Step.OrganizationDetails
};
@@ -109,18 +115,29 @@ export class CreateOrganization extends React.PureComponent<Props & WithRouterPr
componentDidMount() {
this.mounted = true;
addWhitePageClass();
- const initRequests = [this.fetchSubscriptionPlans()];
- if (hasAdvancedALMIntegration(this.props.currentUser)) {
- initRequests.push(this.fetchAlmApplication());
- const query = parseQuery(this.props.location.query);
- if (query.almInstallId) {
- this.fetchAlmOrganization(query.almInstallId);
- } else {
- initRequests.push(this.fetchAlmUnboundApplications());
+ const query = parseQuery(this.props.location.query);
+
+ //highjack the process for the organization settings
+ if (
+ hasAdvancedALMIntegration(this.props.currentUser) &&
+ query.almInstallId &&
+ this.isStoredTimestampValid(BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP)
+ ) {
+ this.bindAndRedirectToOrganizationSettings(query.almInstallId);
+ } else {
+ const initRequests = [this.fetchSubscriptionPlans()];
+ if (hasAdvancedALMIntegration(this.props.currentUser)) {
+ initRequests.push(this.fetchAlmApplication());
+
+ if (query.almInstallId) {
+ this.fetchAlmOrganization(query.almInstallId);
+ } else {
+ initRequests.push(this.fetchAlmUnboundApplications());
+ }
}
+ Promise.all(initRequests).then(this.stopLoading, this.stopLoading);
}
- Promise.all(initRequests).then(this.stopLoading, this.stopLoading);
}
componentDidUpdate(prevProps: WithRouterProps) {
@@ -249,6 +266,36 @@ export class CreateOrganization extends React.PureComponent<Props & WithRouterPr
this.updateUrlState({ tab });
};
+ bindAndRedirectToOrganizationSettings(installationId: string) {
+ const organizationKey = get(BIND_ORGANIZATION_KEY) || '';
+ remove(BIND_ORGANIZATION_KEY);
+
+ this.setState({ bindingExistingOrg: true });
+
+ bindAlmOrganization({
+ installationId,
+ organization: organizationKey
+ }).then(
+ () => {
+ this.props.router.push({
+ pathname: `/organizations/${organizationKey}`
+ });
+ addGlobalSuccessMessage(translate('organization.bind.success'));
+ },
+ () => {}
+ );
+ }
+
+ getHeader = (bindingExistingOrg: boolean, importPersonalOrg: boolean) => {
+ if (bindingExistingOrg) {
+ return translate('onboarding.binding_organization');
+ } else if (importPersonalOrg) {
+ return translate('onboarding.import_organization.personal.page.header');
+ } else {
+ return translate('onboarding.create_organization.page.header');
+ }
+ };
+
setValidOrgKey = (almOrganization: T.AlmOrganization) => {
const key = slugify(almOrganization.key);
const keys = [key, ...times(9, i => `${key}-${i + 1}`)];
@@ -397,13 +444,12 @@ export class CreateOrganization extends React.PureComponent<Props & WithRouterPr
return <AlmApplicationInstalling almKey={query.almKey} />;
}
- const { almOrganization, subscriptionPlans } = this.state;
+ const { almOrganization, bindingExistingOrg, subscriptionPlans } = this.state;
const importPersonalOrg = isPersonal(almOrganization)
? this.props.userOrganizations.find(o => o.key === currentUser.personalOrganization)
: undefined;
- const header = importPersonalOrg
- ? translate('onboarding.import_organization.personal.page.header')
- : translate('onboarding.create_organization.page.header');
+ const header = this.getHeader(bindingExistingOrg, !!importPersonalOrg);
+
const startedPrice = subscriptionPlans && subscriptionPlans[0] && subscriptionPlans[0].price;
return (
diff --git a/server/sonar-web/src/main/js/apps/create/organization/__tests__/CreateOrganization-test.tsx b/server/sonar-web/src/main/js/apps/create/organization/__tests__/CreateOrganization-test.tsx
index fc80778c403..cdc8020444b 100644
--- a/server/sonar-web/src/main/js/apps/create/organization/__tests__/CreateOrganization-test.tsx
+++ b/server/sonar-web/src/main/js/apps/create/organization/__tests__/CreateOrganization-test.tsx
@@ -20,9 +20,10 @@
import * as React from 'react';
import { times } from 'lodash';
import { Location } from 'history';
-import { shallow } from 'enzyme';
+import { shallow, mount } from 'enzyme';
import { CreateOrganization } from '../CreateOrganization';
import {
+ bindAlmOrganization,
getAlmAppInfo,
getAlmOrganization,
listUnboundApplications
@@ -35,7 +36,8 @@ import {
mockOrganizationWithAdminActions,
mockOrganizationWithAlm,
mockAlmOrganization,
- mockCurrentUser
+ mockCurrentUser,
+ mockLocation
} from '../../../../helpers/testMocks';
import { waitAndUpdate } from '../../../../helpers/testUtils';
@@ -67,7 +69,8 @@ jest.mock('../../../../api/alm-integration', () => ({
url: 'https://www.sonarsource.com'
}
}),
- listUnboundApplications: jest.fn().mockResolvedValue([])
+ listUnboundApplications: jest.fn().mockResolvedValue([]),
+ bindAlmOrganization: jest.fn().mockResolvedValue({})
}));
jest.mock('../../../../api/organizations', () => ({
@@ -246,7 +249,9 @@ it('should display AutoOrganizationCreate with already bound organization', asyn
almOrganization: { ...fooBarAlmOrganization, personal: false },
boundOrganization
});
- (get as jest.Mock<any>).mockReturnValueOnce(Date.now().toString());
+ (get as jest.Mock<any>)
+ .mockReturnValueOnce(undefined) // For BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP
+ .mockReturnValueOnce(Date.now().toString()); // For ORGANIZATION_IMPORT_BINDING_IN_PROGRESS_TIMESTAMP
const push = jest.fn();
const wrapper = shallowRender({
currentUser: { ...user, externalProvider: 'github' },
@@ -294,18 +299,50 @@ it('should cancel imports', async () => {
const wrapper = shallowRender({ router: mockRouter({ push }) });
await waitAndUpdate(wrapper);
wrapper.instance().handleCancelImport();
- expect(push).toBeCalledWith({ query: {} });
+ expect(push).toBeCalledWith({ pathname: '/path', query: {}, state: {} });
});
+it('should bind org and redirect to org home when coming from org binding', async () => {
+ const installation_id = '5328';
+ const orgKey = 'org4test';
+ const push = jest.fn();
+
+ (get as jest.Mock<any>)
+ .mockReturnValueOnce(Date.now().toString()) // For BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP
+ .mockReturnValueOnce(orgKey); // For BIND_ORGANIZATION_KEY
+
+ const wrapper = mountRender({
+ currentUser: mockCurrentUser({ ...user, externalProvider: 'github' }),
+ location: mockLocation({ query: { installation_id } }),
+ router: mockRouter({ push })
+ });
+ await waitAndUpdate(wrapper);
+
+ expect(bindAlmOrganization).toBeCalled();
+ expect(getAlmOrganization).not.toBeCalled();
+ expect(push).toBeCalledWith({
+ pathname: `/organizations/${orgKey}`
+ });
+});
+
+function mountRender(props: Partial<CreateOrganization['props']> = {}) {
+ return mount<CreateOrganization>(createComponent(props));
+}
+
function shallowRender(props: Partial<CreateOrganization['props']> = {}) {
- return shallow<CreateOrganization>(
+ return shallow<CreateOrganization>(createComponent(props));
+}
+
+function createComponent(props: Partial<CreateOrganization['props']> = {}) {
+ return (
<CreateOrganization
createOrganization={jest.fn()}
currentUser={user}
deleteOrganization={jest.fn()}
- // @ts-ignore avoid passing everything from WithRouterProps
- location={{}}
+ location={mockLocation()}
+ params={{}}
router={mockRouter()}
+ routes={[]}
skipOnboarding={jest.fn()}
updateOrganization={jest.fn()}
userOrganizations={[
diff --git a/server/sonar-web/src/main/js/apps/create/organization/utils.ts b/server/sonar-web/src/main/js/apps/create/organization/utils.ts
index b6d88841e2b..0d9485bba1a 100644
--- a/server/sonar-web/src/main/js/apps/create/organization/utils.ts
+++ b/server/sonar-web/src/main/js/apps/create/organization/utils.ts
@@ -35,6 +35,10 @@ export const ORGANIZATION_IMPORT_BINDING_IN_PROGRESS_TIMESTAMP =
export const ORGANIZATION_IMPORT_REDIRECT_TO_PROJECT_TIMESTAMP =
'sonarcloud.import_org.redirect_to_projects';
+export const BIND_ORGANIZATION_KEY = 'sonarcloud.bind_org.key';
+
+export const BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP = 'sonarcloud.bind_org.redirect_to_org';
+
export enum Step {
OrganizationDetails,
Plan
diff --git a/server/sonar-web/src/main/js/apps/organizations/components/OrganizationBind.tsx b/server/sonar-web/src/main/js/apps/organizations/components/OrganizationBind.tsx
new file mode 100644
index 00000000000..a46360015c6
--- /dev/null
+++ b/server/sonar-web/src/main/js/apps/organizations/components/OrganizationBind.tsx
@@ -0,0 +1,126 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2019 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 * as React from 'react';
+import {
+ BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP,
+ BIND_ORGANIZATION_KEY
+} from '../../create/organization/utils';
+import IdentityProviderLink from '../../../components/ui/IdentityProviderLink';
+import { getAlmAppInfo } from '../../../api/alm-integration';
+import { sanitizeAlmId } from '../../../helpers/almIntegrations';
+import { translateWithParameters, translate } from '../../../helpers/l10n';
+import { save } from '../../../helpers/storage';
+import { getBaseUrl } from '../../../helpers/urls';
+
+interface Props {
+ currentUser: T.LoggedInUser;
+ organization: T.Organization;
+}
+
+interface State {
+ almApplication?: T.AlmApplication;
+}
+
+export default class OrganizationBind extends React.PureComponent<Props, State> {
+ mounted = false;
+ state: State = {};
+
+ componentDidMount() {
+ this.mounted = true;
+ this.fetchAlmApplication();
+ }
+
+ componentWillUnmount() {
+ this.mounted = false;
+ }
+
+ fetchAlmApplication = () => {
+ return getAlmAppInfo().then(({ application }) => {
+ if (this.mounted) {
+ this.setState({ almApplication: application });
+ }
+ });
+ };
+
+ handleInstallAppClick = () => {
+ save(BIND_ORGANIZATION_KEY, this.props.organization.key);
+ save(BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP, Date.now().toString());
+ };
+
+ render() {
+ const { currentUser, organization } = this.props;
+
+ const { almApplication } = this.state;
+
+ const almKey = sanitizeAlmId(currentUser.externalProvider || '');
+ const orgAlmKey = organization.alm ? sanitizeAlmId(organization.alm.key) : '';
+ return (
+ <div className="boxed-group boxed-group-inner">
+ <h2 className="boxed-title">
+ {translateWithParameters('organization.bind_to_x', translate(almKey))}
+ </h2>
+ {organization.alm ? (
+ <>
+ <span>{translate('organization.bound')}</span>
+ <a
+ className="link-no-underline big-spacer-left"
+ href={organization.alm.url}
+ rel="noopener noreferrer"
+ target="_blank">
+ <img
+ alt={translate(orgAlmKey)}
+ className="text-text-top little-spacer-right"
+ height={16}
+ src={`${getBaseUrl()}/images/sonarcloud/${orgAlmKey}.svg`}
+ width={16}
+ />
+ {translateWithParameters('organization.see_on_x', translate(orgAlmKey))}
+ </a>
+ </>
+ ) : (
+ <>
+ <p className="spacer-bottom">
+ {translateWithParameters('organization.binding_with_x_easy_sync', translate(almKey))}
+ </p>
+ <p className="big-spacer-bottom">
+ {translateWithParameters(
+ 'organization.app_will_be_installed_on_x',
+ translate(almKey)
+ )}
+ </p>
+ {almApplication && (
+ <IdentityProviderLink
+ className="display-inline-block"
+ identityProvider={almApplication}
+ onClick={this.handleInstallAppClick}
+ small={true}
+ url={almApplication.installationUrl}>
+ {translate(
+ 'onboarding.import_organization.choose_the_organization_button',
+ almApplication.key
+ )}
+ </IdentityProviderLink>
+ )}
+ </>
+ )}
+ </div>
+ );
+ }
+}
diff --git a/server/sonar-web/src/main/js/apps/organizations/components/OrganizationDelete.tsx b/server/sonar-web/src/main/js/apps/organizations/components/OrganizationDelete.tsx
index 6954e3bf7e7..ab685c20442 100644
--- a/server/sonar-web/src/main/js/apps/organizations/components/OrganizationDelete.tsx
+++ b/server/sonar-web/src/main/js/apps/organizations/components/OrganizationDelete.tsx
@@ -18,7 +18,6 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
import * as React from 'react';
-import Helmet from 'react-helmet';
import { connect } from 'react-redux';
import ConfirmButton from '../../../components/controls/ConfirmButton';
import InstanceMessage from '../../../components/common/InstanceMessage';
@@ -111,59 +110,53 @@ export class OrganizationDelete extends React.PureComponent<Props, State> {
render() {
const { hasPaidPlan } = this.state;
- const title = translate('organization.delete');
return (
- <>
- <Helmet title={title} />
- <div className="page page-limited">
- <header className="page-header">
- <h1 className="page-title">{title}</h1>
- <div className="page-description">
- <InstanceMessage message={translate('organization.delete.description')} />
- </div>
- </header>
- <ConfirmButton
- confirmButtonText={translate('delete')}
- confirmDisable={!this.isVerified()}
- isDestructive={true}
- modalBody={
- <div>
- {hasPaidPlan && (
- <Alert variant="warning">
- {translate('organization.delete.sonarcloud.paid_plan_info')}
- </Alert>
- )}
- <p>{translate('organization.delete.question')}</p>
- <div className="spacer-top">
- <label htmlFor="downgrade-organization-name">
- {translate('billing.downgrade.modal.type_to_proceed')}
- </label>
- <div className="little-spacer-top">
- <input
- autoFocus={true}
- className="input-super-large"
- id="downgrade-organization-name"
- onChange={this.handleInput}
- type="text"
- value={this.state.verify}
- />
- </div>
+ <div className="boxed-group boxed-group-inner">
+ <h2 className="boxed-title">{translate('organization.delete')}</h2>
+ <p className="big-spacer-bottom width-50">
+ <InstanceMessage message={translate('organization.delete.description')} />
+ </p>
+ <ConfirmButton
+ confirmButtonText={translate('delete')}
+ confirmDisable={!this.isVerified()}
+ isDestructive={true}
+ modalBody={
+ <div>
+ {hasPaidPlan && (
+ <Alert variant="warning">
+ {translate('organization.delete.sonarcloud.paid_plan_info')}
+ </Alert>
+ )}
+ <p>{translate('organization.delete.question')}</p>
+ <div className="spacer-top">
+ <label htmlFor="downgrade-organization-name">
+ {translate('billing.downgrade.modal.type_to_proceed')}
+ </label>
+ <div className="little-spacer-top">
+ <input
+ autoFocus={true}
+ className="input-super-large"
+ id="downgrade-organization-name"
+ onChange={this.handleInput}
+ type="text"
+ value={this.state.verify}
+ />
</div>
</div>
- }
- modalHeader={translateWithParameters(
- 'organization.delete_x',
- this.props.organization.name
- )}
- onConfirm={this.onDelete}>
- {({ onClick }) => (
- <Button className="js-custom-measure-delete button-red" onClick={onClick}>
- {translate('delete')}
- </Button>
- )}
- </ConfirmButton>
- </div>
- </>
+ </div>
+ }
+ modalHeader={translateWithParameters(
+ 'organization.delete_x',
+ this.props.organization.name
+ )}
+ onConfirm={this.onDelete}>
+ {({ onClick }) => (
+ <Button className="js-custom-measure-delete button-red" onClick={onClick}>
+ {translate('delete')}
+ </Button>
+ )}
+ </ConfirmButton>
+ </div>
);
}
}
diff --git a/server/sonar-web/src/main/js/apps/organizations/components/OrganizationEdit.tsx b/server/sonar-web/src/main/js/apps/organizations/components/OrganizationEdit.tsx
index 4d151adf2ab..77942da7b55 100644
--- a/server/sonar-web/src/main/js/apps/organizations/components/OrganizationEdit.tsx
+++ b/server/sonar-web/src/main/js/apps/organizations/components/OrganizationEdit.tsx
@@ -21,9 +21,13 @@ import * as React from 'react';
import Helmet from 'react-helmet';
import { connect } from 'react-redux';
import { debounce } from 'lodash';
+import OrganizationBind from './OrganizationBind';
+import OrganizationDelete from './OrganizationDelete';
+import { updateOrganization } from '../actions';
import OrganizationAvatar from '../../../components/common/OrganizationAvatar';
+import { whenLoggedIn } from '../../../components/hoc/whenLoggedIn';
import { SubmitButton } from '../../../components/ui/buttons';
-import { updateOrganization } from '../actions';
+import { hasAdvancedALMIntegration } from '../../../helpers/almIntegrations';
import { translate } from '../../../helpers/l10n';
interface DispatchProps {
@@ -31,6 +35,7 @@ interface DispatchProps {
}
interface OwnProps {
+ currentUser: T.LoggedInUser;
organization: T.Organization;
}
@@ -100,7 +105,12 @@ export class OrganizationEdit extends React.PureComponent<Props, State> {
};
render() {
- const title = translate('organization.edit');
+ const { currentUser, organization } = this.props;
+ const title = translate('organization.settings');
+
+ const showBinding = hasAdvancedALMIntegration(currentUser);
+ const showDelete = organization.actions && organization.actions.delete;
+
return (
<div className="page page-limited">
<Helmet title={title} />
@@ -110,6 +120,7 @@ export class OrganizationEdit extends React.PureComponent<Props, State> {
</header>
<div className="boxed-group boxed-group-inner">
+ <h2 className="boxed-title">{translate('organization.details')}</h2>
<form onSubmit={this.handleSubmit}>
<div className="form-field">
<label htmlFor="organization-name">
@@ -198,6 +209,10 @@ export class OrganizationEdit extends React.PureComponent<Props, State> {
{this.state.loading && <i className="spinner spacer-left" />}
</form>
</div>
+
+ {showBinding && <OrganizationBind currentUser={currentUser} organization={organization} />}
+
+ {showDelete && <OrganizationDelete organization={organization} />}
</div>
);
}
@@ -208,4 +223,4 @@ const mapDispatchToProps = { updateOrganization: updateOrganization as any };
export default connect(
null,
mapDispatchToProps
-)(OrganizationEdit);
+)(whenLoggedIn(OrganizationEdit));
diff --git a/server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationBind-test.tsx b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationBind-test.tsx
new file mode 100644
index 00000000000..079de73c67e
--- /dev/null
+++ b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationBind-test.tsx
@@ -0,0 +1,72 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2019 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 * as React from 'react';
+import { shallow } from 'enzyme';
+import OrganizationBind from '../OrganizationBind';
+import {
+ BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP,
+ BIND_ORGANIZATION_KEY
+} from '../../../create/organization/utils';
+import { getAlmAppInfo } from '../../../../api/alm-integration';
+import { save } from '../../../../helpers/storage';
+import {
+ mockAlmApplication,
+ mockCurrentUser,
+ mockOrganization
+} from '../../../../helpers/testMocks';
+
+jest.mock('../../../../api/alm-integration', () => ({
+ getAlmAppInfo: jest.fn(() => Promise.resolve({ application: mockAlmApplication() }))
+}));
+jest.mock('../../../../helpers/storage', () => ({
+ save: jest.fn()
+}));
+
+jest.mock('../../../../helpers/system', () => ({ isSonarCloud: jest.fn() }));
+
+beforeEach(() => {
+ (getAlmAppInfo as jest.Mock<any>).mockClear();
+});
+
+it('should render correctly', () => {
+ expect(shallowRender()).toMatchSnapshot();
+});
+
+it('should save state when handling Install App click', () => {
+ const orgKey = '56346';
+ shallowRender({ organization: mockOrganization({ key: orgKey }) })
+ .instance()
+ .handleInstallAppClick();
+
+ expect(save).toBeCalledTimes(2);
+ expect(save).nthCalledWith(1, BIND_ORGANIZATION_KEY, orgKey);
+ const secondCallArguments = (save as jest.Mock<any>).mock.calls[1];
+ expect(secondCallArguments[0]).toBe(BIND_ORGANIZATION_REDIRECT_TO_ORG_TIMESTAMP);
+});
+
+function shallowRender(props: Partial<OrganizationBind['props']> = {}) {
+ return shallow<OrganizationBind>(
+ <OrganizationBind
+ currentUser={mockCurrentUser()}
+ organization={mockOrganization()}
+ {...props}
+ />
+ );
+}
diff --git a/server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationEdit-test.tsx b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationEdit-test.tsx
index 1e320c7865b..1daa8265779 100644
--- a/server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationEdit-test.tsx
+++ b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/OrganizationEdit-test.tsx
@@ -20,11 +20,16 @@
import * as React from 'react';
import { shallow } from 'enzyme';
import { OrganizationEdit } from '../OrganizationEdit';
+import { mockCurrentUser } from '../../../../helpers/testMocks';
it('smoke test', () => {
const organization = { key: 'foo', name: 'Foo' };
const wrapper = shallow(
- <OrganizationEdit organization={organization} updateOrganization={jest.fn()} />
+ <OrganizationEdit
+ currentUser={mockCurrentUser()}
+ organization={organization}
+ updateOrganization={jest.fn()}
+ />
);
expect(wrapper).toMatchSnapshot();
diff --git a/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationBind-test.tsx.snap b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationBind-test.tsx.snap
new file mode 100644
index 00000000000..69821637e05
--- /dev/null
+++ b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationBind-test.tsx.snap
@@ -0,0 +1,23 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`should render correctly 1`] = `
+<div
+ className="boxed-group boxed-group-inner"
+>
+ <h2
+ className="boxed-title"
+ >
+ organization.bind_to_x.
+ </h2>
+ <p
+ className="spacer-bottom"
+ >
+ organization.binding_with_x_easy_sync.
+ </p>
+ <p
+ className="big-spacer-bottom"
+ >
+ organization.app_will_be_installed_on_x.
+ </p>
+</div>
+`;
diff --git a/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationDelete-test.tsx.snap b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationDelete-test.tsx.snap
index 43b66bfaea3..c2dee60e8f3 100644
--- a/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationDelete-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationDelete-test.tsx.snap
@@ -1,140 +1,118 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`should show a info message for paying organization 1`] = `
-<Fragment>
- <HelmetWrapper
- defer={true}
- encodeSpecialCharacters={true}
- title="organization.delete"
- />
- <div
- className="page page-limited"
+<div
+ className="boxed-group boxed-group-inner"
+>
+ <h2
+ className="boxed-title"
>
- <header
- className="page-header"
- >
- <h1
- className="page-title"
- >
- organization.delete
- </h1>
- <div
- className="page-description"
- >
- <InstanceMessage
- message="organization.delete.description"
- />
- </div>
- </header>
- <ConfirmButton
- confirmButtonText="delete"
- confirmDisable={true}
- isDestructive={true}
- modalBody={
- <div>
- <Alert
- variant="warning"
+ organization.delete
+ </h2>
+ <p
+ className="big-spacer-bottom width-50"
+ >
+ <InstanceMessage
+ message="organization.delete.description"
+ />
+ </p>
+ <ConfirmButton
+ confirmButtonText="delete"
+ confirmDisable={true}
+ isDestructive={true}
+ modalBody={
+ <div>
+ <Alert
+ variant="warning"
+ >
+ organization.delete.sonarcloud.paid_plan_info
+ </Alert>
+ <p>
+ organization.delete.question
+ </p>
+ <div
+ className="spacer-top"
+ >
+ <label
+ htmlFor="downgrade-organization-name"
>
- organization.delete.sonarcloud.paid_plan_info
- </Alert>
- <p>
- organization.delete.question
- </p>
+ billing.downgrade.modal.type_to_proceed
+ </label>
<div
- className="spacer-top"
+ className="little-spacer-top"
>
- <label
- htmlFor="downgrade-organization-name"
- >
- billing.downgrade.modal.type_to_proceed
- </label>
- <div
- className="little-spacer-top"
- >
- <input
- autoFocus={true}
- className="input-super-large"
- id="downgrade-organization-name"
- onChange={[Function]}
- type="text"
- value=""
- />
- </div>
+ <input
+ autoFocus={true}
+ className="input-super-large"
+ id="downgrade-organization-name"
+ onChange={[Function]}
+ type="text"
+ value=""
+ />
</div>
</div>
- }
- modalHeader="organization.delete_x.Foo"
- onConfirm={[Function]}
- >
- <Component />
- </ConfirmButton>
- </div>
-</Fragment>
+ </div>
+ }
+ modalHeader="organization.delete_x.Foo"
+ onConfirm={[Function]}
+ >
+ <Component />
+ </ConfirmButton>
+</div>
`;
exports[`smoke test 1`] = `
-<Fragment>
- <HelmetWrapper
- defer={true}
- encodeSpecialCharacters={true}
- title="organization.delete"
- />
- <div
- className="page page-limited"
+<div
+ className="boxed-group boxed-group-inner"
+>
+ <h2
+ className="boxed-title"
>
- <header
- className="page-header"
- >
- <h1
- className="page-title"
- >
- organization.delete
- </h1>
- <div
- className="page-description"
- >
- <InstanceMessage
- message="organization.delete.description"
- />
- </div>
- </header>
- <ConfirmButton
- confirmButtonText="delete"
- confirmDisable={true}
- isDestructive={true}
- modalBody={
- <div>
- <p>
- organization.delete.question
- </p>
+ organization.delete
+ </h2>
+ <p
+ className="big-spacer-bottom width-50"
+ >
+ <InstanceMessage
+ message="organization.delete.description"
+ />
+ </p>
+ <ConfirmButton
+ confirmButtonText="delete"
+ confirmDisable={true}
+ isDestructive={true}
+ modalBody={
+ <div>
+ <p>
+ organization.delete.question
+ </p>
+ <div
+ className="spacer-top"
+ >
+ <label
+ htmlFor="downgrade-organization-name"
+ >
+ billing.downgrade.modal.type_to_proceed
+ </label>
<div
- className="spacer-top"
+ className="little-spacer-top"
>
- <label
- htmlFor="downgrade-organization-name"
- >
- billing.downgrade.modal.type_to_proceed
- </label>
- <div
- className="little-spacer-top"
- >
- <input
- autoFocus={true}
- className="input-super-large"
- id="downgrade-organization-name"
- onChange={[Function]}
- type="text"
- value=""
- />
- </div>
+ <input
+ autoFocus={true}
+ className="input-super-large"
+ id="downgrade-organization-name"
+ onChange={[Function]}
+ type="text"
+ value=""
+ />
</div>
</div>
- }
- modalHeader="organization.delete_x.Foo"
- onConfirm={[Function]}
- >
- <Component />
- </ConfirmButton>
- </div>
-</Fragment>
+ </div>
+ }
+ modalHeader="organization.delete_x.Foo"
+ onConfirm={[Function]}
+ >
+ <Component />
+ </ConfirmButton>
+</div>
`;
diff --git a/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationEdit-test.tsx.snap b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationEdit-test.tsx.snap
index e948f1810c5..b80d299eb16 100644
--- a/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationEdit-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/organizations/components/__tests__/__snapshots__/OrganizationEdit-test.tsx.snap
@@ -7,7 +7,7 @@ exports[`smoke test 1`] = `
<HelmetWrapper
defer={true}
encodeSpecialCharacters={true}
- title="organization.edit"
+ title="organization.settings"
/>
<header
className="page-header"
@@ -15,12 +15,17 @@ exports[`smoke test 1`] = `
<h1
className="page-title"
>
- organization.edit
+ organization.settings
</h1>
</header>
<div
className="boxed-group boxed-group-inner"
>
+ <h2
+ className="boxed-title"
+ >
+ organization.details
+ </h2>
<form
onSubmit={[Function]}
>
@@ -162,7 +167,7 @@ exports[`smoke test 2`] = `
<HelmetWrapper
defer={true}
encodeSpecialCharacters={true}
- title="organization.edit"
+ title="organization.settings"
/>
<header
className="page-header"
@@ -170,12 +175,17 @@ exports[`smoke test 2`] = `
<h1
className="page-title"
>
- organization.edit
+ organization.settings
</h1>
</header>
<div
className="boxed-group boxed-group-inner"
>
+ <h2
+ className="boxed-title"
+ >
+ organization.details
+ </h2>
<form
onSubmit={[Function]}
>
@@ -317,7 +327,7 @@ exports[`smoke test 3`] = `
<HelmetWrapper
defer={true}
encodeSpecialCharacters={true}
- title="organization.edit"
+ title="organization.settings"
/>
<header
className="page-header"
@@ -325,12 +335,17 @@ exports[`smoke test 3`] = `
<h1
className="page-title"
>
- organization.edit
+ organization.settings
</h1>
</header>
<div
className="boxed-group boxed-group-inner"
>
+ <h2
+ className="boxed-title"
+ >
+ organization.details
+ </h2>
<form
onSubmit={[Function]}
>
diff --git a/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigation.tsx b/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigation.tsx
index 3c7bbdb76a8..e9219a6bffd 100644
--- a/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigation.tsx
+++ b/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigation.tsx
@@ -18,7 +18,7 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
import * as React from 'react';
-import OrganizationNavigationHeaderContainer from './OrganizationNavigationHeaderContainer';
+import OrganizationNavigationHeader from './OrganizationNavigationHeader';
import OrganizationNavigationMeta from './OrganizationNavigationMeta';
import OrganizationNavigationMenuContainer from './OrganizationNavigationMenuContainer';
import * as theme from '../../../app/theme';
@@ -40,7 +40,11 @@ export default function OrganizationNavigation({
return (
<ContextNavBar height={theme.contextNavHeightRaw} id="context-navigation">
<div className="navbar-context-justified">
- <OrganizationNavigationHeaderContainer organization={organization} />
+ <OrganizationNavigationHeader
+ currentUser={currentUser}
+ organization={organization}
+ organizations={userOrganizations}
+ />
<OrganizationNavigationMeta
currentUser={currentUser}
organization={organization}
diff --git a/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationAdministration.tsx b/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationAdministration.tsx
index 63228606f33..f1eb48f7807 100644
--- a/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationAdministration.tsx
+++ b/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationAdministration.tsx
@@ -32,7 +32,6 @@ interface Props {
const ADMIN_PATHS = [
'edit',
'groups',
- 'delete',
'permissions',
'permission_templates',
'projects_management',
@@ -40,7 +39,7 @@ const ADMIN_PATHS = [
];
export default function OrganizationNavigationAdministration({ location, organization }: Props) {
- const { actions = {}, adminPages = [] } = organization;
+ const { adminPages = [] } = organization;
const adminPathsWithExtensions = adminPages.map(e => `extension/${e.key}`).concat(ADMIN_PATHS);
const adminActive = adminPathsWithExtensions.some(path =>
location.pathname.endsWith(`organizations/${organization.key}/${path}`)
@@ -50,6 +49,11 @@ export default function OrganizationNavigationAdministration({ location, organiz
<Dropdown
overlay={
<ul className="menu">
+ <li>
+ <Link activeClassName="active" to={`/organizations/${organization.key}/edit`}>
+ {translate('organization.settings')}
+ </Link>
+ </li>
{adminPages.map(extension => (
<li key={extension.key}>
<Link
@@ -88,18 +92,6 @@ export default function OrganizationNavigationAdministration({ location, organiz
{translate('webhooks.page')}
</Link>
</li>
- <li>
- <Link activeClassName="active" to={`/organizations/${organization.key}/edit`}>
- {translate('edit')}
- </Link>
- </li>
- {actions.delete && (
- <li>
- <Link activeClassName="active" to={`/organizations/${organization.key}/delete`}>
- {translate('delete')}
- </Link>
- </li>
- )}
</ul>
}
tagName="li">
diff --git a/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeader.tsx b/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeader.tsx
index 2f7426880c4..28b8d1cb6d9 100644
--- a/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeader.tsx
+++ b/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeader.tsx
@@ -18,22 +18,63 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
import * as React from 'react';
+import { Link } from 'react-router';
import { sortBy } from 'lodash';
import OrganizationAvatar from '../../../components/common/OrganizationAvatar';
import Dropdown from '../../../components/controls/Dropdown';
+import Tooltip from '../../../components/controls/Tooltip';
import DropdownIcon from '../../../components/icons-components/DropdownIcon';
import OrganizationListItem from '../../../components/ui/OrganizationListItem';
-import { sanitizeAlmId } from '../../../helpers/almIntegrations';
+import {
+ sanitizeAlmId,
+ hasAdvancedALMIntegration,
+ getUserAlmKey
+} from '../../../helpers/almIntegrations';
+import { translate, translateWithParameters } from '../../../helpers/l10n';
import { getBaseUrl } from '../../../helpers/urls';
-interface Props {
+export interface Props {
+ currentUser: T.CurrentUser;
organization: T.Organization;
organizations: T.Organization[];
}
-export default function OrganizationNavigationHeader({ organization, organizations }: Props) {
+export default function OrganizationNavigationHeader({
+ currentUser,
+ organization,
+ organizations
+}: Props) {
const other = organizations.filter(o => o.key !== organization.key);
+ let almKey;
+ let tooltipContent;
+ let tooltipIconSrc;
+ if (organization.alm) {
+ almKey = sanitizeAlmId(organization.alm.key);
+ tooltipContent = (
+ <>
+ <p>{translateWithParameters('organization.bound_to_x', translate(almKey))}</p>
+ <hr className="spacer-top spacer-bottom" />
+ <a href={organization.alm.url} rel="noopener noreferrer" target="_blank">
+ {translateWithParameters('organization.see_on_x', translate(almKey))}
+ </a>
+ </>
+ );
+ tooltipIconSrc = `${getBaseUrl()}/images/sonarcloud/${almKey}.svg`;
+ } else if (hasAdvancedALMIntegration(currentUser)) {
+ almKey = getUserAlmKey(currentUser) || '';
+ tooltipContent = (
+ <>
+ <p>{translateWithParameters('organization.not_bound_to_x', translate(almKey))}</p>
+ <hr className="spacer-top spacer-bottom" />
+ <Link to={`/organizations/${organization.key}/edit`}>
+ {translate('organization.go_to_settings_to_bind')}
+ </Link>
+ </>
+ );
+ tooltipIconSrc = `${getBaseUrl()}/images/sonarcloud/${almKey}-unbound.svg`;
+ }
+
return (
<header className="navbar-context-header">
<OrganizationAvatar organization={organization} />
@@ -57,20 +98,16 @@ export default function OrganizationNavigationHeader({ organization, organizatio
) : (
<span className="spacer-left">{organization.name}</span>
)}
- {organization.alm && (
- <a
- className="link-no-underline"
- href={organization.alm.url}
- rel="noopener noreferrer"
- target="_blank">
+ {almKey && (
+ <Tooltip mouseLeaveDelay={0.25} overlay={tooltipContent}>
<img
- alt={sanitizeAlmId(organization.alm.key)}
- className="text-text-top spacer-left"
+ alt={translate(almKey)}
+ className="text-middle spacer-left"
height={16}
- src={`${getBaseUrl()}/images/sonarcloud/${sanitizeAlmId(organization.alm.key)}.svg`}
+ src={tooltipIconSrc}
width={16}
/>
- </a>
+ </Tooltip>
)}
{organization.description != null && (
<div className="navbar-context-description">
diff --git a/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeaderContainer.tsx b/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeaderContainer.tsx
deleted file mode 100644
index 999b68b635f..00000000000
--- a/server/sonar-web/src/main/js/apps/organizations/navigation/OrganizationNavigationHeaderContainer.tsx
+++ /dev/null
@@ -1,32 +0,0 @@
-/*
- * SonarQube
- * Copyright (C) 2009-2019 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 { connect } from 'react-redux';
-import OrganizationNavigationHeader from './OrganizationNavigationHeader';
-import { getMyOrganizations, Store } from '../../../store/rootReducer';
-
-interface StateProps {
- organizations: T.Organization[];
-}
-
-const mapStateToProps = (state: Store): StateProps => ({
- organizations: getMyOrganizations(state)
-});
-
-export default connect(mapStateToProps)(OrganizationNavigationHeader);
diff --git a/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/OrganizationNavigationHeader-test.tsx b/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/OrganizationNavigationHeader-test.tsx
index 39519ffd939..12f1d27fb4c 100644
--- a/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/OrganizationNavigationHeader-test.tsx
+++ b/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/OrganizationNavigationHeader-test.tsx
@@ -19,28 +19,22 @@
*/
import * as React from 'react';
import { shallow } from 'enzyme';
-import OrganizationNavigationHeader from '../OrganizationNavigationHeader';
-import { mockOrganizationWithAlm } from '../../../../helpers/testMocks';
+import OrganizationNavigationHeader, { Props } from '../OrganizationNavigationHeader';
+import { mockOrganizationWithAlm, mockCurrentUser } from '../../../../helpers/testMocks';
it('renders', () => {
+ expect(shallowRender()).toMatchSnapshot();
+});
+
+it('renders with alm integration', () => {
expect(
- shallow(
- <OrganizationNavigationHeader
- organization={{ key: 'foo', name: 'Foo', projectVisibility: 'public' }}
- organizations={[]}
- />
- )
+ shallowRender({ organization: mockOrganizationWithAlm({ projectVisibility: 'public' }) })
).toMatchSnapshot();
});
-it('renders with alm integration', () => {
+it('renders for external user w/o alm integration', () => {
expect(
- shallow(
- <OrganizationNavigationHeader
- organization={mockOrganizationWithAlm({ projectVisibility: 'public' })}
- organizations={[]}
- />
- )
+ shallowRender({ currentUser: mockCurrentUser({ externalProvider: 'github' }) })
).toMatchSnapshot();
});
@@ -49,15 +43,23 @@ it('renders dropdown', () => {
{ actions: { admin: true }, key: 'org1', name: 'org1', projectVisibility: 'public' },
{ actions: { admin: false }, key: 'org2', name: 'org2', projectVisibility: 'public' }
];
- const wrapper = shallow(
+ const wrapper = shallowRender({
+ organizations
+ });
+ expect(wrapper.find('Dropdown')).toMatchSnapshot();
+});
+
+function shallowRender(props: Partial<Props> = {}) {
+ return shallow(
<OrganizationNavigationHeader
+ currentUser={mockCurrentUser()}
organization={{
key: 'foo',
name: 'Foo',
projectVisibility: 'public'
}}
- organizations={organizations}
+ organizations={[]}
+ {...props}
/>
);
- expect(wrapper.find('Dropdown')).toMatchSnapshot();
-});
+}
diff --git a/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigation-test.tsx.snap b/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigation-test.tsx.snap
index 0da0558483a..1db61a2ee48 100644
--- a/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigation-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigation-test.tsx.snap
@@ -8,7 +8,12 @@ exports[`render 1`] = `
<div
className="navbar-context-justified"
>
- <Connect(OrganizationNavigationHeader)
+ <OrganizationNavigationHeader
+ currentUser={
+ Object {
+ "isLoggedIn": false,
+ }
+ }
organization={
Object {
"key": "foo",
@@ -16,6 +21,7 @@ exports[`render 1`] = `
"projectVisibility": "public",
}
}
+ organizations={Array []}
/>
<OrganizationNavigationMeta
currentUser={
diff --git a/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationAdministration-test.tsx.snap b/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationAdministration-test.tsx.snap
index 27d6f917f78..da8dbbc2c87 100644
--- a/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationAdministration-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationAdministration-test.tsx.snap
@@ -11,6 +11,16 @@ exports[`renders 1`] = `
activeClassName="active"
onlyActiveOnIndex={false}
style={Object {}}
+ to="/organizations/foo/edit"
+ >
+ organization.settings
+ </Link>
+ </li>
+ <li>
+ <Link
+ activeClassName="active"
+ onlyActiveOnIndex={false}
+ style={Object {}}
to="/organizations/foo/groups"
>
user_groups.page
@@ -56,16 +66,6 @@ exports[`renders 1`] = `
webhooks.page
</Link>
</li>
- <li>
- <Link
- activeClassName="active"
- onlyActiveOnIndex={false}
- style={Object {}}
- to="/organizations/foo/edit"
- >
- edit
- </Link>
- </li>
</ul>
}
tagName="li"
diff --git a/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationHeader-test.tsx.snap b/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationHeader-test.tsx.snap
index c02c0831a7f..c580daad4e2 100644
--- a/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationHeader-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/organizations/navigation/__tests__/__snapshots__/OrganizationNavigationHeader-test.tsx.snap
@@ -67,6 +67,55 @@ exports[`renders dropdown 1`] = `
</Dropdown>
`;
+exports[`renders for external user w/o alm integration 1`] = `
+<header
+ className="navbar-context-header"
+>
+ <OrganizationAvatar
+ organization={
+ Object {
+ "key": "foo",
+ "name": "Foo",
+ "projectVisibility": "public",
+ }
+ }
+ />
+ <span
+ className="spacer-left"
+ >
+ Foo
+ </span>
+ <Tooltip
+ mouseLeaveDelay={0.25}
+ overlay={
+ <React.Fragment>
+ <p>
+ organization.not_bound_to_x.github
+ </p>
+ <hr
+ className="spacer-top spacer-bottom"
+ />
+ <Link
+ onlyActiveOnIndex={false}
+ style={Object {}}
+ to="/organizations/foo/edit"
+ >
+ organization.go_to_settings_to_bind
+ </Link>
+ </React.Fragment>
+ }
+ >
+ <img
+ alt="github"
+ className="text-middle spacer-left"
+ height={16}
+ src="/images/sonarcloud/github-unbound.svg"
+ width={16}
+ />
+ </Tooltip>
+</header>
+`;
+
exports[`renders with alm integration 1`] = `
<header
className="navbar-context-header"
@@ -90,19 +139,33 @@ exports[`renders with alm integration 1`] = `
>
Foo
</span>
- <a
- className="link-no-underline"
- href="https://github.com/foo"
- rel="noopener noreferrer"
- target="_blank"
+ <Tooltip
+ mouseLeaveDelay={0.25}
+ overlay={
+ <React.Fragment>
+ <p>
+ organization.bound_to_x.github
+ </p>
+ <hr
+ className="spacer-top spacer-bottom"
+ />
+ <a
+ href="https://github.com/foo"
+ rel="noopener noreferrer"
+ target="_blank"
+ >
+ organization.see_on_x.github
+ </a>
+ </React.Fragment>
+ }
>
<img
alt="github"
- className="text-text-top spacer-left"
+ className="text-middle spacer-left"
height={16}
src="/images/sonarcloud/github.svg"
width={16}
/>
- </a>
+ </Tooltip>
</header>
`;
diff --git a/server/sonar-web/src/main/js/components/nav/ContextNavBar.css b/server/sonar-web/src/main/js/components/nav/ContextNavBar.css
index 871963945d7..fa4a52d5b96 100644
--- a/server/sonar-web/src/main/js/components/nav/ContextNavBar.css
+++ b/server/sonar-web/src/main/js/components/nav/ContextNavBar.css
@@ -93,9 +93,8 @@
.navbar-context-description {
display: inline-block;
line-height: var(--controlHeight);
- margin-left: 16px;
+ margin-left: var(--gridSize);
padding-top: 4px;
- padding-left: 4px;
color: var(--secondFontColor);
font-size: var(--smallFontSize);
}
diff --git a/server/sonar-web/src/main/js/helpers/__tests__/almIntegrations-test.ts b/server/sonar-web/src/main/js/helpers/__tests__/almIntegrations-test.ts
index b2a09af3ab0..de98abb4968 100644
--- a/server/sonar-web/src/main/js/helpers/__tests__/almIntegrations-test.ts
+++ b/server/sonar-web/src/main/js/helpers/__tests__/almIntegrations-test.ts
@@ -23,8 +23,10 @@ import {
isPersonal,
isVSTS,
sanitizeAlmId,
- getAlmMembersUrl
+ getAlmMembersUrl,
+ getUserAlmKey
} from '../almIntegrations';
+import { mockCurrentUser } from '../testMocks';
it('#getAlmMembersUrl', () => {
expect(getAlmMembersUrl('github', 'https://github.com/Foo')).toBe(
@@ -69,3 +71,16 @@ it('#sanitizeAlmId', () => {
expect(sanitizeAlmId('bitbucket')).toBe('bitbucket');
expect(sanitizeAlmId('github')).toBe('github');
});
+
+describe('getUserAlmKey', () => {
+ it('should return sanitized almKey', () => {
+ expect(getUserAlmKey(mockCurrentUser({ externalProvider: 'bitbucketcloud' }))).toBe(
+ 'bitbucket'
+ );
+ });
+
+ it('should return undefined', () => {
+ expect(getUserAlmKey(mockCurrentUser())).toBeUndefined();
+ expect(getUserAlmKey(mockCurrentUser({ isLoggedIn: undefined }))).toBeUndefined();
+ });
+});
diff --git a/server/sonar-web/src/main/js/helpers/almIntegrations.ts b/server/sonar-web/src/main/js/helpers/almIntegrations.ts
index c3cfc0744a3..dfe28e3dd53 100644
--- a/server/sonar-web/src/main/js/helpers/almIntegrations.ts
+++ b/server/sonar-web/src/main/js/helpers/almIntegrations.ts
@@ -29,6 +29,12 @@ export function getAlmMembersUrl(key: string, url: string): string {
return url + 'profile/members';
}
+export function getUserAlmKey(user: T.CurrentUser) {
+ return isLoggedIn(user) && user.externalProvider
+ ? sanitizeAlmId(user.externalProvider)
+ : undefined;
+}
+
export function hasAdvancedALMIntegration(user: T.CurrentUser) {
return (
isLoggedIn(user) && (isBitbucket(user.externalProvider) || isGithub(user.externalProvider))