Issue #207 - @PathParam support fixes

+ Arg.tag is now exposed for general use
+ JsrEndpointFunctions now decode (into primitives) the URI template
  exposed @PathParam static arguments
+ JsrEndpointFunctions simplified tracking of static args
+ ServerContainer.assertValidEndpoint() now validates
  added endpoints with @PathParam immediately (per spec)
  using a UriTemplate where each variable is an empty string
This commit is contained in:
Joakim Erdfelt 2016-10-13 15:58:34 -07:00
parent f790bf75d8
commit 75c5793f38
6 changed files with 202 additions and 73 deletions

View File

@ -23,14 +23,12 @@ import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.List;
import java.util.Collections;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.concurrent.Executor;
import java.util.function.BiFunction;
import java.util.function.Function;
import java.util.stream.Collectors;
import javax.websocket.ClientEndpoint;
import javax.websocket.CloseReason;
@ -122,36 +120,10 @@ public class JsrEndpointFunctions extends CommonEndpointFunctions<JsrSession>
}
}
/**
* Represents a static value (as seen from a URI PathParam)
* <p>
* The decoding of the raw String to a object occurs later,
* when the callable/sink/function is created for a method
* that needs it converted to an object.
* </p>
*/
protected static class StaticArg implements Comparable<StaticArg>
{
public final String name;
public final String value;
public StaticArg(String name, String value)
{
this.name = name;
this.value = value;
}
@Override
public int compareTo(StaticArg o)
{
return this.name.compareTo(o.name);
}
}
protected final AvailableEncoders encoders;
protected final AvailableDecoders decoders;
private final EndpointConfig endpointConfig;
private List<StaticArg> staticArgs;
private Map<String, String> staticArgs;
public JsrEndpointFunctions(Object endpoint, WebSocketPolicy policy, Executor executor,
AvailableEncoders encoders, AvailableDecoders decoders,
@ -164,11 +136,7 @@ public class JsrEndpointFunctions extends CommonEndpointFunctions<JsrSession>
if (uriParams != null)
{
this.staticArgs = new ArrayList<>();
this.staticArgs.addAll(uriParams.entrySet().stream()
.map(entry -> new StaticArg(entry.getKey(), entry.getValue()))
.sorted()
.collect(Collectors.toList()));
this.staticArgs = Collections.unmodifiableMap(uriParams);
}
}
@ -914,6 +882,7 @@ public class JsrEndpointFunctions extends CommonEndpointFunctions<JsrSession>
return;
}
// Test specific return type to ensure we have a compatible encoder for it
Class<? extends Encoder> encoderClass = encoders.getEncoderFor(returnType);
if (encoderClass == null)
{
@ -930,26 +899,16 @@ public class JsrEndpointFunctions extends CommonEndpointFunctions<JsrSession>
Object[] args = new Object[callArgs.length];
for (int i = 0; i < len; i++)
{
Object staticValue = getDecodedStaticValue(callArgs[i].getName(), callArgs[i].getType());
if (staticValue != null)
{
args[i] = staticValue;
}
String staticRawValue = staticArgs.get(callArgs[i].getTag());
args[i] = AvailableDecoders.decodePrimitive(staticRawValue, callArgs[i].getType());
}
return args;
}
private Object getDecodedStaticValue(String name, Class<?> type) throws DecodeException
{
for (StaticArg args : staticArgs)
{
if (args.name.equals(name))
{
return AvailableDecoders.decodePrimitive(args.value, type);
}
}
return null;
String value = staticArgs.get(name);
return AvailableDecoders.decodePrimitive(value, type);
}
private DynamicArgs.Builder createDynamicArgs(Arg... args)
@ -975,12 +934,9 @@ public class JsrEndpointFunctions extends CommonEndpointFunctions<JsrSession>
if (this.staticArgs != null)
{
for (StaticArg staticArg : this.staticArgs)
for (Map.Entry<String, String> entry : staticArgs.entrySet())
{
// TODO: translate from UriParam String to method param type?
// TODO: shouldn't this be the Arg seen in the method?
// TODO: use static decoder?
callArgs[idx++] = new Arg(staticArg.value.getClass()).setTag(staticArg.name);
callArgs[idx++] = new Arg(entry.getValue().getClass()).setTag(entry.getKey());
}
}
return callArgs;

View File

@ -126,7 +126,7 @@ public class JsrEndpointFunctions_OnMessage_TextStreamTest
}
@Test
public void testInvokeMessageText() throws Exception
public void testInvokeMessageStream() throws Exception
{
TrackingSocket socket = performOnMessageInvocation(new MessageStreamSocket(), (endpoint) ->
{
@ -135,5 +135,4 @@ public class JsrEndpointFunctions_OnMessage_TextStreamTest
});
socket.assertEvent("onMessage(MessageReader) = \"Hello World\"");
}
}

View File

@ -176,6 +176,14 @@ public class ServerContainer extends ClientContainer implements javax.websocket.
AvailableEncoders availableEncoders = new AvailableEncoders(config);
AvailableDecoders availableDecoders = new AvailableDecoders(config);
Map<String, String> pathParameters = new HashMap<>();
// if any pathspec has a URI Template with variables, we should include them (as empty String value)
// in the test for validity of the declared @OnMessage methods that use @PathParam annotation
for (String variable : new UriTemplatePathSpec(config.getPath()).getVariables())
{
pathParameters.put(variable, "");
}
endpointFunctions = newJsrEndpointFunction(endpoint, availableEncoders, availableDecoders, pathParameters, config);
endpointFunctions.start(); // this should trigger an exception if endpoint is invalid.
}

View File

@ -251,15 +251,6 @@ public class WebSocketServerContainerInitializer implements ServletContainerInit
ServerContainer jettyContainer = configureContext(context,jettyContext);
context.addListener(new ContextDestroyListener()); // make sure context is cleaned up when the context stops
// // Establish the DecoratedObjectFactory thread local
// // for various ServiceLoader initiated components to use.
// DecoratedObjectFactory instantiation = (DecoratedObjectFactory)context.getAttribute(DecoratedObjectFactory.ATTR);
// if (instantiation == null)
// {
// LOG.info("Using WebSocket local DecoratedObjectFactory - none found in ServletContext");
// instantiation = new DecoratedObjectFactory();
// }
if (LOG.isDebugEnabled())
{
LOG.debug("Found {} classes",c.size());

View File

@ -0,0 +1,170 @@
//
// ========================================================================
// Copyright (c) 1995-2016 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.jsr356.server;
import java.io.IOException;
import java.io.Reader;
import java.net.URI;
import java.nio.charset.StandardCharsets;
import java.util.HashMap;
import java.util.Map;
import java.util.function.Function;
import javax.websocket.ClientEndpointConfig;
import javax.websocket.EndpointConfig;
import javax.websocket.OnMessage;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.IO;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.common.function.EndpointFunctions;
import org.eclipse.jetty.websocket.common.test.DummyConnection;
import org.eclipse.jetty.websocket.jsr356.ClientContainer;
import org.eclipse.jetty.websocket.jsr356.ConfiguredEndpoint;
import org.eclipse.jetty.websocket.jsr356.JsrSession;
import org.eclipse.jetty.websocket.jsr356.client.EmptyClientEndpointConfig;
import org.eclipse.jetty.websocket.jsr356.decoders.AvailableDecoders;
import org.eclipse.jetty.websocket.jsr356.encoders.AvailableEncoders;
import org.junit.BeforeClass;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
public class JsrServerEndpointFunctions_OnMessage_TextStreamTest
{
private static ClientContainer container;
@BeforeClass
public static void initContainer()
{
container = new ClientContainer();
}
@Rule
public ExpectedException expectedException = ExpectedException.none();
private AvailableEncoders encoders;
private AvailableDecoders decoders;
private Map<String, String> uriParams = new HashMap<>();
private EndpointConfig endpointConfig;
public JsrServerEndpointFunctions_OnMessage_TextStreamTest()
{
endpointConfig = new EmptyClientEndpointConfig();
encoders = new AvailableEncoders(endpointConfig);
decoders = new AvailableDecoders(endpointConfig);
uriParams = new HashMap<>();
uriParams.put("param", "foo");
}
public JsrSession newSession(Object websocket)
{
String id = JsrServerEndpointFunctions_OnMessage_TextStreamTest.class.getSimpleName();
URI requestURI = URI.create("ws://localhost/" + id);
WebSocketPolicy policy = WebSocketPolicy.newClientPolicy();
DummyConnection connection = new DummyConnection(policy);
ClientEndpointConfig config = new EmptyClientEndpointConfig();
ConfiguredEndpoint ei = new ConfiguredEndpoint(websocket, config);
return new JsrSession(container, id, requestURI, ei, policy, connection);
}
@SuppressWarnings("Duplicates")
private TrackingSocket performOnMessageInvocation(TrackingSocket socket, Function<EndpointFunctions, Void> func) throws Exception
{
// Establish endpoint function
JsrServerEndpointFunctions endpointFunctions = new JsrServerEndpointFunctions(
socket, container.getPolicy(),
container.getExecutor(),
encoders,
decoders,
uriParams,
endpointConfig
);
endpointFunctions.start();
// This invocation is the same for all tests
endpointFunctions.onOpen(newSession(socket));
func.apply(endpointFunctions);
return socket;
}
@ServerEndpoint("/msg")
public static class MessageStreamSocket extends TrackingSocket
{
@OnMessage
public void onMessage(Reader stream)
{
try
{
String msg = IO.toString(stream);
addEvent("onMessage(%s) = \"%s\"", stream.getClass().getSimpleName(), msg);
}
catch (IOException e)
{
throw new RuntimeException(e);
}
}
}
@Test
public void testInvokeMessageStream() throws Exception
{
TrackingSocket socket = performOnMessageInvocation(new MessageStreamSocket(), (endpoint) ->
{
endpoint.onText(BufferUtil.toBuffer("Hello World", StandardCharsets.UTF_8), true);
return null;
});
socket.assertEvent("onMessage(MessageReader) = \"Hello World\"");
}
@ServerEndpoint("/msg/{param}")
public static class MessageStreamParamSocket extends TrackingSocket
{
@OnMessage
public String onMessage(Reader stream, @PathParam("param") String param) throws IOException
{
try
{
String msg = IO.toString(stream);
addEvent("onMessage(%s,%s) = \"%s\"", stream.getClass().getSimpleName(), param, msg);
return msg;
}
catch (IOException e)
{
throw new RuntimeException(e);
}
}
}
@Test
public void testInvokeMessageStreamParam() throws Exception
{
TrackingSocket socket = performOnMessageInvocation(new MessageStreamParamSocket(), (endpoint) ->
{
endpoint.onText(BufferUtil.toBuffer("Hello World", StandardCharsets.UTF_8), true);
return null;
});
socket.assertEvent("onMessage(MessageReader,foo) = \"Hello World\"");
}
}

View File

@ -29,7 +29,7 @@ public class Arg
private final Class<?> type;
private Method method;
private int index;
private Object tag;
private String tag;
private boolean required = false;
public Arg(Class<?> type)
@ -88,6 +88,11 @@ public class Arg
return type.getName();
}
public String getTag()
{
return tag;
}
public Class<?> getType()
{
return type;