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

kvd.client.KvdGet Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2021 Andre Gebers
 *
 * 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 kvd.client;

import java.io.InputStream;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.function.Consumer;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import kvd.common.KvdException;
import kvd.common.packet.Packets;
import kvd.common.packet.proto.Packet;
import kvd.common.packet.proto.PacketType;

class KvdGet implements Abortable {

  private static final Logger log = LoggerFactory.getLogger(KvdGet.class);

  private ClientBackend backend;

  private byte[] key;

  private CompletableFuture future = new CompletableFuture<>();

  private int channelId;

  private Consumer closeListener;

  private KvdGetInputStream stream;

  private AtomicBoolean closed = new AtomicBoolean();

  private int txId;

  public KvdGet(ClientBackend backend, int txId, byte[] key, Consumer closeListener) {
    this.backend = backend;
    this.txId = txId;
    this.key = key;
    this.closeListener = closeListener;
    stream = new KvdGetInputStream(this::closeInternal);
  }

  public void start() {
    channelId = backend.createChannel(this::receive);
    try {
      backend.sendAsync(Packets.packet(PacketType.GET_INIT, channelId, txId, key));
    } catch(Exception e) {
      try {
        close();
      } catch(Exception e2) {
        // ignore
      }
      throw new KvdException("get failed", e);
    }
  }

  @Override
  public void abort() {
    future.completeExceptionally(new KvdException("aborted"));
    stream.abort();
    close();
  }

  private void close() {
    closeInternal();
    stream.close();
  }

  private void closeInternal() {
    if(!closed.getAndSet(true)) {
      this.closeListener.accept(this);
      future.complete(null);
      backend.closeChannel(channelId);
    }
  }

  public void receive(Packet packet) {
    if(PacketType.GET_DATA.equals(packet.getType())) {
      future.complete(stream);
      stream.fill(packet.getByteBody().toByteArray());
    } else if(PacketType.GET_FINISH.equals(packet.getType())) {
      close();
    } else if(PacketType.GET_ABORT.equals(packet.getType())) {
      abort();
    } else {
      log.error("received unexpected packet " + packet.getType());
      future.completeExceptionally(new KvdException("received unexpected packet " + packet.getType()));
      abort();
    }
  }

  public CompletableFuture getFuture() {
    return future;
  }

  @Override
  public String toString() {
    return "GET " + key;
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy