diff options
author | Jenkins CI <ci@sonarsource.com> | 2014-07-10 08:01:48 +0200 |
---|---|---|
committer | Jenkins CI <ci@sonarsource.com> | 2014-07-10 08:01:48 +0200 |
commit | a45436dd83948364b2c4fe9234207b304847992c (patch) | |
tree | 2e5adfdabfecb11b4f9725fc914272dc5125e1b6 /sonar-server/src/main/java | |
parent | 4d5d1f5be3d1ace0199d39a086165e1086600a26 (diff) | |
parent | f1fa4be793dcadbd6628150d24e441c2095e0a54 (diff) | |
download | sonarqube-a45436dd83948364b2c4fe9234207b304847992c.tar.gz sonarqube-a45436dd83948364b2c4fe9234207b304847992c.zip |
Merge commit 'f1fa4be793dcadbd6628150d24e441c2095e0a54' into HEAD
Diffstat (limited to 'sonar-server/src/main/java')
9 files changed, 243 insertions, 183 deletions
diff --git a/sonar-server/src/main/java/org/sonar/server/activity/ws/ActivitiesWebService.java b/sonar-server/src/main/java/org/sonar/server/activity/ws/ActivitiesWebService.java index e6aa30ce395..925a0456d2b 100644 --- a/sonar-server/src/main/java/org/sonar/server/activity/ws/ActivitiesWebService.java +++ b/sonar-server/src/main/java/org/sonar/server/activity/ws/ActivitiesWebService.java @@ -35,7 +35,7 @@ public class ActivitiesWebService implements WebService { public void define(Context context) { NewController controller = context .createController(API_ENDPOINT) - .setDescription("Logs search and views"); + .setDescription("Tracking of activities"); search.define(controller); controller.done(); diff --git a/sonar-server/src/main/java/org/sonar/server/activity/ws/ActivityMapping.java b/sonar-server/src/main/java/org/sonar/server/activity/ws/ActivityMapping.java index ae2deb2f453..e2910b6c4ea 100644 --- a/sonar-server/src/main/java/org/sonar/server/activity/ws/ActivityMapping.java +++ b/sonar-server/src/main/java/org/sonar/server/activity/ws/ActivityMapping.java @@ -22,40 +22,39 @@ package org.sonar.server.activity.ws; import org.sonar.api.resources.Languages; import org.sonar.api.utils.text.JsonWriter; import org.sonar.core.activity.Activity; +import org.sonar.server.activity.index.ActivityDoc; import org.sonar.server.activity.index.ActivityNormalizer; import org.sonar.server.search.ws.BaseMapping; +import org.sonar.server.search.ws.SearchOptions; import org.sonar.server.text.MacroInterpreter; import java.util.Map; /** - * Conversion between Log and WS JSON response + * Conversion between {@link org.sonar.server.activity.index.ActivityDoc} and WS JSON response */ -public class ActivityMapping extends BaseMapping { - +public class ActivityMapping extends BaseMapping<ActivityDoc, Object> { public ActivityMapping(Languages languages, MacroInterpreter macroInterpreter) { - super(); - addIndexStringField("type", ActivityNormalizer.LogFields.TYPE.field()); - addIndexStringField("action", ActivityNormalizer.LogFields.ACTION.field()); - addIndexDatetimeField("createdAt", ActivityNormalizer.LogFields.CREATED_AT.field()); - addIndexStringField("login", ActivityNormalizer.LogFields.LOGIN.field()); - addIndexStringField("message", ActivityNormalizer.LogFields.MESSAGE.field()); - addField("details", new DetailField()); + map("type", ActivityNormalizer.LogFields.TYPE.field()); + map("action", ActivityNormalizer.LogFields.ACTION.field()); + mapDateTime("createdAt", ActivityNormalizer.LogFields.CREATED_AT.field()); + map("login", ActivityNormalizer.LogFields.LOGIN.field()); + map("message", ActivityNormalizer.LogFields.MESSAGE.field()); + map("details", new IndexMapper<ActivityDoc, Object>(ActivityNormalizer.LogFields.DETAILS.field()) { + @Override + public void write(JsonWriter json, ActivityDoc activity, Object context) { + json.name("details").beginObject(); + for (Map.Entry<String, String> detail : activity.details().entrySet()) { + json.prop(detail.getKey(), detail.getValue()); + } + json.endObject(); + } + }); } - private static class DetailField extends IndexField<Activity> { - DetailField() { - super(ActivityNormalizer.LogFields.DETAILS.field()); - } - - @Override - public void write(JsonWriter json, Activity activity) { - json.name("details").beginObject(); - for (Map.Entry<String, String> detail : activity.details().entrySet()) { - json.prop(detail.getKey(), detail.getValue()); - } - json.endObject(); - } + public void write(Activity activity, JsonWriter writer, SearchOptions options) { + doWrite((ActivityDoc)activity, null, writer, options); } + } diff --git a/sonar-server/src/main/java/org/sonar/server/activity/ws/SearchAction.java b/sonar-server/src/main/java/org/sonar/server/activity/ws/SearchAction.java index a6c40219281..d221caad86e 100644 --- a/sonar-server/src/main/java/org/sonar/server/activity/ws/SearchAction.java +++ b/sonar-server/src/main/java/org/sonar/server/activity/ws/SearchAction.java @@ -27,7 +27,6 @@ import org.sonar.api.server.ws.WebService; import org.sonar.api.utils.text.JsonWriter; import org.sonar.core.activity.Activity; import org.sonar.server.activity.ActivityService; -import org.sonar.server.activity.index.ActivityDoc; import org.sonar.server.activity.index.ActivityQuery; import org.sonar.server.search.QueryOptions; import org.sonar.server.search.Result; @@ -53,14 +52,14 @@ public class SearchAction implements RequestHandler { void define(WebService.NewController controller) { WebService.NewAction action = controller .createAction(SEARCH_ACTION) - .setDescription("Search for a logs") + .setDescription("Search for activities") .setSince("4.4") .setInternal(true) .setHandler(this); // Other parameters action.createParam(PARAM_TYPE) - .setDescription("Select types of log to search") + .setDescription("Types of activities to search") .setPossibleValues(Activity.Type.values()) .setDefaultValue(StringUtils.join(Activity.Type.values(), ",")); @@ -87,7 +86,7 @@ public class SearchAction implements RequestHandler { private void writeLogs(Result<Activity> result, JsonWriter json, SearchOptions options) { json.name("logs").beginArray(); for (Activity log : result.getHits()) { - mapping.write((ActivityDoc) log, json, options); + mapping.write(log, json, options); } json.endArray(); } diff --git a/sonar-server/src/main/java/org/sonar/server/rule/ws/CreateAction.java b/sonar-server/src/main/java/org/sonar/server/rule/ws/CreateAction.java index 58a2cf69cf0..fde8fd299c5 100644 --- a/sonar-server/src/main/java/org/sonar/server/rule/ws/CreateAction.java +++ b/sonar-server/src/main/java/org/sonar/server/rule/ws/CreateAction.java @@ -35,7 +35,6 @@ import org.sonar.server.rule.NewRule; import org.sonar.server.rule.ReactivationException; import org.sonar.server.rule.Rule; import org.sonar.server.rule.RuleService; -import org.sonar.server.search.BaseDoc; import java.io.OutputStreamWriter; @@ -158,7 +157,7 @@ public class CreateAction implements RequestHandler { private void writeResponse(Response response, RuleKey ruleKey) { Rule rule = service.getNonNullByKey(ruleKey); JsonWriter json = response.newJsonWriter().beginObject().name("rule"); - mapping.write((BaseDoc) rule, json); + mapping.write(rule, json, null /* TODO replace by SearchOptions immutable constant */); json.endObject().close(); } @@ -169,7 +168,7 @@ public class CreateAction implements RequestHandler { stream.setStatus(409); stream.setMediaType(MimeTypes.JSON); JsonWriter json = JsonWriter.of(new OutputStreamWriter(stream.output())).beginObject().name("rule"); - mapping.write((BaseDoc) rule, json); + mapping.write(rule, json, null /* TODO replace by SearchOptions immutable constant */); json.endObject().close(); } } diff --git a/sonar-server/src/main/java/org/sonar/server/rule/ws/RuleMapping.java b/sonar-server/src/main/java/org/sonar/server/rule/ws/RuleMapping.java index 3f462bd4cba..ecb1f2233a3 100644 --- a/sonar-server/src/main/java/org/sonar/server/rule/ws/RuleMapping.java +++ b/sonar-server/src/main/java/org/sonar/server/rule/ws/RuleMapping.java @@ -19,157 +19,213 @@ */ package org.sonar.server.rule.ws; +import com.google.common.collect.Maps; import org.apache.commons.lang.StringEscapeUtils; import org.sonar.api.resources.Language; import org.sonar.api.resources.Languages; +import org.sonar.api.server.debt.DebtCharacteristic; +import org.sonar.api.server.debt.DebtModel; import org.sonar.api.utils.text.JsonWriter; import org.sonar.markdown.Markdown; import org.sonar.server.rule.Rule; import org.sonar.server.rule.RuleParam; +import org.sonar.server.rule.index.RuleDoc; import org.sonar.server.rule.index.RuleNormalizer; import org.sonar.server.search.ws.BaseMapping; +import org.sonar.server.search.ws.SearchOptions; import org.sonar.server.text.MacroInterpreter; +import javax.annotation.CheckForNull; +import javax.annotation.Nullable; +import java.util.Collection; +import java.util.Map; + /** - * Conversion between RuleDoc and WS JSON response + * Conversion of {@link org.sonar.server.rule.index.RuleDoc} to WS JSON document */ -public class RuleMapping extends BaseMapping { - - public RuleMapping(Languages languages, MacroInterpreter macroInterpreter) { - super(); - addIndexStringField("repo", RuleNormalizer.RuleField.REPOSITORY.field()); - addIndexStringField("name", RuleNormalizer.RuleField.NAME.field()); - addIndexDatetimeField("createdAt", RuleNormalizer.RuleField.CREATED_AT.field()); - addField("htmlDesc", new HtmlDescField(macroInterpreter)); - addIndexStringField("severity", RuleNormalizer.RuleField.SEVERITY.field()); - addIndexStringField("status", RuleNormalizer.RuleField.STATUS.field()); - addIndexStringField("internalKey", RuleNormalizer.RuleField.INTERNAL_KEY.field()); - addIndexBooleanField("isTemplate", RuleNormalizer.RuleField.IS_TEMPLATE.field()); - addIndexStringField("templateKey", RuleNormalizer.RuleField.TEMPLATE_KEY.field()); - addIndexArrayField("tags", RuleNormalizer.RuleField.TAGS.field()); - addIndexArrayField("sysTags", RuleNormalizer.RuleField.SYSTEM_TAGS.field()); - addField("defaultDebtChar", new IndexStringField("defaultDebtChar", RuleNormalizer.RuleField.DEFAULT_CHARACTERISTIC.field())); - addField("defaultDebtChar", new IndexStringField("defaultDebtSubChar", RuleNormalizer.RuleField.DEFAULT_SUB_CHARACTERISTIC.field())); - addField("debtChar", new IndexStringField("debtChar", RuleNormalizer.RuleField.CHARACTERISTIC.field(), +public class RuleMapping extends BaseMapping<RuleDoc, RuleMappingContext> { + + private final DebtModel debtModel; + + public RuleMapping(final Languages languages, final MacroInterpreter macroInterpreter, final DebtModel debtModel) { + this.debtModel = debtModel; + + mapBasicFields(languages); + mapDescriptionFields(macroInterpreter); + mapDebtFields(); + mapParamFields(); + } + + private void mapBasicFields(final Languages languages) { + map("repo", RuleNormalizer.RuleField.REPOSITORY.field()); + map("name", RuleNormalizer.RuleField.NAME.field()); + mapDateTime("createdAt", RuleNormalizer.RuleField.CREATED_AT.field()); + map("severity", RuleNormalizer.RuleField.SEVERITY.field()); + map("status", RuleNormalizer.RuleField.STATUS.field()); + map("internalKey", RuleNormalizer.RuleField.INTERNAL_KEY.field()); + mapBoolean("isTemplate", RuleNormalizer.RuleField.IS_TEMPLATE.field()); + map("templateKey", RuleNormalizer.RuleField.TEMPLATE_KEY.field()); + mapArray("tags", RuleNormalizer.RuleField.TAGS.field()); + mapArray("sysTags", RuleNormalizer.RuleField.SYSTEM_TAGS.field()); + map("lang", RuleNormalizer.RuleField.LANGUAGE.field()); + map("langName", new IndexMapper<RuleDoc, RuleMappingContext>(RuleNormalizer.RuleField.LANGUAGE.field()) { + @Override + public void write(JsonWriter json, RuleDoc rule, RuleMappingContext context) { + Language lang = languages.get(rule.language()); + json.prop("langName", lang != null ? lang.getName() : null); + } + }); + } + + private void mapDescriptionFields(final MacroInterpreter macroInterpreter) { + map("htmlDesc", new Mapper<RuleDoc, RuleMappingContext>() { + @Override + public void write(JsonWriter json, RuleDoc rule, RuleMappingContext context) { + String html = rule.htmlDescription(); + if (html != null) { + if (rule.isManual() || rule.templateKey() != null) { + String desc = StringEscapeUtils.escapeHtml(html); + desc = desc.replaceAll("\\n", "<br/>"); + json.prop("htmlDesc", desc); + } else { + json.prop("htmlDesc", macroInterpreter.interpret(html)); + } + } + } + }); + map("noteLogin", RuleNormalizer.RuleField.NOTE_LOGIN.field()); + map("mdNote", RuleNormalizer.RuleField.NOTE.field()); + map("htmlNote", new IndexMapper<RuleDoc, RuleMappingContext>(RuleNormalizer.RuleField.NOTE.field()) { + @Override + public void write(JsonWriter json, RuleDoc rule, RuleMappingContext context) { + String markdownNote = rule.markdownNote(); + if (markdownNote != null) { + json.prop("htmlNote", macroInterpreter.interpret(Markdown.convertToHtml(markdownNote))); + } + } + }); + } + + private void mapDebtFields() { + map("defaultDebtChar", new IndexStringMapper("defaultDebtChar", RuleNormalizer.RuleField.DEFAULT_CHARACTERISTIC.field())); + map("defaultDebtSubChar", new IndexStringMapper("defaultDebtSubChar", RuleNormalizer.RuleField.DEFAULT_SUB_CHARACTERISTIC.field())); + map("debtChar", new IndexStringMapper("debtChar", RuleNormalizer.RuleField.CHARACTERISTIC.field(), RuleNormalizer.RuleField.DEFAULT_CHARACTERISTIC.field())); - addField("debtChar", new IndexStringField("debtSubChar", RuleNormalizer.RuleField.SUB_CHARACTERISTIC.field(), + map("debtSubChar", new IndexStringMapper("debtSubChar", RuleNormalizer.RuleField.SUB_CHARACTERISTIC.field(), RuleNormalizer.RuleField.DEFAULT_SUB_CHARACTERISTIC.field())); - addField("debtRemFn", new IndexStringField("debtRemFnType", RuleNormalizer.RuleField.DEBT_FUNCTION_TYPE.field(), + map("debtCharName", new CharacteristicNameMapper()); + map("debtSubCharName", new SubCharacteristicNameMapper()); + map("debtRemFn", new IndexStringMapper("debtRemFnType", RuleNormalizer.RuleField.DEBT_FUNCTION_TYPE.field(), RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_TYPE.field())); - addField("debtRemFn", new IndexStringField("debtRemFnCoeff", RuleNormalizer.RuleField.DEBT_FUNCTION_COEFFICIENT.field(), + map("debtRemFn", new IndexStringMapper("debtRemFnCoeff", RuleNormalizer.RuleField.DEBT_FUNCTION_COEFFICIENT.field(), RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_COEFFICIENT.field())); - addField("debtRemFn", new IndexStringField("debtRemFnOffset", RuleNormalizer.RuleField.DEBT_FUNCTION_OFFSET.field(), + map("debtRemFn", new IndexStringMapper("debtRemFnOffset", RuleNormalizer.RuleField.DEBT_FUNCTION_OFFSET.field(), RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_OFFSET.field())); - addField("defaultDebtRemFn", new IndexStringField("defaultDebtRemFnType", RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_TYPE.field())); - addField("defaultDebtRemFn", new IndexStringField("defaultDebtRemFnCoeff", RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_COEFFICIENT.field())); - addField("defaultDebtRemFn", new IndexStringField("defaultDebtRemFnOffset", RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_OFFSET.field())); - addIndexStringField("effortToFixDescription", RuleNormalizer.RuleField.FIX_DESCRIPTION.field()); - addIndexStringField("mdNote", RuleNormalizer.RuleField.NOTE.field()); - addField("htmlNote", new HtmlNoteField(macroInterpreter)); - addIndexStringField("noteLogin", RuleNormalizer.RuleField.NOTE_LOGIN.field()); - addIndexStringField("lang", RuleNormalizer.RuleField.LANGUAGE.field()); - addField("langName", new LangNameField(languages)); - addField("debtCharName", new CharacteristicNameField()); - addField("debtSubCharName", new SubCharacteristicNameField()); - addField("debtOverloaded", new OverriddenField()); - addField("params", new ParamsField()); + map("defaultDebtRemFn", new IndexStringMapper("defaultDebtRemFnType", RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_TYPE.field())); + map("defaultDebtRemFn", new IndexStringMapper("defaultDebtRemFnCoeff", RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_COEFFICIENT.field())); + map("defaultDebtRemFn", new IndexStringMapper("defaultDebtRemFnOffset", RuleNormalizer.RuleField.DEFAULT_DEBT_FUNCTION_OFFSET.field())); + map("effortToFixDescription", RuleNormalizer.RuleField.FIX_DESCRIPTION.field()); + map("debtOverloaded", new OverriddenMapper()); } - private static class ParamsField extends IndexField<Rule> { - ParamsField() { - super(RuleNormalizer.RuleField.PARAMS.field()); - } - - @Override - public void write(JsonWriter json, Rule rule) { - json.name("params").beginArray(); - for (RuleParam param : rule.params()) { - json - .beginObject() - .prop("key", param.key()) - .prop("desc", param.description()) - .prop("defaultValue", param.defaultValue()) - .endObject(); + private void mapParamFields() { + map("params", new IndexMapper<RuleDoc, RuleMappingContext>(RuleNormalizer.RuleField.PARAMS.field()) { + @Override + public void write(JsonWriter json, RuleDoc rule, RuleMappingContext context) { + json.name("params").beginArray(); + for (RuleParam param : rule.params()) { + json + .beginObject() + .prop("key", param.key()) + .prop("desc", param.description()) + .prop("defaultValue", param.defaultValue()) + .endObject(); + } + json.endArray(); } - json.endArray(); - } + }); } - private static class LangNameField extends IndexField<Rule> { - private final Languages languages; - - private LangNameField(Languages languages) { - super(RuleNormalizer.RuleField.LANGUAGE.field()); - this.languages = languages; + public void write(Rule rule, JsonWriter json, @Nullable SearchOptions options) { + RuleMappingContext context = new RuleMappingContext(); + if (needDebtCharacteristicNames(options) && rule.debtCharacteristicKey() != null) { + // load debt characteristics if requested + context.add(debtModel.characteristicByKey(rule.debtCharacteristicKey())); } + if (needDebtSubCharacteristicNames(options) && rule.debtSubCharacteristicKey() != null) { + context.add(debtModel.characteristicByKey(rule.debtSubCharacteristicKey())); + } + doWrite((RuleDoc) rule, context, json, options); + } - @Override - public void write(JsonWriter json, Rule rule) { - String langKey = rule.language(); - Language lang = languages.get(langKey); - json.prop("langName", lang != null ? lang.getName() : null); + public void write(Collection<Rule> rules, JsonWriter json, @Nullable SearchOptions options) { + if (!rules.isEmpty()) { + RuleMappingContext context = new RuleMappingContext(); + if (needDebtCharacteristicNames(options) || needDebtSubCharacteristicNames(options)) { + // load all debt characteristics + context.addAll(debtModel.allCharacteristics()); + } + for (Rule rule : rules) { + doWrite((RuleDoc) rule, context, json, options); + } } } - private static class HtmlNoteField extends IndexField<Rule> { - private final MacroInterpreter macroInterpreter; + private boolean needDebtCharacteristicNames(@Nullable SearchOptions options) { + return options == null || options.hasField("debtCharName"); + } + + private boolean needDebtSubCharacteristicNames(@Nullable SearchOptions options) { + return options == null || options.hasField("debtSubCharName"); + } - private HtmlNoteField(MacroInterpreter macroInterpreter) { - super(RuleNormalizer.RuleField.NOTE.field()); - this.macroInterpreter = macroInterpreter; + private static class CharacteristicNameMapper extends IndexMapper<RuleDoc, RuleMappingContext> { + private CharacteristicNameMapper() { + super(RuleNormalizer.RuleField.CHARACTERISTIC.field(), RuleNormalizer.RuleField.DEFAULT_CHARACTERISTIC.field()); } @Override - public void write(JsonWriter json, Rule rule) { - String markdownNote = rule.markdownNote(); - if (markdownNote != null) { - json.prop("htmlNote", macroInterpreter.interpret(Markdown.convertToHtml(markdownNote))); - } + public void write(JsonWriter json, RuleDoc rule, RuleMappingContext context) { + json.prop("debtCharName", context.debtCharacteristicName(rule.debtCharacteristicKey())); } } - private static class HtmlDescField implements Field<Rule> { - private final MacroInterpreter macroInterpreter; - - private HtmlDescField(MacroInterpreter macroInterpreter) { - this.macroInterpreter = macroInterpreter; + private static class SubCharacteristicNameMapper extends IndexMapper<RuleDoc, RuleMappingContext> { + private SubCharacteristicNameMapper() { + super(RuleNormalizer.RuleField.SUB_CHARACTERISTIC.field(), RuleNormalizer.RuleField.DEFAULT_SUB_CHARACTERISTIC.field()); } @Override - public void write(JsonWriter json, Rule rule) { - String html = rule.htmlDescription(); - if (html != null) { - if (rule.isManual() || rule.templateKey() != null) { - String desc = StringEscapeUtils.escapeHtml(html); - desc = desc.replaceAll("\\n", "<br/>"); - json.prop("htmlDesc", desc); - } else { - json.prop("htmlDesc", macroInterpreter.interpret(html)); - } - } + public void write(JsonWriter json, RuleDoc rule, RuleMappingContext context) { + json.prop("debtSubCharName", context.debtCharacteristicName(rule.debtSubCharacteristicKey())); } } - private static class CharacteristicNameField implements Field<Rule> { + private static class OverriddenMapper implements Mapper<RuleDoc, RuleMappingContext> { @Override - public void write(JsonWriter json, Rule rule) { - // TODO set characteristic name - json.prop("debtCharName", rule.debtCharacteristicKey()); + public void write(JsonWriter json, RuleDoc rule, RuleMappingContext context) { + json.prop("debtOverloaded", rule.debtOverloaded()); } } +} - private static class SubCharacteristicNameField implements Field<Rule> { - @Override - public void write(JsonWriter json, Rule rule) { - // TODO set characteristic name - json.prop("debtSubCharName", rule.debtSubCharacteristicKey()); +class RuleMappingContext { + private final Map<String, String> debtCharacteristicNamesByKey = Maps.newHashMap(); + + @CheckForNull + public String debtCharacteristicName(String key) { + return debtCharacteristicNamesByKey.get(key); + } + + void add(@Nullable DebtCharacteristic c) { + if (c != null) { + debtCharacteristicNamesByKey.put(c.key(), c.name()); } } - private static class OverriddenField implements Field<Rule> { - @Override - public void write(JsonWriter json, Rule rule) { - json.prop("debtOverloaded", rule.debtOverloaded()); + void addAll(Collection<DebtCharacteristic> coll) { + for (DebtCharacteristic c : coll) { + add(c); } } } diff --git a/sonar-server/src/main/java/org/sonar/server/rule/ws/SearchAction.java b/sonar-server/src/main/java/org/sonar/server/rule/ws/SearchAction.java index b4789c99836..8f34fe3015c 100644 --- a/sonar-server/src/main/java/org/sonar/server/rule/ws/SearchAction.java +++ b/sonar-server/src/main/java/org/sonar/server/rule/ws/SearchAction.java @@ -110,7 +110,7 @@ public class SearchAction implements RequestHandler { action .createParam(PARAM_KEY) - .setDescription("Single or list of keys of rule to search for") + .setDescription("Key of rule to search for") .setExampleValue("squid:S001"); action diff --git a/sonar-server/src/main/java/org/sonar/server/rule/ws/ShowAction.java b/sonar-server/src/main/java/org/sonar/server/rule/ws/ShowAction.java index 9bbc2018630..fb2d98994f6 100644 --- a/sonar-server/src/main/java/org/sonar/server/rule/ws/ShowAction.java +++ b/sonar-server/src/main/java/org/sonar/server/rule/ws/ShowAction.java @@ -29,7 +29,6 @@ import org.sonar.api.utils.text.JsonWriter; import org.sonar.server.exceptions.NotFoundException; import org.sonar.server.rule.Rule; import org.sonar.server.rule.RuleService; -import org.sonar.server.search.BaseDoc; /** * @since 4.4 @@ -78,7 +77,7 @@ public class ShowAction implements RequestHandler { throw new NotFoundException("Rule not found: " + key); } JsonWriter json = response.newJsonWriter().beginObject().name("rule"); - mapping.write((BaseDoc) rule, json); + mapping.write(rule, json, null /* TODO replace by SearchOptions immutable constant */); if (request.mandatoryParamAsBoolean(PARAM_ACTIVES)) { activeRuleCompleter.completeShow(rule, json); diff --git a/sonar-server/src/main/java/org/sonar/server/rule/ws/UpdateAction.java b/sonar-server/src/main/java/org/sonar/server/rule/ws/UpdateAction.java index 553491f93dc..a2246626b5f 100644 --- a/sonar-server/src/main/java/org/sonar/server/rule/ws/UpdateAction.java +++ b/sonar-server/src/main/java/org/sonar/server/rule/ws/UpdateAction.java @@ -37,7 +37,6 @@ import org.sonar.server.exceptions.NotFoundException; import org.sonar.server.rule.Rule; import org.sonar.server.rule.RuleService; import org.sonar.server.rule.RuleUpdate; -import org.sonar.server.search.BaseDoc; public class UpdateAction implements RequestHandler { @@ -160,7 +159,7 @@ public class UpdateAction implements RequestHandler { return update; } - private RuleUpdate createRuleUpdate(RuleKey key){ + private RuleUpdate createRuleUpdate(RuleKey key) { Rule rule = service.getByKey(key); if (rule == null) { throw new NotFoundException("This rule does not exists : " + key); @@ -215,7 +214,7 @@ public class UpdateAction implements RequestHandler { private void writeResponse(Response response, RuleKey ruleKey) { Rule rule = service.getNonNullByKey(ruleKey); JsonWriter json = response.newJsonWriter().beginObject().name("rule"); - mapping.write((BaseDoc) rule, json); + mapping.write(rule, json, null /* TODO replace by SearchOptions immutable constant */); json.endObject().close(); } } diff --git a/sonar-server/src/main/java/org/sonar/server/search/ws/BaseMapping.java b/sonar-server/src/main/java/org/sonar/server/search/ws/BaseMapping.java index 0d7e44d56d8..69a73937c6a 100644 --- a/sonar-server/src/main/java/org/sonar/server/search/ws/BaseMapping.java +++ b/sonar-server/src/main/java/org/sonar/server/search/ws/BaseMapping.java @@ -34,15 +34,18 @@ import java.util.List; import java.util.Set; /** - * Mapping of search documents (see BaseDoc) to WS JSON responses + * Mapping of search documents (see {@link org.sonar.server.search.BaseDoc}) to WS JSON responses */ -public abstract class BaseMapping implements ServerComponent { +public abstract class BaseMapping<DOC extends BaseDoc, CTX> implements ServerComponent { - private final Multimap<String, String> indexFields = LinkedHashMultimap.create(); - private final Multimap<String, BaseMapping.Field> fields = LinkedHashMultimap.create(); + private final Multimap<String, String> indexFieldsByWsFields = LinkedHashMultimap.create(); + private final Multimap<String, Mapper> mappers = LinkedHashMultimap.create(); + /** + * All the WS supported fields + */ public Set<String> supportedFields() { - return fields.keySet(); + return mappers.keySet(); } public QueryOptions newQueryOptions(SearchOptions options) { @@ -51,67 +54,73 @@ public abstract class BaseMapping implements ServerComponent { List<String> optionFields = options.fields(); if (optionFields != null) { for (String optionField : optionFields) { - result.addFieldsToReturn(this.indexFields.get(optionField)); + result.addFieldsToReturn(indexFieldsByWsFields.get(optionField)); } } return result; } - public void write(BaseDoc doc, JsonWriter json) { - write(doc, json, null); + /** + * Write all document fields + */ + protected void doWrite(DOC doc, @Nullable CTX context, JsonWriter json) { + doWrite(doc, context, json, null); } - public void write(BaseDoc doc, JsonWriter json, @Nullable SearchOptions options) { + /** + * Write only requested document fields + */ + protected void doWrite(DOC doc, @Nullable CTX context, JsonWriter json, @Nullable SearchOptions options) { json.beginObject(); json.prop("key", doc.keyField()); if (options == null || options.fields() == null) { // return all fields - for (BaseMapping.Field field : fields.values()) { - field.write(json, doc); + for (Mapper mapper : mappers.values()) { + mapper.write(json, doc, context); } } else { for (String optionField : options.fields()) { - for (BaseMapping.Field field : fields.get(optionField)) { - field.write(json, doc); + for (Mapper mapper : mappers.get(optionField)) { + mapper.write(json, doc, context); } } } json.endObject(); } - protected BaseMapping addIndexStringField(String key, String indexKey) { - return addField(key, new IndexStringField(key, indexKey)); + protected BaseMapping map(String key, String indexKey) { + return map(key, new IndexStringMapper(key, indexKey)); } - protected BaseMapping addIndexBooleanField(String key, String indexKey) { - return addField(key, new IndexBooleanField(key, indexKey)); + protected BaseMapping mapBoolean(String key, String indexKey) { + return map(key, new IndexBooleanMapper(key, indexKey)); } - protected BaseMapping addIndexDatetimeField(String key, String indexKey) { - return addField(key, new IndexDatetimeField(key, indexKey)); + protected BaseMapping mapDateTime(String key, String indexKey) { + return map(key, new IndexDatetimeMapper(key, indexKey)); } - protected BaseMapping addIndexArrayField(String key, String indexKey) { - return addField(key, new IndexArrayField(key, indexKey)); + protected BaseMapping mapArray(String key, String indexKey) { + return map(key, new IndexArrayMapper(key, indexKey)); } - protected BaseMapping addField(String key, Field field) { - fields.put(key, field); - if (field instanceof IndexField) { - IndexField indexField = (IndexField) field; - indexFields.putAll(key, Arrays.asList(indexField.indexFields())); + protected BaseMapping map(String key, Mapper mapper) { + mappers.put(key, mapper); + if (mapper instanceof IndexMapper) { + IndexMapper indexField = (IndexMapper) mapper; + indexFieldsByWsFields.putAll(key, Arrays.asList(indexField.indexFields())); } return this; } - public static interface Field<D> { - void write(JsonWriter json, D doc); + public static interface Mapper<DOC extends BaseDoc, CTX> { + void write(JsonWriter json, DOC doc, CTX context); } - public abstract static class IndexField<D> implements Field<D> { + public abstract static class IndexMapper<DOC extends BaseDoc, CTX> implements Mapper<DOC, CTX> { protected final String[] indexFields; - protected IndexField(String... indexFields) { + protected IndexMapper(String... indexFields) { this.indexFields = indexFields; } @@ -123,21 +132,21 @@ public abstract class BaseMapping implements ServerComponent { /** * String field */ - public static class IndexStringField extends IndexField<BaseDoc> { + public static class IndexStringMapper<DOC extends BaseDoc, CTX> extends IndexMapper<DOC,CTX> { private final String key; - public IndexStringField(String key, String indexKey, String defaultIndexKey) { + public IndexStringMapper(String key, String indexKey, String defaultIndexKey) { super(indexKey, defaultIndexKey); this.key = key; } - public IndexStringField(String key, String indexKey) { + public IndexStringMapper(String key, String indexKey) { super(indexKey); this.key = key; } @Override - public void write(JsonWriter json, BaseDoc doc) { + public void write(JsonWriter json, DOC doc, CTX context) { Object val = doc.getNullableField(indexFields[0]); if (val == null && indexFields.length == 2) { // There is an alternative value @@ -147,31 +156,31 @@ public abstract class BaseMapping implements ServerComponent { } } - public static class IndexBooleanField extends IndexField<BaseDoc> { + public static class IndexBooleanMapper<DOC extends BaseDoc, CTX> extends IndexMapper<DOC,CTX> { private final String key; - public IndexBooleanField(String key, String indexKey) { + public IndexBooleanMapper(String key, String indexKey) { super(indexKey); this.key = key; } @Override - public void write(JsonWriter json, BaseDoc doc) { + public void write(JsonWriter json, DOC doc, CTX context) { Boolean val = doc.getNullableField(indexFields[0]); json.prop(key, val != null ? val.booleanValue() : null); } } - public static class IndexArrayField extends IndexField<BaseDoc> { + public static class IndexArrayMapper<DOC extends BaseDoc, CTX> extends IndexMapper<DOC,CTX> { private final String key; - public IndexArrayField(String key, String indexKey) { + public IndexArrayMapper(String key, String indexKey) { super(indexKey); this.key = key; } @Override - public void write(JsonWriter json, BaseDoc doc) { + public void write(JsonWriter json, DOC doc, CTX context) { Iterable<String> values = doc.getNullableField(indexFields[0]); if (values != null) { json.name(key).beginArray().values(values).endArray(); @@ -179,16 +188,16 @@ public abstract class BaseMapping implements ServerComponent { } } - public static class IndexDatetimeField extends IndexField<BaseDoc> { + public static class IndexDatetimeMapper<DOC extends BaseDoc, CTX> extends IndexMapper<DOC,CTX> { private final String key; - public IndexDatetimeField(String key, String indexKey) { + public IndexDatetimeMapper(String key, String indexKey) { super(indexKey); this.key = key; } @Override - public void write(JsonWriter json, BaseDoc doc) { + public void write(JsonWriter json, DOC doc, CTX context) { String val = doc.getNullableField(indexFields[0]); if (val != null) { json.propDateTime(key, IndexUtils.parseDateTime(val)); |