All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.jetty.websocket.common.events.EventDriverFactory Maven / Gradle / Ivy

//
//  ========================================================================
//  Copyright (c) 1995-2013 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.common.events;

import java.io.InputStream;
import java.io.Reader;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.concurrent.ConcurrentHashMap;

import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.api.InvalidWebSocketException;
import org.eclipse.jetty.websocket.api.Session;
import org.eclipse.jetty.websocket.api.WebSocketListener;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketClose;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketConnect;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketError;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketFrame;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketMessage;
import org.eclipse.jetty.websocket.api.annotations.WebSocket;
import org.eclipse.jetty.websocket.api.extensions.Frame;

/**
 * Create EventDriver implementations.
 */
public class EventDriverFactory
{
    /**
     * Parameter list for @OnWebSocketMessage (Binary mode)
     */
    private static final ParamList validBinaryParams;

    /**
     * Parameter list for @OnWebSocketConnect
     */
    private static final ParamList validConnectParams;

    /**
     * Parameter list for @OnWebSocketClose
     */
    private static final ParamList validCloseParams;

    /**
     * Parameter list for @OnWebSocketError
     */
    private static final ParamList validErrorParams;

    /**
     * Parameter list for @OnWebSocketFrame
     */
    private static final ParamList validFrameParams;

    /**
     * Parameter list for @OnWebSocketMessage (Text mode)
     */
    private static final ParamList validTextParams;

    static
    {
        validConnectParams = new ParamList();
        validConnectParams.addParams(Session.class);

        validCloseParams = new ParamList();
        validCloseParams.addParams(int.class,String.class);
        validCloseParams.addParams(Session.class,int.class,String.class);

        validErrorParams = new ParamList();
        validErrorParams.addParams(Throwable.class);
        validErrorParams.addParams(Session.class,Throwable.class);

        validTextParams = new ParamList();
        validTextParams.addParams(String.class);
        validTextParams.addParams(Session.class,String.class);
        validTextParams.addParams(Reader.class);
        validTextParams.addParams(Session.class,Reader.class);

        validBinaryParams = new ParamList();
        validBinaryParams.addParams(byte[].class,int.class,int.class);
        validBinaryParams.addParams(Session.class,byte[].class,int.class,int.class);
        validBinaryParams.addParams(InputStream.class);
        validBinaryParams.addParams(Session.class,InputStream.class);

        validFrameParams = new ParamList();
        validFrameParams.addParams(Frame.class);
        validFrameParams.addParams(Session.class,Frame.class);
    }

    private ConcurrentHashMap, EventMethods> cache;
    private final WebSocketPolicy policy;

    public EventDriverFactory(WebSocketPolicy policy)
    {
        this.policy = policy;
        this.cache = new ConcurrentHashMap<>();
    }

    private void assertIsPublicNonStatic(Method method)
    {
        int mods = method.getModifiers();
        if (!Modifier.isPublic(mods))
        {
            StringBuilder err = new StringBuilder();
            err.append("Invalid declaration of ");
            err.append(method);
            err.append(StringUtil.__LINE_SEPARATOR);

            err.append("Method modifier must be public");

            throw new InvalidWebSocketException(err.toString());
        }

        if (Modifier.isStatic(mods))
        {
            StringBuilder err = new StringBuilder();
            err.append("Invalid declaration of ");
            err.append(method);
            err.append(StringUtil.__LINE_SEPARATOR);

            err.append("Method modifier may not be static");

            throw new InvalidWebSocketException(err.toString());
        }
    }

    private void assertIsReturn(Method method, Class type)
    {
        if (!type.equals(method.getReturnType()))
        {
            StringBuilder err = new StringBuilder();
            err.append("Invalid declaration of ");
            err.append(method);
            err.append(StringUtil.__LINE_SEPARATOR);

            err.append("Return type must be ").append(type);

            throw new InvalidWebSocketException(err.toString());
        }
    }

    private void assertUnset(EventMethod event, Class annoClass, Method method)
    {
        if (event != null)
        {
            // Attempt to add duplicate frame type (a no-no)
            StringBuilder err = new StringBuilder();
            err.append("Duplicate @").append(annoClass.getSimpleName()).append(" declaration on ");
            err.append(method);
            err.append(StringUtil.__LINE_SEPARATOR);

            err.append("@").append(annoClass.getSimpleName()).append(" previously declared at ");
            err.append(event.getMethod());

            throw new InvalidWebSocketException(err.toString());
        }
    }

    private void assertValidSignature(Method method, Class annoClass, ParamList validParams)
    {
        assertIsPublicNonStatic(method);
        assertIsReturn(method,Void.TYPE);

        boolean valid = false;

        // validate parameters
        Class actual[] = method.getParameterTypes();
        for (Class[] params : validParams)
        {
            if (isSameParameters(actual,params))
            {
                valid = true;
                break;
            }
        }

        if (!valid)
        {
            throw InvalidSignatureException.build(method,annoClass,validParams);
        }
    }

    /**
     * Perform the basic discovery mechanism for WebSocket events from the provided pojo.
     * 
     * @param pojo
     *            the pojo to scan
     * @return the discovered event methods
     * @throws InvalidWebSocketException
     */
    private EventMethods discoverMethods(Class pojo) throws InvalidWebSocketException
    {
        WebSocket anno = pojo.getAnnotation(WebSocket.class);
        if (anno == null)
        {
            return null;
        }

        return scanAnnotatedMethods(pojo);
    }

    public EventMethods getMethods(Class pojo) throws InvalidWebSocketException
    {
        if (pojo == null)
        {
            throw new InvalidWebSocketException("Cannot get methods for null class");
        }
        if (cache.containsKey(pojo))
        {
            return cache.get(pojo);
        }
        EventMethods methods = discoverMethods(pojo);
        if (methods == null)
        {
            return null;
        }
        cache.put(pojo,methods);
        return methods;
    }

    private boolean isSameParameters(Class[] actual, Class[] params)
    {
        if (actual.length != params.length)
        {
            // skip
            return false;
        }

        int len = params.length;
        for (int i = 0; i < len; i++)
        {
            if (!actual[i].equals(params[i]))
            {
                return false; // not valid
            }
        }

        return true;
    }

    private boolean isSignatureMatch(Method method, ParamList validParams)
    {
        assertIsPublicNonStatic(method);
        assertIsReturn(method,Void.TYPE);

        // validate parameters
        Class actual[] = method.getParameterTypes();
        for (Class[] params : validParams)
        {
            if (isSameParameters(actual,params))
            {
                return true;
            }
        }

        return false;
    }

    private EventMethods scanAnnotatedMethods(Class pojo)
    {
        Class clazz = pojo;
        EventMethods events = new EventMethods(pojo);

        clazz = pojo;
        while (clazz.getAnnotation(WebSocket.class) != null)
        {
            for (Method method : clazz.getDeclaredMethods())
            {
                if (method.getAnnotation(OnWebSocketConnect.class) != null)
                {
                    assertValidSignature(method,OnWebSocketConnect.class,validConnectParams);
                    assertUnset(events.onConnect,OnWebSocketConnect.class,method);
                    events.onConnect = new EventMethod(pojo,method);
                    continue;
                }

                if (method.getAnnotation(OnWebSocketMessage.class) != null)
                {
                    if (isSignatureMatch(method,validTextParams))
                    {
                        // Text mode
                        // TODO

                        assertUnset(events.onText,OnWebSocketMessage.class,method);
                        events.onText = new EventMethod(pojo,method);
                        continue;
                    }

                    if (isSignatureMatch(method,validBinaryParams))
                    {
                        // Binary Mode
                        // TODO
                        assertUnset(events.onBinary,OnWebSocketMessage.class,method);
                        events.onBinary = new EventMethod(pojo,method);
                        continue;
                    }

                    throw InvalidSignatureException.build(method,OnWebSocketMessage.class,validTextParams,validBinaryParams);
                }

                if (method.getAnnotation(OnWebSocketClose.class) != null)
                {
                    assertValidSignature(method,OnWebSocketClose.class,validCloseParams);
                    assertUnset(events.onClose,OnWebSocketClose.class,method);
                    events.onClose = new EventMethod(pojo,method);
                    continue;
                }

                if (method.getAnnotation(OnWebSocketError.class) != null)
                {
                    assertValidSignature(method,OnWebSocketError.class,validErrorParams);
                    assertUnset(events.onError,OnWebSocketError.class,method);
                    events.onError = new EventMethod(pojo,method);
                    continue;
                }

                if (method.getAnnotation(OnWebSocketFrame.class) != null)
                {
                    assertValidSignature(method,OnWebSocketFrame.class,validFrameParams);
                    assertUnset(events.onFrame,OnWebSocketFrame.class,method);
                    events.onFrame = new EventMethod(pojo,method);
                    continue;
                }

                // Not a tagged method we are interested in, ignore
            }

            // try superclass now
            clazz = clazz.getSuperclass();
        }

        return events;
    }

    @Override
    public String toString()
    {
        return String.format("EventMethodsCache [cache.count=%d]",cache.size());
    }

    /**
     * Wrap the given WebSocket object instance in a suitable EventDriver
     * 
     * @param websocket
     *            the websocket instance to wrap. Must either implement {@link WebSocketListener} or be annotated with {@link WebSocket @WebSocket}
     * @return appropriate EventDriver for this websocket instance.
     */
    public EventDriver wrap(Object websocket)
    {
        if (websocket == null)
        {
            throw new InvalidWebSocketException("null websocket object");
        }

        if (websocket instanceof WebSocketListener)
        {
            WebSocketPolicy pojoPolicy = policy.clonePolicy();
            WebSocketListener listener = (WebSocketListener)websocket;
            return new ListenerEventDriver(pojoPolicy,listener);
        }

        EventMethods methods = getMethods(websocket.getClass());
        if (methods != null)
        {
            WebSocketPolicy pojoPolicy = policy.clonePolicy();
            return new AnnotatedEventDriver(pojoPolicy,websocket,methods);
        }

        throw new InvalidWebSocketException(websocket.getClass().getName() + " does not implement " + WebSocketListener.class.getName()
                + " or declare @WebSocket");
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy