aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTeryk Bellahsene <teryk.bellahsene@sonarsource.com>2016-05-03 18:06:32 +0200
committerTeryk Bellahsene <teryk.bellahsene@sonarsource.com>2016-05-06 11:14:12 +0200
commit2904c7846d8233f71a6340c6f7f884537bdc7b6a (patch)
tree14cb0462f50add7d4fa604ce20747bb4e3829969
parent303324b8c95a9c73870d850e68da5d01a648c0bc (diff)
downloadsonarqube-2904c7846d8233f71a6340c6f7f884537bdc7b6a.tar.gz
sonarqube-2904c7846d8233f71a6340c6f7f884537bdc7b6a.zip
SONAR-7380 Sanitize exception handling of WS parameter reads
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/ws/L10nWs.java1
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/ws/WebServiceEngine.java2
-rw-r--r--sonar-plugin-api/src/main/java/org/sonar/api/server/ws/Request.java39
-rw-r--r--sonar-plugin-api/src/test/java/org/sonar/api/server/ws/RequestTest.java387
4 files changed, 250 insertions, 179 deletions
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/ws/L10nWs.java b/server/sonar-server/src/main/java/org/sonar/server/platform/ws/L10nWs.java
index a3639a6e456..bbfd31e32ed 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/platform/ws/L10nWs.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/ws/L10nWs.java
@@ -76,7 +76,6 @@ public class L10nWs implements WebService {
if (timestamp != null && timestamp.after(server.getStartedAt())) {
response.stream().setStatus(HttpURLConnection.HTTP_NOT_MODIFIED).output().close();
} else {
-
Locale locale = userSession.locale();
String localeParam = request.param("locale");
if (localeParam != null) {
diff --git a/server/sonar-server/src/main/java/org/sonar/server/ws/WebServiceEngine.java b/server/sonar-server/src/main/java/org/sonar/server/ws/WebServiceEngine.java
index f0a5dad038f..274955d3e60 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/ws/WebServiceEngine.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/ws/WebServiceEngine.java
@@ -98,9 +98,7 @@ public class WebServiceEngine implements LocalConnector, Startable {
}
verifyRequest(action, request);
action.handler().handle(request, response);
-
} catch (IllegalArgumentException e) {
- // TODO replace by BadRequestException in Request#mandatoryParam()
sendErrors(response, 400, new Errors().add(Message.of(e.getMessage())));
} catch (BadRequestException e) {
sendErrors(response, 400, e.errors());
diff --git a/sonar-plugin-api/src/main/java/org/sonar/api/server/ws/Request.java b/sonar-plugin-api/src/main/java/org/sonar/api/server/ws/Request.java
index 6396ecab761..c746da58dfa 100644
--- a/sonar-plugin-api/src/main/java/org/sonar/api/server/ws/Request.java
+++ b/sonar-plugin-api/src/main/java/org/sonar/api/server/ws/Request.java
@@ -82,7 +82,7 @@ public abstract class Request {
*/
public int mandatoryParamAsInt(String key) {
String s = mandatoryParam(key);
- return Integer.parseInt(s);
+ return parseInt(key, s);
}
/**
@@ -92,7 +92,7 @@ public abstract class Request {
*/
public long mandatoryParamAsLong(String key) {
String s = mandatoryParam(key);
- return Long.parseLong(s);
+ return parseLong(key, s);
}
public <E extends Enum<E>> E mandatoryParamAsEnum(String key, Class<E> enumClass) {
@@ -148,7 +148,7 @@ public abstract class Request {
@Deprecated
public int paramAsInt(String key, int defaultValue) {
String s = param(key);
- return s == null ? defaultValue : Integer.parseInt(s);
+ return s == null ? defaultValue : parseInt(key, s);
}
/**
@@ -158,7 +158,7 @@ public abstract class Request {
@Deprecated
public long paramAsLong(String key, long defaultValue) {
String s = param(key);
- return s == null ? defaultValue : Long.parseLong(s);
+ return s == null ? defaultValue : parseLong(key, s);
}
@CheckForNull
@@ -170,13 +170,13 @@ public abstract class Request {
@CheckForNull
public Integer paramAsInt(String key) {
String s = param(key);
- return s == null ? null : Integer.parseInt(s);
+ return s == null ? null : parseInt(key, s);
}
@CheckForNull
public Long paramAsLong(String key) {
String s = param(key);
- return s == null ? null : Long.parseLong(s);
+ return s == null ? null : parseLong(key, s);
}
@CheckForNull
@@ -210,7 +210,7 @@ public abstract class Request {
try {
return DateUtils.parseDate(s);
} catch (SonarException notDateEither) {
- throw new SonarException(String.format("'%s' cannot be parsed as either a date or date+time", s));
+ throw new IllegalArgumentException(String.format("'%s' cannot be parsed as either a date or date+time", s));
}
}
}
@@ -220,10 +220,15 @@ public abstract class Request {
@CheckForNull
public Date paramAsDate(String key) {
String s = param(key);
- if (s != null) {
+ if (s == null) {
+ return null;
+ }
+
+ try {
return DateUtils.parseDate(s);
+ } catch (SonarException notDateException) {
+ throw new IllegalArgumentException(notDateException);
}
- return null;
}
private static boolean parseBoolean(String key, String value) {
@@ -236,6 +241,22 @@ public abstract class Request {
throw new IllegalArgumentException(String.format("Property %s is not a boolean value: %s", key, value));
}
+ private static int parseInt(String key, String value) {
+ try {
+ return Integer.parseInt(value);
+ } catch (NumberFormatException expection) {
+ throw new IllegalArgumentException(String.format("The '%s' parameter cannot be parsed as an integer value: %s", key, value));
+ }
+ }
+
+ private static long parseLong(String key, String value) {
+ try {
+ return Long.parseLong(value);
+ } catch (NumberFormatException expection) {
+ throw new IllegalArgumentException(String.format("The '%s' parameter cannot be parsed as a long value: %s", key, value));
+ }
+ }
+
/**
* Allows a web service to call another web service.
* @see LocalConnector
diff --git a/sonar-plugin-api/src/test/java/org/sonar/api/server/ws/RequestTest.java b/sonar-plugin-api/src/test/java/org/sonar/api/server/ws/RequestTest.java
index da619d77894..a25365196a7 100644
--- a/sonar-plugin-api/src/test/java/org/sonar/api/server/ws/RequestTest.java
+++ b/sonar-plugin-api/src/test/java/org/sonar/api/server/ws/RequestTest.java
@@ -20,268 +20,321 @@
package org.sonar.api.server.ws;
import com.google.common.collect.Maps;
+import java.io.InputStream;
+import java.util.Map;
+import javax.annotation.Nullable;
import org.apache.commons.io.IOUtils;
import org.junit.Before;
+import org.junit.Rule;
import org.junit.Test;
+import org.junit.rules.ExpectedException;
import org.sonar.api.rule.RuleStatus;
import org.sonar.api.server.ws.internal.ValidatingRequest;
import org.sonar.api.utils.DateUtils;
-import org.sonar.api.utils.SonarException;
-
-import javax.annotation.Nullable;
-
-import java.io.InputStream;
-import java.util.Map;
import static org.assertj.core.api.Assertions.assertThat;
-import static org.junit.Assert.fail;
import static org.mockito.Mockito.mock;
public class RequestTest {
- private static class SimpleRequest extends ValidatingRequest {
-
- private final Map<String, String> params = Maps.newHashMap();
-
- @Override
- public String method() {
- return "GET";
- }
-
- @Override
- public String getMediaType() {
- return "application/json";
- }
-
- @Override
- public boolean hasParam(String key) {
- return params.keySet().contains(key);
- }
-
- public SimpleRequest setParam(String key, @Nullable String value) {
- if (value != null) {
- params.put(key, value);
- }
- return this;
- }
-
- @Override
- protected String readParam(String key) {
- return params.get(key);
- }
-
- @Override
- protected InputStream readInputStreamParam(String key) {
- String param = readParam(key);
-
- return param == null ? null : IOUtils.toInputStream(param);
- }
- }
-
- private static class SimpleWs implements WebService {
-
- @Override
- public void define(Context context) {
- NewController controller = context.createController("my_controller");
- NewAction action = controller.createAction("my_action")
- .setDescription("Action Description")
- .setPost(true)
- .setSince("5.2")
- .setHandler(mock(RequestHandler.class));
- action
- .createParam("required_param")
- .setRequired(true);
-
- action.createParam("a_string");
- action.createParam("a_boolean");
- action.createParam("a_number");
- action.createParam("a_enum");
- action.createParam("a_date");
- action.createParam("a_datetime");
-
- action.createParam("a_required_string").setRequired(true);
- action.createParam("a_required_boolean").setRequired(true);
- action.createParam("a_required_number").setRequired(true);
- action.createParam("a_required_enum").setRequired(true);
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
- action.createParam("has_default_string").setDefaultValue("the_default_string");
- action.createParam("has_default_number").setDefaultValue("10");
- action.createParam("has_default_boolean").setDefaultValue("true");
-
- action.createParam("has_possible_values").setPossibleValues("foo", "bar");
-
- action.createParam("new_param").setDeprecatedKey("deprecated_param");
- action.createParam("new_param_with_default_value").setDeprecatedKey("deprecated_new_param_with_default_value").setDefaultValue("the_default_string");
-
- controller.done();
- }
- }
-
- SimpleRequest request = new SimpleRequest();
+ FakeRequest underTest = new FakeRequest();
@Before
public void before() {
WebService.Context context = new WebService.Context();
- new SimpleWs().define(context);
- request.setAction(context.controller("my_controller").action("my_action"));
+ new FakeWs().define(context);
+
+ underTest.setAction(context.controller("my_controller").action("my_action"));
}
@Test
public void has_param() {
- request.setParam("a_required_string", "foo");
+ underTest.setParam("a_required_string", "foo");
- assertThat(request.hasParam("a_required_string")).isTrue();
- assertThat(request.hasParam("unknown")).isFalse();
+ assertThat(underTest.hasParam("a_required_string")).isTrue();
+ assertThat(underTest.hasParam("unknown")).isFalse();
}
@Test
public void required_param_is_missing() {
- try {
- request.mandatoryParam("required_param");
- fail();
- } catch (IllegalArgumentException e) {
- assertThat(e).hasMessage("The 'required_param' parameter is missing");
- }
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("The 'required_param' parameter is missing");
+
+ underTest.mandatoryParam("required_param");
}
@Test
public void required_param() {
- request.setParam("a_required_string", "foo");
- request.setParam("a_required_number", "42");
- request.setParam("a_required_boolean", "true");
- request.setParam("a_required_enum", "BETA");
-
- assertThat(request.mandatoryParam("a_required_string")).isEqualTo("foo");
- assertThat(request.mandatoryParamAsBoolean("a_required_boolean")).isTrue();
- assertThat(request.mandatoryParamAsInt("a_required_number")).isEqualTo(42);
- assertThat(request.mandatoryParamAsLong("a_required_number")).isEqualTo(42L);
- assertThat(request.mandatoryParamAsEnum("a_required_enum", RuleStatus.class)).isEqualTo(RuleStatus.BETA);
+ underTest.setParam("a_required_string", "foo");
+ underTest.setParam("a_required_number", "42");
+ underTest.setParam("a_required_boolean", "true");
+ underTest.setParam("a_required_enum", "BETA");
+
+ assertThat(underTest.mandatoryParam("a_required_string")).isEqualTo("foo");
+ assertThat(underTest.mandatoryParamAsBoolean("a_required_boolean")).isTrue();
+ assertThat(underTest.mandatoryParamAsInt("a_required_number")).isEqualTo(42);
+ assertThat(underTest.mandatoryParamAsLong("a_required_number")).isEqualTo(42L);
+ assertThat(underTest.mandatoryParamAsEnum("a_required_enum", RuleStatus.class)).isEqualTo(RuleStatus.BETA);
}
@Test
public void required_param_as_strings() {
- try {
- request.mandatoryParamAsStrings("a_required_string");
- } catch (IllegalArgumentException e) {
- assertThat(e).hasMessage("The 'a_required_string' parameter is missing");
- }
+ underTest.setParam("a_required_string", "foo,bar");
+
+ assertThat(underTest.mandatoryParamAsStrings("a_required_string")).containsExactly("foo", "bar");
+ }
+
+ @Test
+ public void fail_if_no_required_param_as_strings() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("The 'a_required_string' parameter is missing");
- request.setParam("a_required_string", "foo,bar");
- assertThat(request.mandatoryParamAsStrings("a_required_string")).containsExactly("foo", "bar");
+ underTest.mandatoryParamAsStrings("a_required_string");
}
@Test
public void default_value_of_optional_param() {
- assertThat(request.param("has_default_string")).isEqualTo("the_default_string");
+ assertThat(underTest.param("has_default_string")).isEqualTo("the_default_string");
}
@Test
public void param_as_string() {
- assertThat(request.setParam("a_string", "foo").param("a_string")).isEqualTo("foo");
- assertThat(request.setParam("a_string", " f o o \r\n ").param("a_string")).isEqualTo("f o o");
+ assertThat(underTest.setParam("a_string", "foo").param("a_string")).isEqualTo("foo");
+ assertThat(underTest.setParam("a_string", " f o o \r\n ").param("a_string")).isEqualTo("f o o");
}
@Test
public void null_param() {
- assertThat(request.param("a_string")).isNull();
- assertThat(request.paramAsBoolean("a_boolean")).isNull();
- assertThat(request.paramAsInt("a_number")).isNull();
- assertThat(request.paramAsLong("a_number")).isNull();
+ assertThat(underTest.param("a_string")).isNull();
+ assertThat(underTest.paramAsBoolean("a_boolean")).isNull();
+ assertThat(underTest.paramAsInt("a_number")).isNull();
+ assertThat(underTest.paramAsLong("a_number")).isNull();
}
@Test
public void param_as_int() {
- assertThat(request.setParam("a_number", "123").paramAsInt("a_number")).isEqualTo(123);
+ assertThat(underTest.setParam("a_number", "123").paramAsInt("a_number")).isEqualTo(123);
+ assertThat(underTest.setParam("a_number", "123").paramAsInt("a_number", 42)).isEqualTo(123);
+ assertThat(underTest.setParam("a_number", null).paramAsInt("a_number", 42)).isEqualTo(123);
+ }
+
+ @Test
+ public void fail_when_param_is_not_an_int() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("The 'a_number' parameter cannot be parsed as an integer value: not-an-int");
+
+ assertThat(underTest.setParam("a_number", "not-an-int").paramAsInt("a_number")).isEqualTo(123);
+ }
+
+ @Test
+ public void fail_when_param_is_not_an_int_with_default_value() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("The 'a_number' parameter cannot be parsed as an integer value: not_an_int");
+
+ underTest.setParam("a_number", "not_an_int").paramAsInt("a_number", 42);
}
@Test
public void param_as_long() {
- assertThat(request.setParam("a_number", "123").paramAsLong("a_number")).isEqualTo(123L);
- assertThat(request.setParam("a_number", "123").paramAsLong("a_number", 42L)).isEqualTo(123L);
- assertThat(request.setParam("a_number", null).paramAsLong("a_number", 42L)).isEqualTo(123L);
+ assertThat(underTest.setParam("a_number", "123").paramAsLong("a_number")).isEqualTo(123L);
+ assertThat(underTest.setParam("a_number", "123").paramAsLong("a_number", 42L)).isEqualTo(123L);
+ assertThat(underTest.setParam("a_number", null).paramAsLong("a_number", 42L)).isEqualTo(123L);
+ }
+
+ @Test
+ public void fail_when_param_is_not_a_long() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("The 'a_number' parameter cannot be parsed as a long value: not_a_long");
+
+ underTest.setParam("a_number", "not_a_long").paramAsLong("a_number");
+ }
+
+ @Test
+ public void fail_when_param_is_not_a_long_with_default_value() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("The 'a_number' parameter cannot be parsed as a long value: not_a_long");
+
+ underTest.setParam("a_number", "not_a_long").paramAsLong("a_number", 42L);
}
@Test
public void param_as_boolean() {
- assertThat(request.setParam("a_boolean", "true").paramAsBoolean("a_boolean")).isTrue();
- assertThat(request.setParam("a_boolean", "yes").paramAsBoolean("a_boolean")).isTrue();
- assertThat(request.setParam("a_boolean", "false").paramAsBoolean("a_boolean")).isFalse();
- assertThat(request.setParam("a_boolean", "no").paramAsBoolean("a_boolean")).isFalse();
- try {
- request.setParam("a_boolean", "oui").paramAsBoolean("a_boolean");
- fail();
- } catch (IllegalArgumentException e) {
- assertThat(e).hasMessage("Property a_boolean is not a boolean value: oui");
- }
+ assertThat(underTest.setParam("a_boolean", "true").paramAsBoolean("a_boolean")).isTrue();
+ assertThat(underTest.setParam("a_boolean", "yes").paramAsBoolean("a_boolean")).isTrue();
+ assertThat(underTest.setParam("a_boolean", "false").paramAsBoolean("a_boolean")).isFalse();
+ assertThat(underTest.setParam("a_boolean", "no").paramAsBoolean("a_boolean")).isFalse();
+ }
+
+ @Test
+ public void fail_if_incorrect_param_as_boolean() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("Property a_boolean is not a boolean value: oui");
+
+ underTest.setParam("a_boolean", "oui").paramAsBoolean("a_boolean");
}
@Test
public void param_as_enum() {
- assertThat(request.setParam("a_enum", "BETA").paramAsEnum("a_enum", RuleStatus.class)).isEqualTo(RuleStatus.BETA);
+ assertThat(underTest.setParam("a_enum", "BETA").paramAsEnum("a_enum", RuleStatus.class)).isEqualTo(RuleStatus.BETA);
}
@Test
public void param_as_enums() {
- assertThat(request.setParam("a_enum", "BETA,READY").paramAsEnums("a_enum", RuleStatus.class)).containsOnly(
+ assertThat(underTest.setParam("a_enum", "BETA,READY").paramAsEnums("a_enum", RuleStatus.class)).containsOnly(
RuleStatus.BETA, RuleStatus.READY);
}
@Test
public void param_as_date() {
- assertThat(request.setParam("a_date", "2014-05-27").paramAsDate("a_date")).isEqualTo(DateUtils.parseDate("2014-05-27"));
+ assertThat(underTest.setParam("a_date", "2014-05-27").paramAsDate("a_date")).isEqualTo(DateUtils.parseDate("2014-05-27"));
+ }
+
+ @Test
+ public void fail_when_param_as_date_not_a_date() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("The date 'polop' does not respect format 'yyyy-MM-dd'");
+
+ underTest.setParam("a_date", "polop").paramAsDate("a_date");
}
@Test
public void param_as_datetime() {
- assertThat(request.setParam("a_datetime", "2014-05-27T15:50:45+0100").paramAsDateTime("a_datetime")).isEqualTo(DateUtils.parseDateTime("2014-05-27T15:50:45+0100"));
- assertThat(request.setParam("a_datetime", "2014-05-27").paramAsDateTime("a_datetime")).isEqualTo(DateUtils.parseDate("2014-05-27"));
- try {
- request.setParam("a_datetime", "polop").paramAsDateTime("a_datetime");
- } catch (SonarException error) {
- assertThat(error.getMessage()).isEqualTo("'polop' cannot be parsed as either a date or date+time");
- }
+ assertThat(underTest.setParam("a_datetime", "2014-05-27T15:50:45+0100").paramAsDateTime("a_datetime")).isEqualTo(DateUtils.parseDateTime("2014-05-27T15:50:45+0100"));
+ assertThat(underTest.setParam("a_datetime", "2014-05-27").paramAsDateTime("a_datetime")).isEqualTo(DateUtils.parseDate("2014-05-27"));
+ }
+
+ @Test
+ public void fail_when_param_as_datetime_not_a_datetime() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("'polop' cannot be parsed as either a date or date+time");
+
+ underTest.setParam("a_datetime", "polop").paramAsDateTime("a_datetime");
}
@Test
public void param_as_strings() {
- assertThat(request.paramAsStrings("a_string")).isNull();
- assertThat(request.setParam("a_string", "").paramAsStrings("a_string")).isEmpty();
- assertThat(request.setParam("a_string", "bar").paramAsStrings("a_string")).containsExactly("bar");
- assertThat(request.setParam("a_string", "bar,baz").paramAsStrings("a_string")).containsExactly("bar", "baz");
- assertThat(request.setParam("a_string", "bar , baz").paramAsStrings("a_string")).containsExactly("bar", "baz");
+ assertThat(underTest.paramAsStrings("a_string")).isNull();
+ assertThat(underTest.setParam("a_string", "").paramAsStrings("a_string")).isEmpty();
+ assertThat(underTest.setParam("a_string", "bar").paramAsStrings("a_string")).containsExactly("bar");
+ assertThat(underTest.setParam("a_string", "bar,baz").paramAsStrings("a_string")).containsExactly("bar", "baz");
+ assertThat(underTest.setParam("a_string", "bar , baz").paramAsStrings("a_string")).containsExactly("bar", "baz");
}
@Test
public void deprecated_key() {
- assertThat(request.setParam("deprecated_param", "bar").param("new_param")).isEqualTo("bar");
+ assertThat(underTest.setParam("deprecated_param", "bar").param("new_param")).isEqualTo("bar");
}
@Test
public void fail_if_param_is_not_defined() {
- try {
- request.param("unknown");
- } catch (IllegalArgumentException e) {
- assertThat(e).hasMessage("BUG - parameter 'unknown' is undefined for action 'my_action'");
- }
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("BUG - parameter 'unknown' is undefined for action 'my_action'");
+
+ underTest.param("unknown");
}
@Test
public void verify_possible_values() {
- request.setParam("has_possible_values", "foo");
- assertThat(request.param("has_possible_values")).isEqualTo("foo");
-
- try {
- request.setParam("has_possible_values", "not_possible");
- request.param("has_possible_values");
- } catch (IllegalArgumentException e) {
- assertThat(e).hasMessage("Value of parameter 'has_possible_values' (not_possible) must be one of: [foo, bar]");
- }
+ underTest.setParam("has_possible_values", "foo");
+ assertThat(underTest.param("has_possible_values")).isEqualTo("foo");
+ }
+
+ @Test
+ public void fail_if_not_a_possible_value() {
+ underTest.setParam("has_possible_values", "not_possible");
+
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("Value of parameter 'has_possible_values' (not_possible) must be one of: [foo, bar]");
+
+ underTest.param("has_possible_values");
}
@Test
public void param_as_input_stream() throws Exception {
- assertThat(request.paramAsInputStream("a_string")).isNull();
- assertThat(IOUtils.toString(request.setParam("a_string", "").paramAsInputStream("a_string"))).isEmpty();
- assertThat(IOUtils.toString(request.setParam("a_string", "foo").paramAsInputStream("a_string"))).isEqualTo("foo");
+ assertThat(underTest.paramAsInputStream("a_string")).isNull();
+ assertThat(IOUtils.toString(underTest.setParam("a_string", "").paramAsInputStream("a_string"))).isEmpty();
+ assertThat(IOUtils.toString(underTest.setParam("a_string", "foo").paramAsInputStream("a_string"))).isEqualTo("foo");
+ }
+
+ private static class FakeRequest extends ValidatingRequest {
+
+ private final Map<String, String> params = Maps.newHashMap();
+
+ @Override
+ public String method() {
+ return "GET";
+ }
+
+ @Override
+ public String getMediaType() {
+ return "application/json";
+ }
+
+ @Override
+ public boolean hasParam(String key) {
+ return params.keySet().contains(key);
+ }
+
+ public FakeRequest setParam(String key, @Nullable String value) {
+ if (value != null) {
+ params.put(key, value);
+ }
+ return this;
+ }
+
+ @Override
+ protected String readParam(String key) {
+ return params.get(key);
+ }
+
+ @Override
+ protected InputStream readInputStreamParam(String key) {
+ String param = readParam(key);
+
+ return param == null ? null : IOUtils.toInputStream(param);
+ }
}
+
+ private static class FakeWs implements WebService {
+
+ @Override
+ public void define(Context context) {
+ NewController controller = context.createController("my_controller");
+ NewAction action = controller.createAction("my_action")
+ .setDescription("Action Description")
+ .setPost(true)
+ .setSince("5.2")
+ .setHandler(mock(RequestHandler.class));
+ action
+ .createParam("required_param")
+ .setRequired(true);
+
+ action.createParam("a_string");
+ action.createParam("a_boolean");
+ action.createParam("a_number");
+ action.createParam("a_enum");
+ action.createParam("a_date");
+ action.createParam("a_datetime");
+
+ action.createParam("a_required_string").setRequired(true);
+ action.createParam("a_required_boolean").setRequired(true);
+ action.createParam("a_required_number").setRequired(true);
+ action.createParam("a_required_enum").setRequired(true);
+
+ action.createParam("has_default_string").setDefaultValue("the_default_string");
+ action.createParam("has_default_number").setDefaultValue("10");
+ action.createParam("has_default_boolean").setDefaultValue("true");
+
+ action.createParam("has_possible_values").setPossibleValues("foo", "bar");
+
+ action.createParam("new_param").setDeprecatedKey("deprecated_param");
+ action.createParam("new_param_with_default_value").setDeprecatedKey("deprecated_new_param_with_default_value").setDefaultValue("the_default_string");
+
+ controller.done();
+ }
+ }
+
}