summaryrefslogtreecommitdiffstats
path: root/sonar-server
diff options
context:
space:
mode:
Diffstat (limited to 'sonar-server')
-rw-r--r--sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileRule.java8
-rw-r--r--sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileSearch.java22
-rw-r--r--sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfiles.java14
-rw-r--r--sonar-server/src/main/java/org/sonar/server/rule/ProfileRuleQuery.java17
-rw-r--r--sonar-server/src/main/java/org/sonar/server/rule/ProfileRules.java12
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/controllers/profiles_controller.rb11
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/helpers/profiles_helper.rb7
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/views/profiles/inheritance.html.erb14
-rw-r--r--sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileSearchTest.java28
-rw-r--r--sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfilesTest.java13
-rw-r--r--sonar-server/src/test/java/org/sonar/server/rule/ProfileRuleQueryTest.java20
-rw-r--r--sonar-server/src/test/java/org/sonar/server/rule/ProfileRulesTest.java9
-rw-r--r--sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule25.json2
-rw-r--r--sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule391.json2
14 files changed, 153 insertions, 26 deletions
diff --git a/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileRule.java b/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileRule.java
index 837c52f1508..d7a6693e8c0 100644
--- a/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileRule.java
+++ b/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileRule.java
@@ -23,7 +23,6 @@ import org.apache.commons.lang.builder.ReflectionToStringBuilder;
import org.elasticsearch.common.collect.Lists;
import org.elasticsearch.common.collect.Maps;
import org.elasticsearch.common.joda.time.format.ISODateTimeFormat;
-import org.sonar.api.rules.ActiveRule;
import org.sonar.check.Cardinality;
import org.sonar.server.rule.ActiveRuleDocument;
import org.sonar.server.rule.RuleDocument;
@@ -36,6 +35,9 @@ import java.util.Map;
public class QProfileRule {
+ public static final String INHERITED = "INHERITED";
+ public static final String OVERRIDES = "OVERRIDES";
+
private final Integer id;
private final Integer parentId;
private final String key;
@@ -200,11 +202,11 @@ public class QProfileRule {
}
public boolean isInherited() {
- return ActiveRule.INHERITED.equals(inheritance);
+ return INHERITED.equals(inheritance);
}
public boolean isOverrides() {
- return ActiveRule.OVERRIDES.equals(inheritance);
+ return OVERRIDES.equals(inheritance);
}
public boolean isTemplate() {
diff --git a/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileSearch.java b/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileSearch.java
index 27d9ad2abc9..60370762bc2 100644
--- a/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileSearch.java
+++ b/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfileSearch.java
@@ -44,6 +44,10 @@ public class QProfileSearch implements ServerComponent {
return toQProfiles(dao.selectAll());
}
+ public List<QProfile> profiles(String language) {
+ return toQProfiles(dao.selectByLanguage(language));
+ }
+
@CheckForNull
public QProfile defaultProfile(String language) {
QualityProfileDto dto = dao.selectDefaultProfile(language, QProfileProjectService.PROPERTY_PREFIX + language);
@@ -57,6 +61,24 @@ public class QProfileSearch implements ServerComponent {
return toQProfiles(dao.selectChildren(profile.name(), profile.language()));
}
+ public List<QProfile> ancestors(QProfile profile) {
+ List<QProfile> ancestors = newArrayList();
+ incrementAncestors(profile, ancestors);
+ return ancestors;
+ }
+
+ private void incrementAncestors(QProfile profile, List<QProfile> ancestors){
+ if (profile.parent() != null) {
+ QualityProfileDto parentDto = dao.selectParent(profile.id());
+ if (parentDto == null) {
+ throw new IllegalStateException("Cannot find parent of profile : "+ profile.id());
+ }
+ QProfile parent = QProfile.from(parentDto);
+ ancestors.add(parent);
+ incrementAncestors(parent, ancestors);
+ }
+ }
+
public int countChildren(QProfile profile) {
return dao.countChildren(profile.name(), profile.language());
}
diff --git a/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfiles.java b/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfiles.java
index 9095be6bcc6..633b353f202 100644
--- a/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfiles.java
+++ b/sonar-server/src/main/java/org/sonar/server/qualityprofile/QProfiles.java
@@ -94,13 +94,17 @@ public class QProfiles implements ServerComponent {
// ACTIVE RULES
// bulk activate all
// bulk deactivate all
- // active rule parameter validation (only Integer types are checked)
+ // active rule parameter validation (only Integer and Boolean types are checked)
public List<QProfile> allProfiles() {
return search.allProfiles();
}
+ public List<QProfile> profilesByLanguage(String language) {
+ return search.profiles(language);
+ }
+
public QProfile profile(int id) {
return QProfile.from(findNotNull(id));
}
@@ -123,6 +127,10 @@ public class QProfiles implements ServerComponent {
return search.children(profile);
}
+ public List<QProfile> ancestors(QProfile profile) {
+ return search.ancestors(profile);
+ }
+
public int countChildren(QProfile profile) {
return search.countChildren(profile);
}
@@ -219,6 +227,10 @@ public class QProfiles implements ServerComponent {
return rules.countProfileRules(ProfileRuleQuery.create(profile.id()));
}
+ public long countOverridingProfileRules(QProfile profile) {
+ return rules.countProfileRules(ProfileRuleQuery.create(profile.id()).setInheritance(QProfileRule.OVERRIDES));
+ }
+
public ProfileRuleChanged activateRule(int profileId, int ruleId, String severity) {
QualityProfileDto qualityProfile = findNotNull(profileId);
RuleDto rule = findRuleNotNull(ruleId);
diff --git a/sonar-server/src/main/java/org/sonar/server/rule/ProfileRuleQuery.java b/sonar-server/src/main/java/org/sonar/server/rule/ProfileRuleQuery.java
index 397db375757..1ec3113831f 100644
--- a/sonar-server/src/main/java/org/sonar/server/rule/ProfileRuleQuery.java
+++ b/sonar-server/src/main/java/org/sonar/server/rule/ProfileRuleQuery.java
@@ -26,6 +26,7 @@ import org.sonar.server.exceptions.BadRequestException;
import org.sonar.server.util.RubyUtils;
import javax.annotation.CheckForNull;
+
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
@@ -44,6 +45,7 @@ public class ProfileRuleQuery {
private static final String PARAM_REPOSITORY_KEYS = "repositoryKeys";
private static final String PARAM_SEVERITIES = "severities";
private static final String PARAM_STATUSES = "statuses";
+ private static final String PARAM_INHERITANCE = "inheritance";
private int profileId;
private String language;
@@ -51,6 +53,7 @@ public class ProfileRuleQuery {
private List<String> repositoryKeys;
private List<String> severities;
private List<String> statuses;
+ private String inheritance;
private ProfileRuleQuery() {
repositoryKeys = Lists.newArrayList();
@@ -80,6 +83,9 @@ public class ProfileRuleQuery {
if (params.get(PARAM_STATUSES) != null) {
result.addStatuses(optionalVarargs(params.get(PARAM_STATUSES)));
}
+ if (params.containsKey(PARAM_INHERITANCE)) {
+ result.setInheritance((String) params.get(PARAM_INHERITANCE));
+ }
if (!errors.isEmpty()) {
throw BadRequestException.of("Incorrect rule search parameters", errors);
@@ -128,6 +134,12 @@ public class ProfileRuleQuery {
return this;
}
+ public ProfileRuleQuery setInheritance(String inheritance) {
+ this.inheritance = inheritance;
+ return this;
+ }
+
+
public int profileId() {
return profileId;
}
@@ -154,6 +166,11 @@ public class ProfileRuleQuery {
return ImmutableList.copyOf(statuses);
}
+ @CheckForNull
+ public String inheritance() {
+ return inheritance;
+ }
+
private static String[] optionalVarargs(Object jRubyArray) {
List<String> items = RubyUtils.toStrings(jRubyArray);
String[] empty = new String[0];
diff --git a/sonar-server/src/main/java/org/sonar/server/rule/ProfileRules.java b/sonar-server/src/main/java/org/sonar/server/rule/ProfileRules.java
index 0402d82a117..64bafb28dc9 100644
--- a/sonar-server/src/main/java/org/sonar/server/rule/ProfileRules.java
+++ b/sonar-server/src/main/java/org/sonar/server/rule/ProfileRules.java
@@ -158,12 +158,20 @@ public class ProfileRules implements ServerExtension {
hasParentFilter(TYPE_RULE, parentRuleFilter(query))
);
addMustTermOrTerms(filter, ActiveRuleDocument.FIELD_SEVERITY, query.severities());
+ String inheritance = query.inheritance();
+ if (inheritance != null) {
+ addMustTermOrTerms(filter, ActiveRuleDocument.FIELD_INHERITANCE, newArrayList(inheritance));
+ }
return filter;
}
public long countProfileRules(ProfileRuleQuery query) {
- return index.executeCount(index.client().prepareCount(INDEX_RULES).setTypes(TYPE_ACTIVE_RULE)
- .setQuery(QueryBuilders.filteredQuery(QueryBuilders.matchAllQuery(), activeRuleFilter(query))));
+ return index.executeCount(
+ index.client()
+ .prepareCount(INDEX_RULES)
+ .setTypes(TYPE_ACTIVE_RULE)
+ .setQuery(QueryBuilders.filteredQuery(QueryBuilders.matchAllQuery(), activeRuleFilter(query)))
+ );
}
public long countInactiveProfileRules(ProfileRuleQuery query) {
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/controllers/profiles_controller.rb b/sonar-server/src/main/webapp/WEB-INF/app/controllers/profiles_controller.rb
index b4bce6d2c83..cab5e63475a 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/controllers/profiles_controller.rb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/controllers/profiles_controller.rb
@@ -180,10 +180,15 @@ class ProfilesController < ApplicationController
# GET /profiles/inheritance?id=<profile id>
def inheritance
require_parameters 'id'
- @profile = Profile.find(params[:id])
- profiles=Profile.all(:conditions => ['language=? and id<>? and (parent_name is null or parent_name<>?)', @profile.language, @profile.id, @profile.name], :order => 'name')
- @select_parent = [[message('none'), nil]] + profiles.collect { |profile| [profile.name, profile.name] }
+ call_backend do
+ @profile = Internal.quality_profiles.profile(params[:id].to_i)
+ @ancestors = Internal.quality_profiles.ancestors(@profile).to_a
+ @children = Internal.quality_profiles.children(@profile).to_a
+ profiles = Internal.quality_profiles.profilesByLanguage(@profile.language()).to_a.reject{|p| p.id == @profile.id() || p.parent() == @profile.name()}
+ profiles = Api::Utils.insensitive_sort(profiles) { |p| p.name()}
+ @select_parent = [[message('none'), nil]] + profiles.collect { |profile| [profile.name(), profile.name()] }
+ end
set_profile_breadcrumbs
end
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/helpers/profiles_helper.rb b/sonar-server/src/main/webapp/WEB-INF/app/helpers/profiles_helper.rb
index eb019e18398..71df4bfae66 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/helpers/profiles_helper.rb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/helpers/profiles_helper.rb
@@ -23,10 +23,11 @@ module ProfilesHelper
controller.java_facade.getLanguages()
end
- def label_for_rules_count(profile)
- label="#{profile.count_active_rules} #{message('rules').downcase}"
+ def label_for_rules_count(qProfile)
+ profile_rules_count = profile_rules_count(qProfile)
+ label = "#{profile_rules_count} #{message('rules').downcase}"
- count_overriding=profile.count_overriding_rules
+ count_overriding = Internal.quality_profiles.countOverridingProfileRules(qProfile).to_i
if count_overriding>0
label += message('quality_profiles.including_x_overriding.suffix', :params => count_overriding)
label += image_tag('overrides.png')
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/views/profiles/inheritance.html.erb b/sonar-server/src/main/webapp/WEB-INF/app/views/profiles/inheritance.html.erb
index 06cb2996288..806111b245c 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/views/profiles/inheritance.html.erb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/views/profiles/inheritance.html.erb
@@ -7,19 +7,19 @@
<tr>
<td align="center" valign="top">
<div>
- <% @profile.ancestors.reverse.each do |parent| %>
- <a href="<%= url_for :action => 'inheritance', :id => parent.id -%>"><%= parent.name -%></a>
+ <% @ancestors.reverse.each do |parent| %>
+ <a href="<%= url_for :action => 'inheritance', :id => parent.id() -%>"><%= parent.name() -%></a>
<span class="note">(<%= label_for_rules_count(parent) -%>)</span><br/>
<%= image_tag 'blue-up.png' -%><br/>
<% end %>
<b><%= @profile.name -%></b> <span class="note">(<%= label_for_rules_count(@profile) -%>)</span><br/>
- <% if @profile.children.size>0 %>
+ <% if @children.size>0 %>
<%= image_tag 'blue-up.png' -%><br/>
- <% @profile.children.each_with_index do |child,index| %>
+ <% @children.each_with_index do |child,index| %>
<%= ', ' if index>0 -%>
- <a href="<%= url_for :action => 'inheritance', :id => child.id -%>"><%= child.name -%></a>
+ <a href="<%= url_for :action => 'inheritance', :id => child.id() -%>"><%= child.name() -%></a>
<span class="note">(<%= label_for_rules_count(child) -%>)</span>
<% end %>
<br/><%= image_tag 'blue-up.png' -%><br/>
@@ -33,8 +33,8 @@
<h3><%= message('quality_profiles.set_parent') -%>:</h3>
<p><%= message('quality_profiles.inherit_rules_from_profile') -%>:</p>
<% form_tag({:action => 'change_parent'}, {:method => 'post'}) do %>
- <%= hidden_field_tag "id", @profile.id %>
- <%= select_tag "parent_name", options_for_select(@select_parent, @profile.parent_name) %>
+ <%= hidden_field_tag "id", @profile.id() %>
+ <%= select_tag "parent_name", options_for_select(@select_parent, @profile.parent()) %>
<%= submit_tag message('change_verb'), :id => 'submit_parent'%>
<% end %>
</div>
diff --git a/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileSearchTest.java b/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileSearchTest.java
index 1dcb06b73ab..966a4b65210 100644
--- a/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileSearchTest.java
+++ b/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfileSearchTest.java
@@ -67,6 +67,12 @@ public class QProfileSearchTest {
}
@Test
+ public void search_profiles_by_language() throws Exception {
+ search.profiles("java");
+ verify(dao).selectByLanguage("java");
+ }
+
+ @Test
public void search_children_profiles() throws Exception {
search.children(new QProfile().setName("Sonar Way").setLanguage("java"));
verify(dao).selectChildren("Sonar Way", "java");
@@ -94,5 +100,25 @@ public class QProfileSearchTest {
assertThat(search.defaultProfile("java")).isNull();
}
-}
+ @Test
+ public void search_ancestors() throws Exception {
+ when(dao.selectParent(3)).thenReturn(new QualityProfileDto().setId(2).setName("Child").setLanguage("java").setParent("Parent"));
+ when(dao.selectParent(2)).thenReturn(new QualityProfileDto().setId(1).setName("Parent").setLanguage("java"));
+ when(dao.selectParent(1)).thenReturn(null);
+ List<QProfile> result = search.ancestors(new QProfile().setId(3).setName("Grandchild").setLanguage("java").setParent("Child"));
+ assertThat(result).hasSize(2);
+ }
+
+ @Test
+ public void fail_to_get_ancestors_if_parent_cannot_be_found() throws Exception {
+ when(dao.selectParent(3)).thenReturn(null);
+
+ try {
+ search.ancestors(new QProfile().setId(3).setName("Grandchild").setLanguage("java").setParent("Child"));
+ } catch (Exception e) {
+ assertThat(e).isInstanceOf(IllegalStateException.class);
+ }
+ }
+
+}
diff --git a/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfilesTest.java b/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfilesTest.java
index abfd25fe1df..6175c1e5a14 100644
--- a/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfilesTest.java
+++ b/sonar-server/src/test/java/org/sonar/server/qualityprofile/QProfilesTest.java
@@ -106,6 +106,12 @@ public class QProfilesTest {
}
@Test
+ public void search_profiles_by_language() throws Exception {
+ qProfiles.profilesByLanguage("java");
+ verify(search).profiles("java");
+ }
+
+ @Test
public void search_default_profile_by_language() throws Exception {
qProfiles.defaultProfile("java");
verify(search).defaultProfile("java");
@@ -132,6 +138,13 @@ public class QProfilesTest {
}
@Test
+ public void search_ancestors() throws Exception {
+ QProfile profile = new QProfile();
+ qProfiles.ancestors(profile);
+ verify(search).ancestors(profile);
+ }
+
+ @Test
public void count_children() throws Exception {
QProfile profile = new QProfile();
qProfiles.countChildren(profile);
diff --git a/sonar-server/src/test/java/org/sonar/server/rule/ProfileRuleQueryTest.java b/sonar-server/src/test/java/org/sonar/server/rule/ProfileRuleQueryTest.java
index 252a3f909c6..8c8121c8a71 100644
--- a/sonar-server/src/test/java/org/sonar/server/rule/ProfileRuleQueryTest.java
+++ b/sonar-server/src/test/java/org/sonar/server/rule/ProfileRuleQueryTest.java
@@ -32,14 +32,14 @@ import static org.fest.assertions.Assertions.assertThat;
public class ProfileRuleQueryTest {
@Test
- public void should_create_basic_query() {
+ public void create_basic_query() {
final int profileId = 42;
ProfileRuleQuery query = ProfileRuleQuery.create(profileId);
assertThat(query.profileId()).isEqualTo(profileId);
}
@Test
- public void should_parse_nominal_request() {
+ public void parse_nominal_request() {
final int profileId = 42;
Map<String, Object> params = ImmutableMap.of("profileId", (Object) Integer.toString(profileId));
ProfileRuleQuery query = ProfileRuleQuery.parse(params);
@@ -47,7 +47,19 @@ public class ProfileRuleQueryTest {
}
@Test
- public void should_fail_on_missing_profileId() {
+ public void parse_with_inheritance() {
+ final int profileId = 42;
+ Map<String, Object> params = ImmutableMap.of(
+ "profileId", (Object) Integer.toString(profileId),
+ "inheritance", "OVERRIDES"
+ );
+ ProfileRuleQuery query = ProfileRuleQuery.parse(params);
+ assertThat(query.profileId()).isEqualTo(profileId);
+ assertThat(query.inheritance()).isEqualTo("OVERRIDES");
+ }
+
+ @Test
+ public void fail_on_missing_profileId() {
Map<String, Object> params = ImmutableMap.of();
try {
ProfileRuleQuery.parse(params);
@@ -58,7 +70,7 @@ public class ProfileRuleQueryTest {
}
@Test
- public void should_fail_on_incorrect_profileId() {
+ public void fail_on_incorrect_profileId() {
Map<String, Object> params = ImmutableMap.of("profileId", (Object) "not an integer");
try {
ProfileRuleQuery.parse(params);
diff --git a/sonar-server/src/test/java/org/sonar/server/rule/ProfileRulesTest.java b/sonar-server/src/test/java/org/sonar/server/rule/ProfileRulesTest.java
index 8f24536f6da..2d2d9caca77 100644
--- a/sonar-server/src/test/java/org/sonar/server/rule/ProfileRulesTest.java
+++ b/sonar-server/src/test/java/org/sonar/server/rule/ProfileRulesTest.java
@@ -116,6 +116,15 @@ public class ProfileRulesTest {
}
@Test
+ public void find_profile_rules_with_inheritance() {
+ Paging paging = Paging.create(10, 1);
+
+ assertThat(profileRules.searchProfileRules(ProfileRuleQuery.create(1).setInheritance(null), paging).rules()).hasSize(3);
+ assertThat(profileRules.searchProfileRules(ProfileRuleQuery.create(1).setInheritance(QProfileRule.INHERITED), paging).rules()).hasSize(1);
+ assertThat(profileRules.searchProfileRules(ProfileRuleQuery.create(1).setInheritance(QProfileRule.OVERRIDES), paging).rules()).hasSize(1);
+ }
+
+ @Test
@Ignore("bug in E/S : fail to do a scroll when filter contain has_parent -> return good total_hits but hits is empty")
public void find_profile_rule_ids() {
// All rules for profile 1
diff --git a/sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule25.json b/sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule25.json
index 717f25ab7df..312be662990 100644
--- a/sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule25.json
+++ b/sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule25.json
@@ -1 +1 @@
-{"id": 25, "severity": "MINOR", "profileId": 1, "inheritance": null}
+{"id": 25, "severity": "MINOR", "profileId": 1, "inheritance": "OVERRIDES"}
diff --git a/sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule391.json b/sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule391.json
index 7c279b9abd7..14b8c18903b 100644
--- a/sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule391.json
+++ b/sonar-server/src/test/resources/org/sonar/server/rule/ProfileRulesTest/should_find_active_rules/active_rule391.json
@@ -1 +1 @@
-{"id": 391, "severity": "MAJOR", "profileId": 1, "inheritance": null} \ No newline at end of file
+{"id": 391, "severity": "MAJOR", "profileId": 1, "inheritance": "INHERITED"}