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

com.google.cloud.storage.GrpcResumableSession Maven / Gradle / Ivy

There is a newer version: 2.45.0
Show newest version
/*
 * Copyright 2023 Google LLC
 *
 * 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.cloud.storage;

import com.google.api.core.ApiFutures;
import com.google.api.gax.grpc.GrpcCallContext;
import com.google.api.gax.retrying.ResultRetryAlgorithm;
import com.google.api.gax.rpc.ClientStreamingCallable;
import com.google.api.gax.rpc.UnaryCallable;
import com.google.cloud.storage.BufferedWritableByteChannelSession.BufferedWritableByteChannel;
import com.google.cloud.storage.Conversions.Decoder;
import com.google.cloud.storage.Retrying.RetryingDependencies;
import com.google.storage.v2.Object;
import com.google.storage.v2.QueryWriteStatusRequest;
import com.google.storage.v2.QueryWriteStatusResponse;
import com.google.storage.v2.WriteObjectRequest;
import com.google.storage.v2.WriteObjectResponse;
import java.util.concurrent.atomic.AtomicBoolean;
import org.checkerframework.checker.nullness.qual.Nullable;

final class GrpcResumableSession {

  private final RetryingDependencies deps;
  private final ResultRetryAlgorithm alg;
  private final ClientStreamingCallable writeCallable;
  private final UnaryCallable
      queryWriteStatusCallable;
  private final ResumableWrite resumableWrite;
  private final Hasher hasher;

  GrpcResumableSession(
      RetryingDependencies deps,
      ResultRetryAlgorithm alg,
      ClientStreamingCallable writeCallable,
      UnaryCallable queryWriteStatusCallable,
      ResumableWrite resumableWrite,
      Hasher hasher) {
    this.deps = deps;
    this.alg = alg;
    this.writeCallable = writeCallable;
    this.queryWriteStatusCallable = queryWriteStatusCallable;
    this.resumableWrite = resumableWrite;
    this.hasher = hasher;
  }

  ResumableOperationResult<@Nullable Object> query() {
    QueryWriteStatusRequest.Builder b =
        QueryWriteStatusRequest.newBuilder().setUploadId(resumableWrite.getRes().getUploadId());
    if (resumableWrite.getReq().hasCommonObjectRequestParams()) {
      b.setCommonObjectRequestParams(resumableWrite.getReq().getCommonObjectRequestParams());
    }
    QueryWriteStatusRequest req = b.build();
    try {
      QueryWriteStatusResponse response = queryWriteStatusCallable.call(req);
      if (response.hasResource()) {
        return ResumableOperationResult.complete(
            response.getResource(), response.getResource().getSize());
      } else {
        return ResumableOperationResult.incremental(response.getPersistedSize());
      }
    } catch (Exception e) {
      throw StorageException.coalesce(e);
    }
  }

  ResumableOperationResult<@Nullable Object> put(RewindableContent content) {
    AtomicBoolean dirty = new AtomicBoolean(false);
    GrpcCallContext retryingCallContext = Retrying.newCallContext();
    BufferHandle handle = BufferHandle.allocate(ByteSizeConstants._2MiB);

    return Retrying.run(
        deps,
        alg,
        () -> {
          if (dirty.getAndSet(true)) {
            ResumableOperationResult<@Nullable Object> query = query();
            if (query.getObject() != null) {
              return query;
            } else {
              handle.get().clear();
              content.rewindTo(query.getPersistedSize());
            }
          }
          WritableByteChannelSession session =
              ResumableMedia.gapic()
                  .write()
                  .byteChannel(writeCallable.withDefaultCallContext(retryingCallContext))
                  .setByteStringStrategy(ByteStringStrategy.copy())
                  .setHasher(hasher)
                  .resumable()
                  .setFsyncEvery(false)
                  .buffered(handle)
                  .setStartAsync(ApiFutures.immediateFuture(resumableWrite))
                  .build();

          try (BufferedWritableByteChannel channel = session.open()) {
            content.writeTo(channel);
          }

          WriteObjectResponse response = session.getResult().get();
          if (response.hasResource()) {
            return ResumableOperationResult.complete(
                response.getResource(), response.getResource().getSize());
          } else {
            return ResumableOperationResult.incremental(response.getPersistedSize());
          }
        },
        Decoder.identity());
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy