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

com.google.api.gax.rpc.testing.FakeStreamingApi Maven / Gradle / Ivy

/*
 * Copyright 2017 Google LLC
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are
 * met:
 *
 *     * Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer.
 *     * Redistributions in binary form must reproduce the above
 * copyright notice, this list of conditions and the following disclaimer
 * in the documentation and/or other materials provided with the
 * distribution.
 *     * Neither the name of Google LLC nor the names of its
 * contributors may be used to endorse or promote products derived from
 * this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package com.google.api.gax.rpc.testing;

import com.google.api.core.InternalApi;
import com.google.api.gax.rpc.ApiCallContext;
import com.google.api.gax.rpc.ApiExceptionFactory;
import com.google.api.gax.rpc.ApiStreamObserver;
import com.google.api.gax.rpc.BidiStreamingCallable;
import com.google.api.gax.rpc.ClientStream;
import com.google.api.gax.rpc.ClientStreamReadyObserver;
import com.google.api.gax.rpc.ClientStreamingCallable;
import com.google.api.gax.rpc.ResponseObserver;
import com.google.api.gax.rpc.ServerStreamingCallable;
import com.google.api.gax.rpc.StatusCode.Code;
import com.google.api.gax.rpc.StreamController;
import com.google.common.base.Preconditions;
import com.google.common.collect.Queues;
import java.util.ArrayList;
import java.util.List;
import java.util.Queue;
import java.util.concurrent.CancellationException;

@InternalApi("for testing")
public class FakeStreamingApi {

  public static class BidiStreamingStashCallable
      extends BidiStreamingCallable {
    private ApiCallContext context;
    private ResponseObserver responseObserver;
    private AccumulatingClientStream clientStream;
    private List responseList;

    public BidiStreamingStashCallable() {
      responseList = new ArrayList<>();
    }

    public BidiStreamingStashCallable(List responseList) {
      this.responseList = responseList;
    }

    @Override
    public ClientStream internalCall(
        ResponseObserver responseObserver,
        ClientStreamReadyObserver onReady,
        ApiCallContext context) {
      Preconditions.checkNotNull(responseObserver);
      this.responseObserver = responseObserver;
      this.context = context;
      this.clientStream = new AccumulatingClientStream<>();

      StreamControllerStash controller =
          new StreamControllerStash<>(responseList, responseObserver);
      controller.startBidi();
      onReady.onReady(clientStream);

      return clientStream;
    }

    public ApiCallContext getContext() {
      return context;
    }

    public ResponseObserver getActualObserver() {
      return responseObserver;
    }

    public List getActualRequests() {
      return clientStream.getValues();
    }

    private void sendResponses() {
      for (ResponseT response : responseList) {
        responseObserver.onResponse(response);
      }
      responseObserver.onComplete();
    }

    private class AccumulatingClientStream implements ClientStream {
      private List requestList = new ArrayList<>();
      private Throwable error;
      private boolean completed = false;

      @Override
      public void send(T value) {
        requestList.add(value);
      }

      @Override
      public void closeSendWithError(Throwable t) {
        error = t;
      }

      @Override
      public void closeSend() {
        completed = true;
      }

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

      public List getValues() {
        if (!completed) {
          throw new IllegalStateException("Stream not completed.");
        }
        if (error != null) {
          throw ApiExceptionFactory.createException(error, FakeStatusCode.of(Code.UNKNOWN), false);
        }
        return requestList;
      }
    }
  }

  public static class ServerStreamingStashCallable
      extends ServerStreamingCallable {
    private ApiCallContext context;
    private ResponseObserver actualObserver;
    private RequestT actualRequest;
    private List responseList;

    public ServerStreamingStashCallable() {
      responseList = new ArrayList<>();
    }

    public ServerStreamingStashCallable(List responseList) {
      this.responseList = responseList;
    }

    @Override
    public void call(
        RequestT request, ResponseObserver responseObserver, ApiCallContext context) {
      Preconditions.checkNotNull(request);
      Preconditions.checkNotNull(responseObserver);
      this.actualRequest = request;
      this.actualObserver = responseObserver;
      this.context = context;

      StreamControllerStash controller =
          new StreamControllerStash<>(responseList, responseObserver);
      controller.start();
    }

    public ApiCallContext getContext() {
      return context;
    }

    public ResponseObserver getActualObserver() {
      return actualObserver;
    }

    public RequestT getActualRequest() {
      return actualRequest;
    }
  }

  // Minimal implementation of back pressure aware stream controller. Not threadsafe
  private static class StreamControllerStash implements StreamController {
    final ResponseObserver observer;
    final Queue queue;
    boolean autoFlowControl = true;
    long numPending;
    Throwable error;
    boolean delivering, closed;

    public StreamControllerStash(
        List responseList, ResponseObserver observer) {
      this.observer = observer;
      this.queue = Queues.newArrayDeque(responseList);
    }

    public void start() {
      observer.onStart(this);
      if (autoFlowControl) {
        numPending = Integer.MAX_VALUE;
      }
      deliver();
    }

    public void startBidi() {
      start();
    }

    @Override
    public void disableAutoInboundFlowControl() {
      autoFlowControl = false;
    }

    @Override
    public void request(int count) {
      numPending += count;
      deliver();
    }

    @Override
    public void cancel() {
      error = new CancellationException("User cancelled stream");
      deliver();
    }

    private void deliver() {
      if (delivering || closed) return;
      delivering = true;

      try {
        while (error == null && numPending > 0 && !queue.isEmpty()) {
          numPending--;
          observer.onResponse(queue.poll());
        }

        if (error != null || queue.isEmpty()) {
          if (error != null) {
            observer.onError(error);
          } else {
            observer.onComplete();
          }
          closed = true;
        }
      } finally {
        delivering = false;
      }
    }
  }

  public static class ClientStreamingStashCallable
      extends ClientStreamingCallable {
    private ApiCallContext context;
    private ApiStreamObserver responseObserver;
    private AccumulatingStreamObserver requestObserver;
    private ResponseT response;

    public ClientStreamingStashCallable() {}

    public ClientStreamingStashCallable(ResponseT response) {
      this.response = response;
    }

    @Override
    public ApiStreamObserver clientStreamingCall(
        ApiStreamObserver responseObserver, ApiCallContext context) {
      Preconditions.checkNotNull(responseObserver);
      this.responseObserver = responseObserver;
      this.context = context;
      this.requestObserver = new AccumulatingStreamObserver<>();
      return requestObserver;
    }

    public ApiCallContext getContext() {
      return context;
    }

    public ApiStreamObserver getActualObserver() {
      return responseObserver;
    }

    public List getActualRequests() {
      return requestObserver.getValues();
    }

    private void sendResponses() {
      responseObserver.onNext(response);
      responseObserver.onCompleted();
    }

    private class AccumulatingStreamObserver implements ApiStreamObserver {
      private List requestList = new ArrayList<>();
      private Throwable error;
      private boolean completed = false;

      @Override
      public void onNext(T value) {
        requestList.add(value);
      }

      @Override
      public void onError(Throwable t) {
        error = t;
      }

      @Override
      public void onCompleted() {
        completed = true;
        ClientStreamingStashCallable.this.sendResponses();
      }

      public List getValues() {
        if (!completed) {
          throw new IllegalStateException("Stream not completed.");
        }
        if (error != null) {
          throw ApiExceptionFactory.createException(error, FakeStatusCode.of(Code.UNKNOWN), false);
        }
        return requestList;
      }
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy