From 477cfbd296c49853604563e24fc2c9987b4f751e Mon Sep 17 00:00:00 2001 From: Grégoire Aubert Date: Fri, 29 Jun 2018 16:33:30 +0200 Subject: SONAR-10945 QP and QG pages should only be visible only to members of paid organizations * SONAR-10968 Warn user about project privacy after billing upgrade * SONAR-10949 Show QG, QP, members and rules only when the user has correct access * SONAR-10959 Do not display Rules, QP and QG pages for non members of paid organizations * SONAR-10961 Do not display Members page for non member of private organizations * Remove rule permalink in issues page for non members of paid orgs * Do not display QP, QG on project overview page --- .../src/main/js/store/organizations/duck.ts | 196 +++++++++++++++++++++ 1 file changed, 196 insertions(+) create mode 100644 server/sonar-web/src/main/js/store/organizations/duck.ts (limited to 'server/sonar-web/src/main/js/store/organizations/duck.ts') diff --git a/server/sonar-web/src/main/js/store/organizations/duck.ts b/server/sonar-web/src/main/js/store/organizations/duck.ts new file mode 100644 index 00000000000..8b6c76e3993 --- /dev/null +++ b/server/sonar-web/src/main/js/store/organizations/duck.ts @@ -0,0 +1,196 @@ +/* + * SonarQube + * Copyright (C) 2009-2018 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 { combineReducers } from 'redux'; +import { omit, uniq, without } from 'lodash'; +import { Group, Organization } from '../../app/types'; + +interface ReceiveOrganizationsAction { + type: 'RECEIVE_ORGANIZATIONS'; + organizations: Organization[]; +} + +interface ReceiveMyOrganizationsAction { + type: 'RECEIVE_MY_ORGANIZATIONS'; + organizations: Organization[]; +} + +interface ReceiveOrganizationGroups { + type: 'RECEIVE_ORGANIZATION_GROUPS'; + key: string; + groups: Group[]; +} + +interface CreateOrganizationAction { + type: 'CREATE_ORGANIZATION'; + organization: Organization; +} + +interface UpdateOrganizationAction { + type: 'UPDATE_ORGANIZATION'; + key: string; + changes: {}; +} + +interface DeleteOrganizationAction { + type: 'DELETE_ORGANIZATION'; + key: string; +} + +type Action = + | ReceiveOrganizationsAction + | ReceiveMyOrganizationsAction + | ReceiveOrganizationGroups + | CreateOrganizationAction + | UpdateOrganizationAction + | DeleteOrganizationAction; + +interface ByKeyState { + [key: string]: Organization; +} + +interface GroupsState { + [key: string]: Group[]; +} + +type MyState = string[]; + +interface State { + byKey: ByKeyState; + my: MyState; + groups: GroupsState; +} + +export function receiveOrganizations(organizations: Organization[]): ReceiveOrganizationsAction { + return { + type: 'RECEIVE_ORGANIZATIONS', + organizations + }; +} + +export function receiveMyOrganizations( + organizations: Organization[] +): ReceiveMyOrganizationsAction { + return { + type: 'RECEIVE_MY_ORGANIZATIONS', + organizations + }; +} + +export function receiveOrganizationGroups(key: string, groups: Group[]): ReceiveOrganizationGroups { + return { + type: 'RECEIVE_ORGANIZATION_GROUPS', + key, + groups + }; +} + +export function createOrganization(organization: Organization): CreateOrganizationAction { + return { + type: 'CREATE_ORGANIZATION', + organization + }; +} + +export function updateOrganization(key: string, changes: {}): UpdateOrganizationAction { + return { + type: 'UPDATE_ORGANIZATION', + key, + changes + }; +} + +export function deleteOrganization(key: string): DeleteOrganizationAction { + return { + type: 'DELETE_ORGANIZATION', + key + }; +} + +function onReceiveOrganizations( + state: ByKeyState, + action: ReceiveOrganizationsAction | ReceiveMyOrganizationsAction +): ByKeyState { + const nextState = { ...state }; + action.organizations.forEach(organization => { + nextState[organization.key] = { ...state[organization.key], ...organization }; + }); + return nextState; +} + +function byKey(state: ByKeyState = {}, action: Action) { + switch (action.type) { + case 'RECEIVE_ORGANIZATIONS': + case 'RECEIVE_MY_ORGANIZATIONS': + return onReceiveOrganizations(state, action); + case 'CREATE_ORGANIZATION': + return { ...state, [action.organization.key]: { ...action.organization, isAdmin: true } }; + case 'UPDATE_ORGANIZATION': + return { + ...state, + [action.key]: { + ...state[action.key], + key: action.key, + ...action.changes + } + }; + case 'DELETE_ORGANIZATION': + return omit(state, action.key); + default: + return state; + } +} + +function my(state: MyState = [], action: Action) { + switch (action.type) { + case 'RECEIVE_MY_ORGANIZATIONS': + return uniq([...state, ...action.organizations.map(o => o.key)]); + case 'CREATE_ORGANIZATION': + return uniq([...state, action.organization.key]); + case 'DELETE_ORGANIZATION': + return without(state, action.key); + default: + return state; + } +} + +function groups(state: GroupsState = {}, action: Action) { + if (action.type === 'RECEIVE_ORGANIZATION_GROUPS') { + return { ...state, [action.key]: action.groups }; + } + return state; +} + +export default combineReducers({ byKey, my, groups }); + +export function getOrganizationByKey(state: State, key: string): Organization | undefined { + return state.byKey[key]; +} + +export function getOrganizationGroupsByKey(state: State, key: string): Group[] { + return state.groups[key] || []; +} + +export function getMyOrganizations(state: State): Organization[] { + return state.my.map(key => getOrganizationByKey(state, key) as Organization); +} + +export function areThereCustomOrganizations(state: State): boolean { + return Object.keys(state.byKey).length > 1; +} -- cgit v1.2.3