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

org.atmosphere.stomp.handler.StompSendActionAtmosphereHandler Maven / Gradle / Ivy

/*
 * Copyright 2014 Jeanfrancois Arcand
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
 * use this file except in compliance with the License. You may obtain a copy of
 * the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
 * License for the specific language governing permissions and limitations under
 * the License.
 */


package org.atmosphere.stomp.handler;

import org.atmosphere.config.managed.Decoder;
import org.atmosphere.config.managed.Encoder;
import org.atmosphere.config.service.Message;
import org.atmosphere.cpr.AtmosphereResource;
import org.atmosphere.cpr.AtmosphereResourceEvent;
import org.atmosphere.cpr.AtmosphereResourceHeartbeatEventListener;
import org.atmosphere.cpr.Broadcaster;
import org.atmosphere.handler.AbstractReflectorAtmosphereHandler;
import org.atmosphere.stomp.interceptor.FrameInterceptor;
import org.atmosphere.stomp.annotation.StompEndpoint;
import org.atmosphere.stomp.protocol.Action;
import org.atmosphere.stomp.protocol.Header;
import org.atmosphere.util.Utils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/**
 * 

* This handler wraps the method to be invoked when the {@link org.atmosphere.stomp.protocol.Action#SEND send} action is performed * with a STOMP frame. The frame indicates the particular {@link org.atmosphere.stomp.protocol.Header#DESTINATION destination} * which is mapped to the appropriate annotated method. *

* * @author Guillaume DROUET * @since 0.1 * @version 1.0 */ public class StompSendActionAtmosphereHandler extends AbstractReflectorAtmosphereHandler implements AtmosphereResourceHeartbeatEventListener { /** * Method signature requirement message. */ private static final String IAE_MESSAGE = String.format( "Method can expects as parameter '%s', '%s'. Otherwise it must provides decoders/encoders through the '%s' annotation", AtmosphereResource.class.getName(), String.class, Message.class.getName()); /** * The logger. */ private final Logger logger = LoggerFactory.getLogger(getClass()); /** * The object which provides the method. */ private final Object toProxy; /** * The method to invoke. */ private final Method method; /** * Provide each parameter required to invoke the method. */ private final ParamProvider[] paramProviders; /** * Optional encoder that converts {@code String} to expected parameter type. */ private final Encoder encoder; /** * The broadcaster associated to this handler. */ private final Broadcaster broadcaster; /** * The heartbeat listener. */ private final Method onHeartbeatMethod; /** *

* Creates a new instance. *

* * @param toProxy the object to proxy * @param method the method to invoke on proxy object * @param encoder encodes into expected parameter type * @param decoder converts returned type into {@code String} wrapped in text frame * @param broadcaster the broadcaster associated to the destination declared in the annotated method * @param onHeartbeatMethod the heartbeat method */ public StompSendActionAtmosphereHandler(final Object toProxy, final Method method, final Encoder encoder, final Decoder decoder, final Broadcaster broadcaster, final Method onHeartbeatMethod) { this.toProxy = toProxy; this.method = method; this.encoder = encoder; this.broadcaster = broadcaster; this.onHeartbeatMethod = onHeartbeatMethod; // Detect appropriate provider for each parameter type final Class[] paramTypes = method.getParameterTypes(); paramProviders = new ParamProvider[paramTypes.length]; for (int i = 0; i < paramTypes.length; i++) { final Class paramType = paramTypes[i]; // The atmosphere resource is just the one that sent the message if (paramType.isAssignableFrom(AtmosphereResource.class)) { paramProviders[i] = new ParamProvider() { @Override public Object getParam(final AtmosphereResource atmosphereResource) { return atmosphereResource; } }; } else if (paramType.isAssignableFrom(Broadcaster.class)) { paramProviders[i] = new ParamProvider() { @Override public Object getParam(final AtmosphereResource atmosphereResource) { return broadcaster; } }; // The string will be the raw message body } else if (paramType.isAssignableFrom(String.class)) { paramProviders[i] = new ParamProvider() { @Override public Object getParam(final AtmosphereResource atmosphereResource) { return atmosphereResource.getRequest().getAttribute(FrameInterceptor.STOMP_MESSAGE_BODY); } }; // Otherwise we use the decoder to compute the appropriate parameter type } else if (decoder != null) { paramProviders[i] = new ParamProvider() { @Override public Object getParam(final AtmosphereResource atmosphereResource) { return decoder.decode(atmosphereResource.getRequest().getAttribute(FrameInterceptor.STOMP_MESSAGE_BODY).toString()); } }; // No decoder provided, we don't know how to convert raw string into expected parameter type } else { throw new IllegalArgumentException(IAE_MESSAGE); } } } /** * {@inheritDoc} */ @Override public void onRequest(final AtmosphereResource atmosphereResource) throws IOException { try { // Compute parameters final Object[] params = new Object[paramProviders.length]; for (int i = 0; i < params.length; i++) { params[i] = paramProviders[i].getParam(atmosphereResource); } // Invoke stomp service final Object retval = method.invoke(toProxy, params); if (retval != null) { broadcaster.broadcast(encoder == null ? retval : encoder.encode(retval)); } else { // TODO: ack? } } catch (IllegalAccessException iae) { logger.warn("Failed to process class annotated {}", StompEndpoint.class.getName(), iae); } catch (InvocationTargetException ite) { logger.info("Invoked method thrown an exception", ite); // Push the error in appropriate frame final StringBuilder sb = new StringBuilder(); sb.append(Action.ERROR.toString()) .append("\n") .append(Header.MESSAGE) .append(":") .append(ite.getCause().getMessage()) .append("\n\n\n") .append(0x00); atmosphereResource.write(sb.toString()); } } /** * {@inheritDoc} */ @Override public void onHeartbeat(final AtmosphereResourceEvent event) { if (onHeartbeatMethod != null && !Utils.pollableTransport(event.getResource().transport())) { Utils.invoke(toProxy, onHeartbeatMethod, event); } } /** *

* Provides a parameter of expected type from the given {@link AtmosphereResource}. *

* * @author Guillaume DROUET * @since 0.1 * @version 1.0 */ private interface ParamProvider { /** *

* Gets the parameter. *

* * @param atmosphereResource the request resource * @return the object of expected type */ Object getParam(AtmosphereResource atmosphereResource); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy