This is an automated email from the ASF dual-hosted git repository.
jamesbognar pushed a commit to branch master
in repository https://gitbox.apache.org/repos/asf/juneau.git
The following commit(s) were added to refs/heads/master by this push:
new 79d14da39 RequestQueryParams should be a list.
79d14da39 is described below
commit 79d14da393430c7bc929ce1448c0d6f94003f0e5
Author: JamesBognar <[email protected]>
AuthorDate: Tue Aug 16 11:52:23 2022 -0400
RequestQueryParams should be a list.
---
.../org/apache/juneau/rest/arg/HasQueryArg.java | 2 +-
.../java/org/apache/juneau/rest/arg/QueryArg.java | 3 +-
.../juneau/rest/httppart/RequestHeaders.java | 19 +-
.../juneau/rest/httppart/RequestQueryParams.java | 295 ++++++++++-----------
.../http/remote/Remote_QueryAnnotation_Test.java | 6 +-
5 files changed, 159 insertions(+), 166 deletions(-)
diff --git
a/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/arg/HasQueryArg.java
b/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/arg/HasQueryArg.java
index df37c2c51..a9499ff99 100644
---
a/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/arg/HasQueryArg.java
+++
b/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/arg/HasQueryArg.java
@@ -32,7 +32,7 @@ import org.apache.juneau.rest.httppart.*;
* <jv>opSession</jv>
* .{@link RestOpSession#getRequest() getRequest}()
* .{@link RestRequest#getQueryParams() getQueryParams}()
- * .{@link RequestQueryParams#contains(String...)
contains}(<jv>name</jv>);
+ * .{@link RequestQueryParams#contains(String)
contains}(<jv>name</jv>);
* </p>
*
* <p>
diff --git
a/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/arg/QueryArg.java
b/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/arg/QueryArg.java
index 2e143726f..4dd9dc205 100644
---
a/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/arg/QueryArg.java
+++
b/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/arg/QueryArg.java
@@ -109,8 +109,7 @@ public class QueryArg implements RestOpArg {
if (cm.isMapOrBean() && isOneOf(name, "*", "")) {
JsonMap m = new JsonMap();
- for (RequestQueryParam e : rh.getAll())
- m.put(e.getName(), e.parser(ps).schema(schema
== null ? null :
schema.getProperty(e.getName())).as(cm.getValueType()).orElse(null));
+ rh.forEach(e -> m.put(e.getName(),
e.parser(ps).schema(schema == null ? null :
schema.getProperty(e.getName())).as(cm.getValueType()).orElse(null)));
return req.getBeanSession().convertToType(m, cm);
}
diff --git
a/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/httppart/RequestHeaders.java
b/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/httppart/RequestHeaders.java
index adb7ba30b..4b17f03c2 100644
---
a/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/httppart/RequestHeaders.java
+++
b/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/httppart/RequestHeaders.java
@@ -137,13 +137,13 @@ public class RequestHeaders extends
ArrayList<RequestHeader> {
// Parameters defined on the request URL overwrite existing
headers.
Set<String> allowedHeaderParams =
req.getContext().getAllowedHeaderParams();
- for (RequestQueryParam p : query.getAll()) {
+ query.forEach(p -> {
String name = p.getName();
String key = key(name);
if (allowedHeaderParams.contains(key) ||
allowedHeaderParams.contains("*")) {
set(name, p.getValue());
}
- }
+ });
}
/**
@@ -182,12 +182,13 @@ public class RequestHeaders extends
ArrayList<RequestHeader> {
}
/**
- * Sets case sensitivity to <jk>true</jk> for names in this set.
+ * Sets case sensitivity for names in this list.
*
+ * @param value The new value for this setting.
* @return This object (for method chaining).
*/
- public RequestHeaders caseSensitive() {
- this.caseSensitive = true;
+ public RequestHeaders caseSensitive(boolean value) {
+ this.caseSensitive = value;
return this;
}
@@ -324,18 +325,18 @@ public class RequestHeaders extends
ArrayList<RequestHeader> {
*/
public RequestHeaders remove(String name) {
assertArgNotNull("name", name);
- stream(name).forEach(x -> remove(x));
+ removeIf(x -> eq(x.getName(), name));
return this;
}
/**
* Returns a copy of this object but only with the specified header
names copied.
*
- * @param headers The list to include in the copy.
+ * @param names The list to include in the copy.
* @return A new list object.
*/
- public RequestHeaders subset(String...headers) {
- return new RequestHeaders(this, headers);
+ public RequestHeaders subset(String...names) {
+ return new RequestHeaders(this, names);
}
//-----------------------------------------------------------------------------------------------------------------
diff --git
a/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/httppart/RequestQueryParams.java
b/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/httppart/RequestQueryParams.java
index 4b333182e..47f9dccb5 100644
---
a/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/httppart/RequestQueryParams.java
+++
b/juneau-rest/juneau-rest-server/src/main/java/org/apache/juneau/rest/httppart/RequestQueryParams.java
@@ -18,8 +18,10 @@ import static org.apache.juneau.internal.CollectionUtils.*;
import static org.apache.juneau.internal.StringUtils.*;
import static org.apache.juneau.httppart.HttpPartType.*;
import static java.util.Optional.*;
+import static java.util.stream.Collectors.*;
import java.util.*;
+import java.util.stream.*;
import org.apache.http.*;
import org.apache.juneau.httppart.*;
@@ -72,11 +74,10 @@ import org.apache.juneau.http.part.*;
* <ul class='spaced-list'>
* <li>Methods for retrieving query parameters:
* <ul class='javatreec'>
- * <li class='jm'>{@link
RequestQueryParams#contains(String...) contains(String...)}
+ * <li class='jm'>{@link
RequestQueryParams#contains(String) contains(String)}
* <li class='jm'>{@link
RequestQueryParams#containsAny(String...) containsAny(String...)}
* <li class='jm'>{@link RequestQueryParams#get(Class)
get(Class)}
* <li class='jm'>{@link RequestQueryParams#get(String)
get(String)}
- * <li class='jm'>{@link RequestQueryParams#getAll()
getAll()}
* <li class='jm'>{@link RequestQueryParams#getAll(String)
getAll(String)}
* <li class='jm'>{@link
RequestQueryParams#getFirst(String) getFirst(String)}
* <li class='jm'>{@link
RequestQueryParams#getLast(String) getLast(String)}
@@ -92,8 +93,7 @@ import org.apache.juneau.http.part.*;
* <li class='jm'>{@link
RequestQueryParams#addDefault(List) addDefault(List)}
* <li class='jm'>{@link
RequestQueryParams#addDefault(NameValuePair...) addDefault(NameValuePair...)}
* <li class='jm'>{@link
RequestQueryParams#addDefault(String,String) addDefault(String,String)}
- * <li class='jm'>{@link
RequestQueryParams#remove(NameValuePair...) remove(NameValuePair...)}
- * <li class='jm'>{@link
RequestQueryParams#remove(String...) remove(String...)}
+ * <li class='jm'>{@link RequestQueryParams#remove(String)
remove(String)}
* <li class='jm'>{@link
RequestQueryParams#set(NameValuePair...) set(NameValuePair...)}
* <li class='jm'>{@link
RequestQueryParams#set(String,Object) set(String,Object)}
* </ul>
@@ -117,16 +117,15 @@ import org.apache.juneau.http.part.*;
* <li class='extlink'>{@source}
* </ul>
*/
-public class RequestQueryParams {
+public class RequestQueryParams extends ArrayList<RequestQueryParam> {
+
+ private static final long serialVersionUID = 1L;
private final RestRequest req;
- private final boolean caseSensitive;
+ private boolean caseSensitive;
private final VarResolverSession vs;
private HttpPartParserSession parser;
- private List<RequestQueryParam> list = new LinkedList<>();
- private Map<String,List<RequestQueryParam>> map = new TreeMap<>();
-
/**
* Constructor.
*
@@ -141,8 +140,6 @@ public class RequestQueryParams {
for (Map.Entry<String,String[]> e : query.entrySet()) {
String name = e.getKey();
- String key = key(name);
- List<RequestQueryParam> l = list();
String[] values = e.getValue();
if (values == null)
@@ -154,12 +151,11 @@ public class RequestQueryParams {
if (values.length == 1 && values[0] == null)
values[0] = "";
- for (String value : values) {
- RequestQueryParam p = new
RequestQueryParam(req, name, value);
- list.add(p);
- l.add(p);
- }
- map.put(key, l);
+ if (values.length == 0)
+ values = new String[]{null};
+
+ for (String value : values)
+ add(new RequestQueryParam(req, name, value));
}
}
@@ -170,9 +166,20 @@ public class RequestQueryParams {
req = copyFrom.req;
caseSensitive = copyFrom.caseSensitive;
parser = copyFrom.parser;
- list.addAll(copyFrom.list);
- map.putAll(copyFrom.map);
+ addAll(copyFrom);
+ vs = copyFrom.vs;
+ }
+
+ /**
+ * Subset constructor.
+ */
+ private RequestQueryParams(RequestQueryParams copyFrom, String...names)
{
+ this.req = copyFrom.req;
+ caseSensitive = copyFrom.caseSensitive;
+ parser = copyFrom.parser;
vs = copyFrom.vs;
+ for (String n : names)
+ copyFrom.stream().filter(x -> eq(x.getName(),
n)).forEach(x -> add(x));
}
/**
@@ -183,8 +190,18 @@ public class RequestQueryParams {
*/
public RequestQueryParams parser(HttpPartParserSession value) {
this.parser = value;
- for (RequestQueryParam p : list)
- p.parser(parser);
+ forEach(x -> x.parser(parser));
+ return this;
+ }
+
+ /**
+ * Sets case sensitivity for names in this list.
+ *
+ * @param value The new value for this setting.
+ * @return This object (for method chaining).
+ */
+ public RequestQueryParams caseSensitive(boolean value) {
+ this.caseSensitive = value;
return this;
}
@@ -206,15 +223,11 @@ public class RequestQueryParams {
public RequestQueryParams addDefault(List<? extends NameValuePair>
pairs) {
for (NameValuePair p : pairs) {
String name = p.getName();
- String key = key(name);
- List<RequestQueryParam> l = map.get(key);
- boolean hasAllBlanks = l != null &&
l.stream().allMatch(x -> StringUtils.isEmpty(x.getValue()));
- if (l == null || hasAllBlanks) {
- if (hasAllBlanks)
- list.removeAll(l);
- RequestQueryParam x = new
RequestQueryParam(req, name, vs.resolve(p.getValue()));
- list.add(x);
- map.put(key, list(x));
+ Stream<RequestQueryParam> l = stream(name);
+ boolean hasAllBlanks = l.allMatch(x ->
StringUtils.isEmpty(x.getValue()));
+ if (hasAllBlanks) {
+ removeAll(getAll(name));
+ add(new RequestQueryParam(req, name,
vs.resolve(p.getValue())));
}
}
return this;
@@ -246,64 +259,6 @@ public class RequestQueryParams {
return addDefault(BasicStringPart.of(name, value));
}
- /**
- * Returns all the parameters with the specified name.
- *
- * @param name The parameter name.
- * @return The list of all parameters with the specified name, or an
empty list if none are found.
- */
- public List<RequestQueryParam> getAll(String name) {
- assertArgNotNull("name", name);
- List<RequestQueryParam> l = map.get(key(name));
- return l == null ? emptyList() : unmodifiable(l);
- }
-
- /**
- * Returns all the parameters on this request.
- *
- * @return All the parameters on this request.
- */
- public List<RequestQueryParam> getAll() {
- return unmodifiable(list);
- }
-
- /**
- * Returns <jk>true</jk> if the parameters with the specified names are
present.
- *
- * @param names The parameter names. Must not be <jk>null</jk>.
- * @return <jk>true</jk> if the parameters with the specified names are
present.
- */
- public boolean contains(String...names) {
- assertArgNotNull("names", names);
- for (String n : names)
- if (! map.containsKey(key(n)))
- return false;
- return true;
- }
-
- /**
- * Returns <jk>true</jk> if the parameter with any of the specified
names are present.
- *
- * @param names The parameter names. Must not be <jk>null</jk>.
- * @return <jk>true</jk> if the parameter with any of the specified
names are present.
- */
- public boolean containsAny(String...names) {
- assertArgNotNull("names", names);
- for (String n : names)
- if (map.containsKey(key(n)))
- return true;
- return false;
- }
-
- /**
- * Returns <jk>true</jk> if these parameters are empty.
- *
- * @return <jk>true</jk> if these parameters are empty.
- */
- public boolean isEmpty() {
- return list.isEmpty();
- }
-
/**
* Adds a parameter value.
*
@@ -317,13 +272,7 @@ public class RequestQueryParams {
*/
public RequestQueryParams add(String name, Object value) {
assertArgNotNull("name", name);
- String key = key(name);
- RequestQueryParam h = new RequestQueryParam(req, name,
stringify(value)).parser(parser);
- if (map.containsKey(key))
- map.get(key).add(h);
- else
- map.put(key, list(h));
- list.add(h);
+ add(new RequestQueryParam(req, name,
stringify(value)).parser(parser));
return this;
}
@@ -339,10 +288,9 @@ public class RequestQueryParams {
*/
public RequestQueryParams add(NameValuePair...parameters) {
assertArgNotNull("parameters", parameters);
- for (NameValuePair p : parameters) {
+ for (NameValuePair p : parameters)
if (p != null)
add(p.getName(), p.getValue());
- }
return this;
}
@@ -362,12 +310,7 @@ public class RequestQueryParams {
*/
public RequestQueryParams set(String name, Object value) {
assertArgNotNull("name", name);
- String key = key(name);
- RequestQueryParam p = new RequestQueryParam(req, name,
stringify(value)).parser(parser);
- if (map.containsKey(key))
-
list.removeIf(x->caseSensitive?x.getName().equals(name):x.getName().equalsIgnoreCase(name));
- list.add(p);
- map.put(key, list(p));
+ set(new RequestQueryParam(req, name,
stringify(value)).parser(parser));
return this;
}
@@ -397,33 +340,92 @@ public class RequestQueryParams {
* @param name The parameter names. Must not be <jk>null</jk>.
* @return This object.
*/
- public RequestQueryParams remove(String...name) {
+ public RequestQueryParams remove(String name) {
assertArgNotNull("name", name);
- for (String n : name) {
- String key = key(n);
- if (map.containsKey(key))
- list.removeAll(map.get(key));
- map.remove(key);
- }
+ removeIf(x -> eq(x.getName(), name));
return this;
}
/**
- * Remove parameters.
+ * Returns a copy of this object but only with the specified param
names copied.
*
- * @param parameters The parameters to remove. Must not be
<jk>null</jk>.
- * @return This object.
+ * @param names The list to include in the copy.
+ * @return A new list object.
*/
- public RequestQueryParams remove(NameValuePair...parameters) {
- for (NameValuePair p : parameters)
- remove(p.getName());
- return this;
+ public RequestQueryParams subset(String...names) {
+ return new RequestQueryParams(this, names);
}
//-----------------------------------------------------------------------------------------------------------------
// Convenience getters.
//-----------------------------------------------------------------------------------------------------------------
+ /**
+ * Returns <jk>true</jk> if the parameters with the specified name is
present.
+ *
+ * @param name The parameter name. Must not be <jk>null</jk>.
+ * @return <jk>true</jk> if the parameters with the specified names are
present.
+ */
+ public boolean contains(String name) {
+ return stream(name).findAny().isPresent();
+ }
+
+ /**
+ * Returns <jk>true</jk> if the parameter with any of the specified
names are present.
+ *
+ * @param names The parameter names. Must not be <jk>null</jk>.
+ * @return <jk>true</jk> if the parameter with any of the specified
names are present.
+ */
+ public boolean containsAny(String...names) {
+ assertArgNotNull("names", names);
+ for (String n : names)
+ if (stream(n).findAny().isPresent())
+ return true;
+ return false;
+ }
+
+ /**
+ * Returns all the parameters with the specified name.
+ *
+ * @param name The parameter name.
+ * @return The list of all parameters with the specified name, or an
empty list if none are found.
+ */
+ public List<RequestQueryParam> getAll(String name) {
+ return stream(name).collect(toList());
+ }
+
+ /**
+ * Returns all headers with the specified name.
+ *
+ * @param name The header name.
+ * @return The stream of all headers with matching names. Never
<jk>null</jk>.
+ */
+ public Stream<RequestQueryParam> stream(String name) {
+ return stream().filter(x -> eq(x.getName(), name));
+ }
+
+ /**
+ * Returns all headers in sorted order.
+ *
+ * @return The stream of all headers in sorted order.
+ */
+ public Stream<RequestQueryParam> getSorted() {
+ Comparator<RequestQueryParam> x;
+ if (caseSensitive)
+ x = (x1,x2) -> x1.getName().compareTo(x2.getName());
+ else
+ x = (x1,x2) ->
String.CASE_INSENSITIVE_ORDER.compare(x1.getName(), x2.getName());
+ return stream().sorted(x);
+ }
+
+ /**
+ * Returns all the unique header names in this list.
+ * @return The list of all unique header names in this list.
+ */
+ public List<String> getNames() {
+ return stream().map(x -> x.getName()).map(x -> caseSensitive ?
x : x.toLowerCase()).distinct().collect(toList());
+ }
+
/**
* Returns the first parameter with the specified name.
*
@@ -436,8 +438,7 @@ public class RequestQueryParams {
*/
public RequestQueryParam getFirst(String name) {
assertArgNotNull("name", name);
- List<RequestQueryParam> l = map.get(key(name));
- return (l == null || l.isEmpty() ? new RequestQueryParam(req,
name, null).parser(parser) : l.get(0));
+ return stream(name).findFirst().orElseGet(()->new
RequestQueryParam(req, name, null).parser(parser));
}
/**
@@ -452,21 +453,33 @@ public class RequestQueryParams {
*/
public RequestQueryParam getLast(String name) {
assertArgNotNull("name", name);
- List<RequestQueryParam> l = map.get(key(name));
- return (l == null || l.isEmpty() ? new RequestQueryParam(req,
name, null).parser(parser) : l.get(l.size()-1));
+ Value<RequestQueryParam> v = Value.empty();
+ stream(name).forEach(x -> v.set(x));
+ return v.orElseGet(() -> new RequestQueryParam(req, name,
null).parser(parser));
}
/**
- * Returns the last parameter with the specified name.
+ * Returns the condensed header with the specified name.
*
* <p>
- * This is equivalent to {@link #getLast(String)}.
+ * If multiple headers are present, they will be combined into a single
comma-delimited list.
*
- * @param name The parameter name.
- * @return The parameter value, or {@link Optional#empty()} if it
doesn't exist.
+ * @param name The header name.
+ * @return The header, never <jk>null</jk>.
*/
public RequestQueryParam get(String name) {
- return getLast(name);
+ List<RequestQueryParam> l = getAll(name);
+ if (l.isEmpty())
+ return new RequestQueryParam(req, name,
null).parser(parser);
+ if (l.size() == 1)
+ return l.get(0);
+ StringBuilder sb = new StringBuilder(128);
+ for (int i = 0, j = l.size(); i < j; i++) {
+ if (i > 0)
+ sb.append(", ");
+ sb.append(l.get(i).getValue());
+ }
+ return new RequestQueryParam(req, name,
sb.toString()).parser(parser);
}
/**
@@ -500,7 +513,7 @@ public class RequestQueryParams {
*/
public String asQueryString() {
StringBuilder sb = new StringBuilder();
- for (RequestQueryParam e : list) {
+ for (RequestQueryParam e : this) {
if (sb.length() > 0)
sb.append("&");
sb.append(urlEncode(e.getName())).append('=').append(urlEncode(e.getValue()));
@@ -557,31 +570,11 @@ public class RequestQueryParams {
return
ofNullable(PageArgs.create(get("p").asInteger().orElse(null),
get("l").asInteger().orElse(null)));
}
- /**
- * Converts the parameters to a readable string.
- *
- * @param sorted Sort the parameters by name.
- * @return A JSON string containing the contents of the parameters.
- */
- public String toString(boolean sorted) {
- JsonMap m = JsonMap.create();
- if (sorted) {
- for (List<RequestQueryParam> p1 : map.values())
- for (RequestQueryParam p2 : p1)
- m.append(p2.getName(), p2.getValue());
- } else {
- for (RequestQueryParam p : list)
- m.append(p.getName(), p.getValue());
- }
- return m.toString();
- }
-
- private String key(String name) {
- return caseSensitive ? name : name.toLowerCase();
- }
-
@Override /* Object */
public String toString() {
- return toString(false);
+ JsonMap m = new JsonMap();
+ for (String n : getNames())
+ m.put(n, get(n).asString().orElse(null));
+ return m.asJson();
}
}
diff --git
a/juneau-utest/src/test/java/org/apache/juneau/http/remote/Remote_QueryAnnotation_Test.java
b/juneau-utest/src/test/java/org/apache/juneau/http/remote/Remote_QueryAnnotation_Test.java
index 838bed870..43edf0866 100644
---
a/juneau-utest/src/test/java/org/apache/juneau/http/remote/Remote_QueryAnnotation_Test.java
+++
b/juneau-utest/src/test/java/org/apache/juneau/http/remote/Remote_QueryAnnotation_Test.java
@@ -698,7 +698,7 @@ public class Remote_QueryAnnotation_Test {
public static class K {
@RestOp
public String get(RestRequest req) throws Exception {
- return req.getQueryParams().toString(true);
+ return req.getQueryParams().toString();
}
}
@@ -845,7 +845,7 @@ public class Remote_QueryAnnotation_Test {
@Test
public void k04_requestBean_charSequence() throws Exception {
K4 x = remote(K.class,K4.class);
- assertEquals("{baz:'qux',foo:'bar'}",x.get(new K4a()));
+ assertEquals("{foo:'bar',baz:'qux'}",x.get(new K4a()));
}
//-----------------------------------------------------------------------------------------------------------------
@@ -867,7 +867,7 @@ public class Remote_QueryAnnotation_Test {
@Test
public void k05_requestBean_reader() throws Exception {
K5 x = remote(K.class,K5.class);
- assertEquals("{baz:'qux',foo:'bar'}",x.get(new K5a()));
+ assertEquals("{foo:'bar',baz:'qux'}",x.get(new K5a()));
}
//-----------------------------------------------------------------------------------------------------------------