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

com.google.cloud.storage.DefaultBlobWriteSessionConfig 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.ApiFuture;
import com.google.api.core.ApiFutures;
import com.google.api.core.BetaApi;
import com.google.api.core.InternalApi;
import com.google.api.gax.grpc.GrpcCallContext;
import com.google.api.services.storage.model.StorageObject;
import com.google.cloud.storage.BufferedWritableByteChannelSession.BufferedWritableByteChannel;
import com.google.cloud.storage.Conversions.Decoder;
import com.google.cloud.storage.TransportCompatibility.Transport;
import com.google.cloud.storage.UnifiedOpts.ObjectTargetOpt;
import com.google.cloud.storage.UnifiedOpts.Opts;
import com.google.cloud.storage.spi.v1.StorageRpc;
import com.google.common.base.Preconditions;
import com.google.common.util.concurrent.MoreExecutors;
import com.google.storage.v2.WriteObjectRequest;
import com.google.storage.v2.WriteObjectResponse;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.WritableByteChannel;
import java.time.Clock;
import java.util.Map;
import java.util.Objects;
import java.util.function.Supplier;
import javax.annotation.concurrent.Immutable;

/**
 * Default Configuration to represent uploading to Google Cloud Storage in a chunked manner.
 *
 * 

Perform a resumable upload, uploading at most {@code chunkSize} bytes each PUT. * *

Configuration of chunk size can be performed via {@link * DefaultBlobWriteSessionConfig#withChunkSize(int)}. * *

An instance of this class will provide a {@link BlobWriteSession} is logically equivalent to * the following: * *

{@code
 * Storage storage = ...;
 * WriteChannel writeChannel = storage.writer(BlobInfo, BlobWriteOption);
 * writeChannel.setChunkSize(chunkSize);
 * }
* * @since 2.26.0 This new api is in preview and is subject to breaking changes. */ @Immutable @BetaApi @TransportCompatibility({Transport.GRPC, Transport.HTTP}) public final class DefaultBlobWriteSessionConfig extends BlobWriteSessionConfig implements BlobWriteSessionConfig.HttpCompatible, BlobWriteSessionConfig.GrpcCompatible { private static final long serialVersionUID = -6873740918589930633L; private final int chunkSize; @InternalApi DefaultBlobWriteSessionConfig(int chunkSize) { this.chunkSize = chunkSize; } /** * The number of bytes each chunk can be. * *

Default: {@code 16777216 (16 MiB)} * * @see #withChunkSize(int) * @since 2.26.0 This new api is in preview and is subject to breaking changes. */ public int getChunkSize() { return chunkSize; } /** * Create a new instance with the {@code chunkSize} set to the specified value. * *

Default: {@code 16777216 (16 MiB)} * * @param chunkSize The number of bytes each chunk should be. Must be >= {@code 262144 (256 * KiB)} * @return The new instance * @see #getChunkSize() * @since 2.26.0 This new api is in preview and is subject to breaking changes. */ @BetaApi public DefaultBlobWriteSessionConfig withChunkSize(int chunkSize) { Preconditions.checkArgument( chunkSize >= ByteSizeConstants._256KiB, "chunkSize must be >= %d", ByteSizeConstants._256KiB); return new DefaultBlobWriteSessionConfig(chunkSize); } @Override public boolean equals(Object o) { if (this == o) { return true; } if (!(o instanceof DefaultBlobWriteSessionConfig)) { return false; } DefaultBlobWriteSessionConfig that = (DefaultBlobWriteSessionConfig) o; return chunkSize == that.chunkSize; } @Override public int hashCode() { return Objects.hashCode(chunkSize); } @Override @InternalApi WriterFactory createFactory(Clock clock) { return new Factory(chunkSize); } @InternalApi private static final class Factory implements WriterFactory { private static final Decoder WRITE_OBJECT_RESPONSE_BLOB_INFO_DECODER = Conversions.grpc().blobInfo().compose(WriteObjectResponse::getResource); private final int chunkSize; private Factory(int chunkSize) { this.chunkSize = chunkSize; } @InternalApi @Override public WritableByteChannelSession writeSession( StorageInternal s, BlobInfo info, Opts opts) { if (s instanceof GrpcStorageImpl) { return new DecoratedWritableByteChannelSession<>( new LazySession<>( new LazyWriteChannel<>( () -> { GrpcStorageImpl grpc = (GrpcStorageImpl) s; GrpcCallContext grpcCallContext = opts.grpcMetadataMapper().apply(GrpcCallContext.createDefault()); WriteObjectRequest req = grpc.getWriteObjectRequest(info, opts); ApiFuture startResumableWrite = grpc.startResumableWrite(grpcCallContext, req); return ResumableMedia.gapic() .write() .byteChannel( grpc.storageClient .writeObjectCallable() .withDefaultCallContext(grpcCallContext)) .setHasher(Hasher.noop()) .setByteStringStrategy(ByteStringStrategy.copy()) .resumable() .withRetryConfig( grpc.getOptions(), grpc.retryAlgorithmManager.idempotent()) .buffered(BufferHandle.allocate(chunkSize)) .setStartAsync(startResumableWrite) .build(); })), WRITE_OBJECT_RESPONSE_BLOB_INFO_DECODER); } else if (s instanceof StorageImpl) { StorageImpl json = (StorageImpl) s; return new DecoratedWritableByteChannelSession<>( new LazySession<>( new LazyWriteChannel<>( () -> { final Map optionsMap = opts.getRpcOptions(); BlobInfo.Builder builder = info.toBuilder().setMd5(null).setCrc32c(null); BlobInfo updated = opts.blobInfoMapper().apply(builder).build(); StorageObject encode = Conversions.json().blobInfo().encode(updated); Supplier uploadIdSupplier = ResumableMedia.startUploadForBlobInfo( json.getOptions(), updated, optionsMap, json.retryAlgorithmManager.getForResumableUploadSessionCreate( optionsMap)); ApiFuture startAsync = ApiFutures.immediateFuture( JsonResumableWrite.of( encode, optionsMap, uploadIdSupplier.get(), 0L)); return ResumableMedia.http() .write() .byteChannel(HttpClientContext.from(json.storageRpc)) .resumable() .buffered(BufferHandle.allocate(chunkSize)) .setStartAsync(startAsync) .build(); })), Conversions.json().blobInfo()); } else { throw new IllegalStateException( "Unknown Storage implementation: " + s.getClass().getName()); } } } static final class DecoratedWritableByteChannelSession implements WritableByteChannelSession { private final WritableByteChannelSession delegate; private final Decoder decoder; DecoratedWritableByteChannelSession( WritableByteChannelSession delegate, Decoder decoder) { this.delegate = delegate; this.decoder = decoder; } @Override public ApiFuture openAsync() { return ApiFutures.catchingAsync( delegate.openAsync(), Throwable.class, throwable -> ApiFutures.immediateFailedFuture(StorageException.coalesce(throwable)), MoreExecutors.directExecutor()); } @Override public ApiFuture getResult() { ApiFuture decodeResult = ApiFutures.transform( delegate.getResult(), decoder::decode, MoreExecutors.directExecutor()); return ApiFutures.catchingAsync( decodeResult, Throwable.class, throwable -> ApiFutures.immediateFailedFuture(StorageException.coalesce(throwable)), MoreExecutors.directExecutor()); } } static final class LazySession implements WritableByteChannelSession { private final LazyWriteChannel lazy; LazySession(LazyWriteChannel lazy) { this.lazy = lazy; } @Override public ApiFuture openAsync() { // make sure the errors coming out of the BufferedWritableByteChannel are either IOException // or StorageException return ApiFutures.transform( lazy.getSession().openAsync(), delegate -> new BufferedWritableByteChannel() { @Override public int write(ByteBuffer src) throws IOException { try { return delegate.write(src); } catch (IOException e) { throw e; } catch (Exception e) { throw StorageException.coalesce(e); } } @Override public void flush() throws IOException { try { delegate.flush(); } catch (IOException e) { throw e; } catch (Exception e) { throw StorageException.coalesce(e); } } @Override public boolean isOpen() { try { return delegate.isOpen(); } catch (Exception e) { throw StorageException.coalesce(e); } } @Override public void close() throws IOException { try { delegate.close(); } catch (IOException e) { throw e; } catch (Exception e) { throw StorageException.coalesce(e); } } }, MoreExecutors.directExecutor()); } @Override public ApiFuture getResult() { return lazy.getSession().getResult(); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy