diff options
author | Stas Vilchik <stas.vilchik@sonarsource.com> | 2017-09-26 14:41:01 +0200 |
---|---|---|
committer | Stas Vilchik <stas.vilchik@sonarsource.com> | 2017-10-02 17:18:15 +0200 |
commit | 6a3ae4ac87b1e12a9bcd496c57bb6e2918cc345e (patch) | |
tree | 38f63a05fdf9797438a600544a3fdacd789c7615 /server/sonar-web/src/main/js | |
parent | 77d3e365acbfd9095efe4458c643a9e8ccdada13 (diff) | |
download | sonarqube-6a3ae4ac87b1e12a9bcd496c57bb6e2918cc345e.tar.gz sonarqube-6a3ae4ac87b1e12a9bcd496c57bb6e2918cc345e.zip |
SONAR-1330 Allow rule changes only for quality profile admins
Diffstat (limited to 'server/sonar-web/src/main/js')
15 files changed, 62 insertions, 62 deletions
diff --git a/server/sonar-web/src/main/js/api/rules.ts b/server/sonar-web/src/main/js/api/rules.ts index 20a0b26f5a2..ded000324d1 100644 --- a/server/sonar-web/src/main/js/api/rules.ts +++ b/server/sonar-web/src/main/js/api/rules.ts @@ -18,6 +18,15 @@ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ import { getJSON, RequestData } from '../helpers/request'; +import throwGlobalError from '../app/utils/throwGlobalError'; + +export interface GetRulesAppResponse { + respositories: Array<{ key: string; language: string; name: string }>; +} + +export function getRulesApp(): Promise<GetRulesAppResponse> { + return getJSON('/api/rules/app').catch(throwGlobalError); +} export function searchRules(data: RequestData) { return getJSON('/api/rules/search', data); diff --git a/server/sonar-web/src/main/js/apps/coding-rules/bulk-change-modal-view.js b/server/sonar-web/src/main/js/apps/coding-rules/bulk-change-modal-view.js index 07277f983f6..953c9c7900e 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/bulk-change-modal-view.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/bulk-change-modal-view.js @@ -107,9 +107,11 @@ export default ModalFormView.extend({ const languages = queryLanguages && queryLanguages.length > 0 ? queryLanguages.split(',') : []; let profiles = this.options.app.qualityProfiles; if (languages.length > 0) { - profiles = profiles.filter(profile => languages.indexOf(profile.lang) !== -1); + profiles = profiles.filter(profile => languages.indexOf(profile.language) !== -1); } - return profiles.filter(profile => !profile.isBuiltIn); + return profiles + .filter(profile => profile.actions && profile.actions.edit) + .filter(profile => !profile.isBuiltIn); }, serializeData() { diff --git a/server/sonar-web/src/main/js/apps/coding-rules/bulk-change-popup-view.js b/server/sonar-web/src/main/js/apps/coding-rules/bulk-change-popup-view.js index 50c6c6d4676..d038bf5c9df 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/bulk-change-popup-view.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/bulk-change-popup-view.js @@ -44,12 +44,14 @@ export default PopupView.extend({ const profileKey = query.qprofile; const profile = this.options.app.qualityProfiles.find(p => p.key === profileKey); const activation = '' + query.activation; + const canChangeProfile = + profile != null && !profile.isBuiltIn && profile.actions && profile.actions.edit; return { qualityProfile: profileKey, qualityProfileName: profile != null ? profile.name : null, - allowActivateOnProfile: profile != null && activation === 'false' && !profile.isBuiltIn, - allowDeactivateOnProfile: profileKey != null && activation === 'true' + allowActivateOnProfile: canChangeProfile && activation === 'false', + allowDeactivateOnProfile: canChangeProfile && activation === 'true' }; } }); diff --git a/server/sonar-web/src/main/js/apps/coding-rules/facets/quality-profile-facet.js b/server/sonar-web/src/main/js/apps/coding-rules/facets/quality-profile-facet.js index 3efdac89cfb..d0329e4caaa 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/facets/quality-profile-facet.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/facets/quality-profile-facet.js @@ -50,9 +50,9 @@ export default BaseFacet.extend({ const languages = languagesQuery != null ? languagesQuery.split(',') : []; const lang = languages.length === 1 ? languages[0] : null; const values = this.options.app.qualityProfiles - .filter(profile => (lang != null ? profile.lang === lang : true)) + .filter(profile => (lang != null ? profile.language === lang : true)) .map(profile => ({ - extra: that.options.app.languages[profile.lang], + extra: that.options.app.languages[profile.language], isBuiltIn: profile.isBuiltIn, label: profile.name, val: profile.key diff --git a/server/sonar-web/src/main/js/apps/coding-rules/init.js b/server/sonar-web/src/main/js/apps/coding-rules/init.js index 36d7f608fda..ed38639c9e0 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/init.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/init.js @@ -32,6 +32,8 @@ import Router from '../../components/navigator/router'; import WorkspaceListView from './workspace-list-view'; import WorkspaceHeaderView from './workspace-header-view'; import FacetsView from './facets-view'; +import { searchQualityProfiles } from '../../api/quality-profiles'; +import { getRulesApp } from '../../api/rules'; import { areThereCustomOrganizations } from '../../store/organizations/utils'; const App = new Marionette.Application(); @@ -45,20 +47,16 @@ App.on('start', function( ) { App.organization = options.organization; const data = options.organization ? { organization: options.organization } : {}; - $.get(window.baseUrl + '/api/rules/app', data) - .done(r => { + Promise.all([getRulesApp(data), searchQualityProfiles(data)]) + .then(([appResponse, profilesResponse]) => { App.customRules = !areThereCustomOrganizations(); - App.canWrite = r.canWrite; + App.canWrite = appResponse.canWrite; App.organization = options.organization; - App.qualityProfiles = sortBy(r.qualityprofiles, ['name', 'lang']); - App.languages = { ...r.languages, none: 'None' }; - App.qualityProfiles.forEach(profile => { - profile.language = App.languages[profile.lang]; - }); - App.repositories = r.repositories; - App.statuses = r.statuses; - }) - .done(() => { + App.qualityProfiles = sortBy(profilesResponse.profiles, ['name', 'lang']); + App.languages = { ...appResponse.languages, none: 'None' }; + App.repositories = appResponse.repositories; + App.statuses = appResponse.statuses; + this.layout = new Layout({ el: options.el }); this.layout.render(); $('#footer').addClass('page-footer-with-sidebar'); @@ -109,6 +107,9 @@ App.on('start', function( app: this }); Backbone.history.start(); + }) + .catch(() => { + // do nothing in case of WS error }); }); diff --git a/server/sonar-web/src/main/js/apps/coding-rules/rule/profile-activation-view.js b/server/sonar-web/src/main/js/apps/coding-rules/rule/profile-activation-view.js index f55361b621f..c4c34051150 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/rule/profile-activation-view.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/rule/profile-activation-view.js @@ -132,8 +132,10 @@ export default ModalForm.extend({ const inactiveProfiles = this.options.app.qualityProfiles.filter( profile => !activeQualityProfiles.findWhere({ key: profile.key }) ); + // choose QP which a user can administrate, which are the same language and which are not built-in return inactiveProfiles - .filter(profile => profile.lang === lang) + .filter(profile => profile.actions && profile.actions.edit) + .filter(profile => profile.language === lang) .filter(profile => !profile.isBuiltIn); }, diff --git a/server/sonar-web/src/main/js/apps/coding-rules/rule/rule-profile-view.js b/server/sonar-web/src/main/js/apps/coding-rules/rule/rule-profile-view.js index 916af2e5f03..3ddbfd50961 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/rule/rule-profile-view.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/rule/rule-profile-view.js @@ -164,12 +164,13 @@ export default Marionette.ItemView.extend({ return { ...Marionette.ItemView.prototype.serializeData.apply(this, arguments), parent, + actions: this.model.get('actions') || {}, canWrite: this.options.app.canWrite, parameters: this.enhanceParameters(parent), templateKey: this.options.rule.get('templateKey'), isTemplate: this.options.rule.get('isTemplate'), - profilePath: this.getProfilePath(this.model.get('lang'), this.model.get('name')), - parentProfilePath: parent && this.getProfilePath(parent.lang, parent.name) + profilePath: this.getProfilePath(this.model.get('language'), this.model.get('name')), + parentProfilePath: parent && this.getProfilePath(parent.language, parent.name) }; } }); diff --git a/server/sonar-web/src/main/js/apps/coding-rules/rule/rule-profiles-view.js b/server/sonar-web/src/main/js/apps/coding-rules/rule/rule-profiles-view.js index dadabe8ea18..81781afab8b 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/rule/rule-profiles-view.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/rule/rule-profiles-view.js @@ -85,9 +85,14 @@ export default Marionette.CompositeView.extend({ }, serializeData() { + // show "Activate" button only if user has at least one QP which he administates + const canActivate = this.options.app.qualityProfiles.some( + profile => profile.actions && profile.actions.edit + ); + return { ...Marionette.ItemView.prototype.serializeData.apply(this, arguments), - canWrite: this.options.app.canWrite + canActivate }; } }); diff --git a/server/sonar-web/src/main/js/apps/coding-rules/templates/_coding-rules-workspace-list-item-activation.hbs b/server/sonar-web/src/main/js/apps/coding-rules/templates/_coding-rules-workspace-list-item-activation.hbs deleted file mode 100644 index 84f89974491..00000000000 --- a/server/sonar-web/src/main/js/apps/coding-rules/templates/_coding-rules-workspace-list-item-activation.hbs +++ /dev/null @@ -1,29 +0,0 @@ -{{#each activeProfiles}} - {{severityIcon severity}} - - {{#eq inherit 'OVERRIDES'}} - <i class="icon-inheritance" title="{{tp 'coding_rules.overrides' name parent.name}}"></i> - {{/eq}} - {{#eq inherit 'INHERITED'}} - <i class="icon-inheritance" title="{{tp 'coding_rules.inherits' name parent.name}}"></i> - {{/eq}} - - {{#if ../canWrite}} - <div class="button-group"> - {{#unless isTemplate}} - <button class="coding-rules-detail-quality-profile-change">{{t 'change_verb'}}</button> - {{/unless}} - {{#if parent}} - {{#eq inherit 'OVERRIDES'}} - <button class="coding-rules-detail-quality-profile-revert button-red"> - {{t 'coding_rules.revert_to_parent_definition'}} - </button> - {{/eq}} - {{else}} - <button class="coding-rules-detail-quality-profile-deactivate button-red"> - {{t 'coding_rules.deactivate'}} - </button> - {{/if}} - </div> - {{/if}} -{{/each}} diff --git a/server/sonar-web/src/main/js/apps/coding-rules/templates/coding-rules-workspace-header.hbs b/server/sonar-web/src/main/js/apps/coding-rules/templates/coding-rules-workspace-header.hbs index 53dcba3b520..97623f1e77a 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/templates/coding-rules-workspace-header.hbs +++ b/server/sonar-web/src/main/js/apps/coding-rules/templates/coding-rules-workspace-header.hbs @@ -29,7 +29,7 @@ <div class="search-navigator-header-buttons button-group"> <button class="js-reload">{{t 'reload'}}</button> <button class="js-new-search" id="coding-rules-new-search">{{t 'coding_rules.new_search'}}</button> - {{#if canWrite}} + {{#if canBulkChange}} <button class="js-bulk-change">{{t 'bulk_change'}}</button> {{/if}} </div> diff --git a/server/sonar-web/src/main/js/apps/coding-rules/templates/coding-rules-workspace-list-item.hbs b/server/sonar-web/src/main/js/apps/coding-rules/templates/coding-rules-workspace-list-item.hbs index 23324db3e9e..139d46cdcb6 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/templates/coding-rules-workspace-list-item.hbs +++ b/server/sonar-web/src/main/js/apps/coding-rules/templates/coding-rules-workspace-list-item.hbs @@ -46,16 +46,15 @@ </td> {{#any activation selectedProfile}} - {{#if canWrite}} + {{#if canEditQualityProfile}} {{#unless isSelectedProfileBuiltIn}} <td class="coding-rule-table-meta-cell coding-rule-activation-actions"> <div class="button-group"> {{#if activation}} - {{#eq activation.inherit 'NONE'}} - <button class="coding-rules-detail-quality-profile-deactivate button-red"> - {{t 'coding_rules.deactivate'}} - </button> - {{/eq}} + <button class="coding-rules-detail-quality-profile-deactivate button-red" + {{#notEq activation.inherit 'NONE'}}disabled title="{{t 'coding_rules.can_not_deactivate'}}"{{/notEq}}> + {{t 'coding_rules.deactivate'}} + </button> {{else}} {{#unless isTemplate}} <button class="coding-rules-detail-quality-profile-activate">{{t 'coding_rules.activate'}}</button> diff --git a/server/sonar-web/src/main/js/apps/coding-rules/templates/rule/coding-rules-rule-profile.hbs b/server/sonar-web/src/main/js/apps/coding-rules/templates/rule/coding-rules-rule-profile.hbs index 00a1c805411..e65ab663892 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/templates/rule/coding-rules-rule-profile.hbs +++ b/server/sonar-web/src/main/js/apps/coding-rules/templates/rule/coding-rules-rule-profile.hbs @@ -52,7 +52,7 @@ </td> {{/unless}} - {{#if canWrite}} + {{#if actions.edit}} {{#unless isBuiltIn}} <td class="coding-rules-detail-quality-profile-actions"> <div class="button-group"> diff --git a/server/sonar-web/src/main/js/apps/coding-rules/templates/rule/coding-rules-rule-profiles.hbs b/server/sonar-web/src/main/js/apps/coding-rules/templates/rule/coding-rules-rule-profiles.hbs index 2a28c69d62e..efd296d2074 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/templates/rule/coding-rules-rule-profiles.hbs +++ b/server/sonar-web/src/main/js/apps/coding-rules/templates/rule/coding-rules-rule-profiles.hbs @@ -3,7 +3,7 @@ <h3 class="coding-rules-detail-title">{{t 'coding_rules.quality_profiles'}}</h3> - {{#if canWrite}} + {{#if canActivate}} {{#unless isTemplate}} <div class="button-group coding-rules-detail-quality-profiles-activation"> <button id="coding-rules-quality-profile-activate">{{t 'coding_rules.activate'}}</button> diff --git a/server/sonar-web/src/main/js/apps/coding-rules/workspace-header-view.js b/server/sonar-web/src/main/js/apps/coding-rules/workspace-header-view.js index a5ac0165f28..708164d62ae 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/workspace-header-view.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/workspace-header-view.js @@ -58,9 +58,14 @@ export default WorkspaceHeaderView.extend({ }, serializeData() { + // show "Bulk Change" button only if user has at least one QP which he administates + const canBulkChange = this.options.app.qualityProfiles.some( + profile => profile.actions && profile.actions.edit + ); + return { ...WorkspaceHeaderView.prototype.serializeData.apply(this, arguments), - canWrite: this.options.app.canWrite + canBulkChange }; } }); diff --git a/server/sonar-web/src/main/js/apps/coding-rules/workspace-list-item-view.js b/server/sonar-web/src/main/js/apps/coding-rules/workspace-list-item-view.js index b5d0fafe2ee..6235c15c7ab 100644 --- a/server/sonar-web/src/main/js/apps/coding-rules/workspace-list-item-view.js +++ b/server/sonar-web/src/main/js/apps/coding-rules/workspace-list-item-view.js @@ -118,10 +118,13 @@ export default WorkspaceListItemView.extend(RuleFilterMixin).extend({ this.options.app.qualityProfiles.find(profile => profile.key === selectedProfileKey); const isSelectedProfileBuiltIn = selectedProfile != null && selectedProfile.isBuiltIn; + const canEditQualityProfile = + selectedProfile && selectedProfile.actions && selectedProfile.actions.edit; + return { ...WorkspaceListItemView.prototype.serializeData.apply(this, arguments), + canEditQualityProfile, tags: union(this.model.get('sysTags'), this.model.get('tags')), - canWrite: this.options.app.canWrite, selectedProfile: selectedProfileKey, isSelectedProfileBuiltIn }; |