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

com.backendless.rt.RTClientSocketIO Maven / Gradle / Ivy

The newest version!
package com.backendless.rt;

import com.backendless.BackendlessInjector;
import com.backendless.async.callback.AsyncCallback;
import com.backendless.async.callback.Fault;
import com.backendless.async.callback.Result;
import com.backendless.async.message.AsyncMessage;
import com.backendless.exceptions.BackendlessFault;
import io.socket.emitter.Emitter;
import weborb.reader.AnonymousObject;
import weborb.types.IAdaptingType;

import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedDeque;
import java.util.logging.Logger;

import static com.backendless.utils.WeborbSerializationHelper.asAdaptingType;
import static com.backendless.utils.WeborbSerializationHelper.asObject;
import static com.backendless.utils.WeborbSerializationHelper.asString;
import static com.backendless.utils.WeborbSerializationHelper.deserializeNotAdapt;
import static com.backendless.utils.WeborbSerializationHelper.serialize;


class RTClientSocketIO implements RTClient
{
  private static final Logger logger = Logger.getLogger( "RTClient" );

  private final BackendlessInjector injector = BackendlessInjector.getInstance();
  private final Map subscriptions = new ConcurrentHashMap<>();

  private final Map sentRequests = new ConcurrentHashMap<>();

  private final ConcurrentLinkedDeque methodsToSend = new ConcurrentLinkedDeque<>();

  private final SocketIOConnectionManager connectionManager;

  private Result connectCallback;
  private Fault connectErrorCallback;
  private Result disconnectCallback;
  private Result reconnectAttemptCallback;

  RTClientSocketIO()
  {
    this.connectionManager = new SocketIOConnectionManager()
    {
      @Override
      void connected()
      {
        resubscribe();
        connectCallback.handle( null );
      }

      @Override
      void reconnectAttempt( int attempt, int timeout )
      {
        reconnectAttemptCallback.handle( new ReconnectAttempt( attempt, timeout ) );
      }

      @Override
      void connectError( String error )
      {
        connectErrorCallback.handle( new BackendlessFault( error ) );
      }

      @Override
      void disconnected( String error )
      {
        disconnectCallback.handle( null );
      }

      @Override
      void subscriptionResult( Object... args )
      {
        logger.info( "subscription result " + Arrays.toString( args ) );
        handleResult( args, subscriptions, "data" );
      }

      @Override
      void invocationResult( Object... args )
      {
        logger.info( "invocation result " + Arrays.toString( args ) );
        RTRequest request = handleResult( args, sentRequests, "result" );

        if( request != null )
        {
          sentRequests.remove( request.getId() );
        }
      }
    };
  }

  private RTRequest handleResult( Object[] args, Map requestMap, String resultKey )
  {
    if( args == null || args.length < 1 )
    {
      logger.warning( "subscription result is null or empty" );
      return null;
    }

    AnonymousObject result = (AnonymousObject) deserializeNotAdapt( args[ 0 ] );

    String id = asString( result, "id" );

    logger.info( "Got result for subscription " + id );

    RTRequest request = requestMap.get( id );

    if( request == null )
    {
      logger.info( "There is no handler for subscription " + id );
      return null;
    }

    final Object error = asObject( result, "error" );

    if( error != null )
    {
      logger.severe( "got error " + error );
      final BackendlessFault fault = new BackendlessFault( error.toString() );
      injector.getHandleCarrier().deliverMessage( new AsyncMessage<>( fault, request.getCallback() ) );
      return request;
    }

    IAdaptingType data = asAdaptingType( result, resultKey );
    injector.getHandleCarrier().deliverMessage( new AsyncMessage<>( data, request.getCallback() ) );

    return request;
  }

  @Override
  public void subscribe( RTSubscription subscription )
  {
    logger.info( "try to subscribe " + subscription );
    subscriptions.put( subscription.getId(), subscription );

    if( connectionManager.get().connected() )
      subOn( subscription );
  }

  @Override
  public void unsubscribe( String subscriptionId )
  {
    logger.info( "unsubscribe for " + subscriptionId + " subscrition called" );

    if( !subscriptions.containsKey( subscriptionId ) )
    {
      logger.info( "subscriber " + subscriptionId + " is not subscribed" );
      return;
    }

    subOff( subscriptionId );
    subscriptions.remove( subscriptionId );
    logger.info( "subscription removed" );

    if( subscriptions.isEmpty() && sentRequests.isEmpty() )
    {
      connectionManager.disconnect();
    }
  }

  @Override
  public void userLoggedIn( String userToken )
  {
    if( connectionManager.isConnected() )
    {
      RTMethodRequest methodRequest = new RTMethodRequest( MethodTypes.SET_USER_TOKEN, new RTCallbackWithFault()
      {
        @Override
        public AsyncCallback usersCallback()
        {
          return null;
        }

        @Override
        public void handleResponse( IAdaptingType response )
        {
          logger.fine( "user logged in/out success" );
        }
      } );

      methodRequest.putOption( "userToken", userToken );
      invoke( methodRequest );
    }
  }

  @Override
  public void userLoggedOut()
  {
    userLoggedIn( null );
  }

  @Override
  public void invoke( RTMethodRequest methodRequest )
  {
    sentRequests.put( methodRequest.getId(), methodRequest );
    if( connectionManager.isConnected() )
    {
      metReq( methodRequest );
    }
    else
    {
      methodsToSend.addLast( methodRequest );
    }
  }

  @Override
  public void setConnectEventListener( Result callback )
  {
    connectCallback = callback;
  }

  @Override
  public void setReconnectAttemptEventListener( Result callback )
  {
    reconnectAttemptCallback = callback;
  }

  @Override
  public void setConnectErrorEventListener( Fault fault )
  {
    connectErrorCallback = fault;
  }

  @Override
  public void setDisconnectEventListener( Result callback )
  {
    disconnectCallback = callback;
  }

  @Override
  public boolean isConnected()
  {
    return connectionManager.isConnected();
  }

  @Override
  public void connect()
  {
    connectionManager.get();
  }

  @Override
  public void disconnect()
  {
    connectionManager.disconnect();
  }

  @Override
  public boolean isAvailable()
  {
    return true;
  }

  private void resubscribe()
  {
    for( RTSubscription rtSubscription : subscriptions.values() )
    {
      subOn( rtSubscription );
    }

    RTMethodRequest methodRequest = methodsToSend.poll();
    while( methodRequest != null )
    {
      metReq( methodRequest );
      methodRequest = methodsToSend.poll();
    }
  }

  private Emitter subOn( RTSubscription subscription )
  {
    final Emitter emitter = connectionManager.get().emit( "SUB_ON", new Object[] { serialize( subscription.toArgs() ) } );

    logger.info( "subOn called" );
    return emitter;
  }

  private Emitter subOff( String subscriptionId )
  {
    Map request = new HashMap<>();
    request.put( "id", subscriptionId );
    final Emitter emitter = connectionManager.get().emit( "SUB_OFF", new Object[] { serialize( request ) } );
    logger.info( "subOff called" );
    return emitter;
  }

  private Emitter metReq( RTMethodRequest methodRequest )
  {
    final Emitter emitter = connectionManager.get().emit( "MET_REQ", new Object[] { serialize( methodRequest.toArgs() ) } );

    logger.info( "subOn called" );
    return emitter;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy