Package com.google.collide.client.collaboration.cc

Source Code of com.google.collide.client.collaboration.cc.GenericOperationChannel

// Copyright 2012 Google Inc. All Rights Reserved.
//
// 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 com.google.collide.client.collaboration.cc;

import com.google.collide.client.collaboration.cc.TransformQueue.Transformer;
import com.google.collide.client.util.logging.Log;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;

import org.waveprotocol.wave.client.scheduler.Scheduler;
import org.waveprotocol.wave.client.scheduler.TimerService;
import org.waveprotocol.wave.model.util.FuzzingBackOffGenerator;

import java.util.EnumSet;
import java.util.List;

/*
* Forked from Wave. Currently, the only changes are exposing some otherwise
* internal state (such as queuedClientOps).
*
* TODO: Make it fit in with Collide: no JRE collections, reduce Wave
* dependencies
*/
/**
* Service that handles transportation and transforming of client and server
* operations.
*
* Design document:
* http://goto.google.com/generic-operation-channel/
*
*
* @param <M> Mutation type.
*/
public class GenericOperationChannel<M> implements RevisionProvider {

  /** Whether debug/info logging is enabled */
  private static final boolean LOG_ENABLED = false;
 
  /**
   * Provides a channel for incoming operations.
   */
  public interface ReceiveOpChannel<M> {
    public interface Listener<M> {
      void onMessage(int resultingRevision, String sid, M mutation);
      void onError(Throwable e);
    }

    void connect(int revision, Listener<M> listener);
    void disconnect();
  }

  /**
   * Provides a service to send outgoing operations and synchronize the
   * concurrent object with the server.
   */
  public interface SendOpService<M> {
    public interface Callback {
      void onSuccess(int appliedRevision);
      void onConnectionError(Throwable e);
      void onFatalError(Throwable e);
    }

    /**
     * Submit operations at the given revision.
     *
     * <p>Will be called back with the revision at which the ops were applied.
     */
    void submitOperations(int revision, List<M> operations, Callback callback);

    /**
     * Lightweight request to get the current revision of the object without
     * submitting operations (somewhat equivalent to applying no operations).
     *
     * <p>Useful for synchronizing with the channel for retrying/reconnection.
     */
    void requestRevision(Callback callback);

    /**
     * Called to indicate that the channel is no longer interested in being
     * notified via the given callback object, so implementations may optionally
     * discard the associated request state and callback. It is safe to do
     * nothing with this method.
     *
     * @param callback
     */
    void callbackNotNeeded(Callback callback);
  }

  /**
   * Notifies when operations and acknowledgments come in. The values passed to
   * the methods can be used to reconstruct the exact server history.
   *
   * <p>WARNING: The server history ops cannot be applied to local client
   * state, because they have not been transformed properly. Server history
   * ops are for other uses. To get the server ops to apply locally, use
   * {@link GenericOperationChannel#receive()}
   */
  public interface Listener<M> {
    /**
     * A remote op has been received. Do not use the parameter to apply to local
     * state, instead use {@link GenericOperationChannel#receive()}.
     *
     * @param serverHistoryOp the operation as it appears in the server history
     *        (do not apply this to local state).
     * @param pretransformedQueuedClientOps
     * @param pretransformedUnackedClientOps
     */
    void onRemoteOp(M serverHistoryOp, List<M> pretransformedUnackedClientOps,
        List<M> pretransformedQueuedClientOps);

    /**
     * A local op is acknowledged as applied at this point in the server history
     * op stream.
     *
     * @param serverHistoryOp the operation as it appears in the server history,
     *          not necessarily as it was when passed into the channel.
     * @param clean true if the channel is now clean.
     */
    void onAck(M serverHistoryOp, boolean clean);

    /**
     * Called when some unrecoverable problem occurs.
     */
    void onError(Throwable e);
  }

  private final Scheduler.Task maybeSendTask = new Scheduler.Task() {
    @Override public void execute() {
      maybeSend();
    }
  };

  private final Scheduler.Task delayedResyncTask = new Scheduler.Task() {
    @Override public void execute() {
      doResync();
    }
  };

  private final ReceiveOpChannel.Listener<M> receiveListener = new ReceiveOpChannel.Listener<M>() {
    @Override public void onMessage(int resultingRevision, String sid, M operation) {
      if (!isConnected()) {
        return;
      }

      if (LOG_ENABLED) {
        Log.debug(getClass(), "my sid=", sessionId, ", incoming sid=", sid);
      }
      if (sessionId.equals(sid)) {
        onAckOwnOperation(resultingRevision, operation);
      } else {
        onIncomingOperation(resultingRevision, operation);
      }

      maybeSynced();
    }

    @Override public void onError(Throwable e) {
      if (!isConnected()) {
        return;
      }
      listener.onError(e);
    }
  };

  /**
   * To reduce the risk of the channel behaving unpredictably due to poor
   * external implementations, we handle discarding callbacks internally and
   * merely hint to the service that it may be discarded.
   */
  abstract class DiscardableCallback implements SendOpService.Callback {
    private boolean discarded = false;

    void discard() {
      if (discarded) {
        return;
      }
      discarded = true;
      submitService.callbackNotNeeded(this);
    }

    @Override
    public void onConnectionError(Throwable e) {
      if (!isConnected()) {
        return;
      }

      if (discarded) {
        Log.warn(getClass(), "Ignoring failure, ", e);
        return;
      }
      discarded = true;

      Log.warn(getClass(), "Retryable failure, will resync.", e);
      delayResync();
    }

    @Override
    public void onSuccess(int appliedRevision) {
      if (!isConnected()) {
        return;
      }

      if (discarded) {
        if (LOG_ENABLED) {
          Log.info(getClass(), "Ignoring success @", appliedRevision);
        }
        return;
      }
      discarded = true;

      success(appliedRevision);
    }

    @Override
    public final void onFatalError(Throwable e) {
      fail(e);
    }

    abstract void success(int appliedRevision);
  }

  enum State {
    /**
     * Cannot send ops in this state. All states can transition here if either
     * explicitly requested, or if there is a permanent failure.
     */
    UNINITIALISED,

    /**
     * No unacked ops. There may be queued ops though.
     */
    ALL_ACKED,

    /**
     * Waiting for an ack for sent ops. Will transition back to ALL_ACKED if
     * successful, or to DELAY_RESYNC if there is a retryable failure.
     */
    WAITING_ACK,

    /**
     * Waiting to attempt a resync/reconnect (delay can be large due to
     * exponential backoff). Will transition to WAITING_SYNC when the delay is
     * up and we send off the version request, or to ALL_ACKED if all ops get
     * acked while waiting.
     */
    DELAY_RESYNC,

    /**
     * Waiting for our version sync. If it turns out that all ops get acked down
     * the channel in the meantime, we can return to ALL_ACKED. Otherwise, we
     * can resend and go to WAITING_ACK. If there is a retryable failure, we
     * will go to DELAY_RESYNC
     */
    WAITING_SYNC;

    private EnumSet<State> to;
    static {
      UNINITIALISED.transitionsTo(ALL_ACKED);
      ALL_ACKED.transitionsTo(WAITING_ACK);
      WAITING_ACK.transitionsTo(ALL_ACKED, DELAY_RESYNC);
      DELAY_RESYNC.transitionsTo(ALL_ACKED, WAITING_SYNC);
      WAITING_SYNC.transitionsTo(ALL_ACKED, WAITING_ACK, DELAY_RESYNC);
    }

    private void transitionsTo(State... validTransitionStates) {
      // Also, everything may transition to UNINITIALISED
      to = EnumSet.of(UNINITIALISED, validTransitionStates);
    }
  }

  private final FuzzingBackOffGenerator backoffGenerator;
  private final TimerService scheduler;
  private final ReceiveOpChannel<M> channel;
  private final SendOpService<M> submitService;
  private final Listener<M> listener;

  // State variables
  private State state = State.UNINITIALISED;
  private final TransformQueue<M> queue;
  private String sessionId;
  private int retryVersion = -1;
  private DiscardableCallback submitCallback; // mutable to discard out of date ones
  private DiscardableCallback versionCallback;

  public GenericOperationChannel(TimerService scheduler, Transformer<M> transformer,
      ReceiveOpChannel<M> channel, SendOpService<M> submitService,
      Listener<M> listener) {
    this(new FuzzingBackOffGenerator(1500, 1800 * 1000, 0.5),
        scheduler, transformer, channel, submitService, listener);
  }

  public GenericOperationChannel(FuzzingBackOffGenerator generator, TimerService scheduler,
      Transformer<M> transformer, ReceiveOpChannel<M> channel, SendOpService<M> submitService,
      Listener<M> listener) {
    this.backoffGenerator = generator;
    this.scheduler = scheduler;
    this.queue = new TransformQueue<M>(transformer);
    this.channel = channel;
    this.submitService = submitService;
    this.listener = listener;
  }

  public boolean isConnected() {
    // UNINITIALISED implies sessionId == null.
    assert state != State.UNINITIALISED || sessionId == null;
    return sessionId != null;
  }

  @Override
  public int revision() {
    checkConnected();
    return queue.revision();
  }

  public void connect(int revision, String sessionId) {
    Preconditions.checkState(!isConnected(), "Already connected");
    Preconditions.checkNotNull(sessionId, "Null sessionId");
    Preconditions.checkArgument(revision >= 0, "Invalid revision, %s", revision);
    this.sessionId = sessionId;
    channel.connect(revision, receiveListener);
    queue.init(revision);
    setState(State.ALL_ACKED);
  }

  public void disconnect() {
    checkConnected();
    channel.disconnect();
    sessionId = null;
    scheduler.cancel(maybeSendTask);
    setState(State.UNINITIALISED);
  }

  /**
   * @return true if there are no queued or unacknowledged ops
   */
  public boolean isClean() {
    checkConnected();
    boolean ret = !queue.hasQueuedClientOps() && !queue.hasUnacknowledgedClientOps();
    // isClean() implies ALL_ACKED
    assert !ret || state == State.ALL_ACKED;
    return ret;
  }

  public void send(M operation) {
    checkConnected();
    queue.clientOp(operation);
    // Defer the send to allow multiple ops to batch up, and
    // to avoid waiting for the browser's network stack in case
    // we are in a time critical piece of code. Note, we could even
    // go further and avoid doing the transform inside the queue.
    if (!queue.hasUnacknowledgedClientOps()) {
      assert state == State.ALL_ACKED;
      scheduler.schedule(maybeSendTask);
    }
  }

  public M peek() {
    checkConnected();
    return queue.hasServerOp() ? queue.peekServerOp() : null;
  }

  public M receive() {
    checkConnected();
    return queue.hasServerOp() ? queue.removeServerOp() : null;
  }

  public int getQueuedClientOpCount() {
    return queue.getQueuedClientOpCount();
  }

  public int getUnacknowledgedClientOpCount() {
    return queue.getUnacknowledgedClientOpCount();
  }

  /**
   * Brings the state variable to the given value.
   *
   * <p>Verifies that other member variables are are in the correct state.
   */
  private void setState(State newState) {
    // Check transitioning from valid old state
    State oldState = state;
    assert oldState.to.contains(newState)
        : "Invalid state transition " + oldState + " -> " + newState;

    // Check consistency of variables with new state
    checkState(newState);

    state = newState;
  }

  private void checkState(State newState) {

    switch (newState) {
      case UNINITIALISED:
        assert sessionId == null;
        break;
      case ALL_ACKED:
        assert sessionId != null;
        assert queue.revision() >= 0;
        assert isDiscarded(submitCallback);
        assert isDiscarded(versionCallback);
        assert retryVersion == -1;
        assert !queue.hasUnacknowledgedClientOps();
        assert !scheduler.isScheduled(delayedResyncTask);
        break;
      case WAITING_ACK:
        assert !isDiscarded(submitCallback);
        assert isDiscarded(versionCallback);
        assert retryVersion == -1;
        assert !scheduler.isScheduled(maybeSendTask);
        assert !scheduler.isScheduled(delayedResyncTask);
        break;
      case DELAY_RESYNC:
        assert isDiscarded(submitCallback);
        assert isDiscarded(versionCallback);
        assert retryVersion == -1;
        assert !scheduler.isScheduled(maybeSendTask);
        assert scheduler.isScheduled(delayedResyncTask);
        break;
      case WAITING_SYNC:
        assert isDiscarded(submitCallback);
        assert !isDiscarded(versionCallback);
        assert !scheduler.isScheduled(maybeSendTask);
        assert !scheduler.isScheduled(delayedResyncTask);
        break;
      default:
        throw new AssertionError("State " + state + " not implemented");
    }
  }

  private void delayResync() {
    scheduler.scheduleDelayed(delayedResyncTask, backoffGenerator.next().targetDelay);
    setState(State.DELAY_RESYNC);
  }

  private void doResync() {
    versionCallback = new DiscardableCallback() {
      @Override public void success(int appliedRevision) {
        if (LOG_ENABLED) {
          Log.info(getClass(), "version callback returned @", appliedRevision);
        }
        retryVersion = appliedRevision;
        maybeSynced();
      }
    };
    submitService.requestRevision(versionCallback);
    setState(State.WAITING_SYNC);
  }

  private void maybeSend() {
    if (queue.hasUnacknowledgedClientOps()) {
      if (LOG_ENABLED) {
        Log.info(getClass(), state, ", Has ", queue.unackedClientOps.size(), " unacked...");
      }
      return;
    }

    queue.pushQueuedOpsToUnacked();
    sendUnackedOps();
  }

  /**
   * Sends unacknowledged ops and transitions to the WAITING_ACK state
   */
  private void sendUnackedOps() {
    List<M> ops = queue.unackedClientOps();
    assert ops.size() > 0;
    if (LOG_ENABLED) {
      Log.info(getClass(), "Sending ", ops.size(), " ops @", queue.revision());
    }
    submitCallback = new DiscardableCallback() {
      @Override void success(int appliedRevision) {
        maybeEagerlyHandleAck(appliedRevision);
      }
    };

    submitService.submitOperations(queue.revision(), ops, submitCallback);
    setState(State.WAITING_ACK);
  }

  private void onIncomingOperation(int revision, M operation) {
    if (LOG_ENABLED) {
      Log.info(getClass(), "Incoming ", revision, " ", state);
    }

    List<M> pretransformedUnackedClientOps = queue.unackedClientOps();
    List<M> pretransformedQueuedClientOps = queue.queuedClientOps();

    queue.serverOp(revision, operation);

    listener.onRemoteOp(operation, pretransformedUnackedClientOps, pretransformedQueuedClientOps);
  }

  private void onAckOwnOperation(int resultingRevision, M ackedOp) {
    boolean alreadyAckedByXhr = queue.expectedAck(resultingRevision);
    if (alreadyAckedByXhr) {
      // Nothing to do, just receiving expected operations that we've
      // already handled by the optimization in maybeEagerlyHandleAck()
      return;
    }

    boolean allAcked = queue.ackClientOp(resultingRevision);
    if (LOG_ENABLED) {
      Log.info(getClass(), "Ack @", resultingRevision, ", ",
          queue.unackedClientOps.size(), " ops remaining");
    }

    // If we have more ops to send and no unacknowledged ops,
    // then schedule a send.
    if (allAcked) {
      allAcked();
    }

    listener.onAck(ackedOp, isClean());
  }

  private void maybeEagerlyHandleAck(int appliedRevision) {
    List<M> ownOps = queue.ackOpsIfVersionMatches(appliedRevision);
    if (ownOps == null) {
      return;
    }

    if (LOG_ENABLED) {
      Log.info(getClass(), "Eagerly acked @", appliedRevision);
    }

    // Special optimization: there were no concurrent ops on the server,
    // so we don't need to wait for them or even our own ops on the channel.
    // We just throw back our own ops to our listeners as if we had
    // received them from the server (we expect they should exactly
    // match the server history we will shortly receive on the channel).

    assert !queue.hasUnacknowledgedClientOps();
    allAcked();

    boolean isClean = isClean();
    for (int i = 0; i < ownOps.size(); i++) {
      boolean isLast = i == ownOps.size() - 1;
      listener.onAck(ownOps.get(i), isClean && isLast);
    }
  }

  private void allAcked() {

    // This also counts as an early sync
    synced();

    // No point waiting for the XHR to come back, we're already acked.
    submitCallback.discard();

    setState(State.ALL_ACKED);
    if (queue.hasQueuedClientOps()) {
      scheduler.schedule(maybeSendTask);
    }
  }

  private void maybeSynced() {
    if (state == State.WAITING_SYNC && retryVersion != -1 && queue.revision() >= retryVersion) {

      // Our ping has returned.
      synced();

      if (queue.hasUnacknowledgedClientOps()) {
        // We've synced and didn't see our unacked ops, so they never made it (we
        // are not handling the case of ops that hang around on the network and
        // make it after a very long time, i.e. after a sync round trip. This
        // scenario most likely extremely rare).

        // Send the unacked ops again.
        sendUnackedOps();
      }
    }
  }

  /**
   * We have reached a state where we are confident we know whether any unacked
   * ops made it to the server.
   */
  private void synced() {
    if (LOG_ENABLED) {
      Log.info(getClass(), "synced @", queue.revision());
    }

    retryVersion = -1;
    scheduler.cancel(delayedResyncTask);
    backoffGenerator.reset();

    if (versionCallback != null) {
      versionCallback.discard();
    }
  }

  private void checkConnected() {
    Preconditions.checkState(isConnected(), "Not connected");
  }

  private boolean isDiscarded(DiscardableCallback c) {
    return c == null || c.discarded;
  }

  private void fail(Throwable e) {

    Log.warn(getClass(), "channel.fail()");
    if (!isConnected()) {
      Log.warn(getClass(), "not connected");
      return;
    }

    Log.warn(getClass(), "Permanent failure");

    disconnect();

    listener.onError(e);
  }

  @VisibleForTesting State getState() {
    return state;
  }
}
TOP

Related Classes of com.google.collide.client.collaboration.cc.GenericOperationChannel

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.