com.azure.storage.blob.specialized.AppendBlobAsyncClient Maven / Gradle / Ivy
Show all versions of azure-storage-blob Show documentation
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
package com.azure.storage.blob.specialized;
import com.azure.core.annotation.ReturnType;
import com.azure.core.annotation.ServiceClient;
import com.azure.core.annotation.ServiceMethod;
import com.azure.core.http.HttpPipeline;
import com.azure.core.http.HttpResponse;
import com.azure.core.http.RequestConditions;
import com.azure.core.http.rest.Response;
import com.azure.core.http.rest.SimpleResponse;
import com.azure.core.util.Context;
import com.azure.core.util.FluxUtil;
import com.azure.core.util.logging.ClientLogger;
import com.azure.storage.blob.BlobAsyncClient;
import com.azure.storage.blob.BlobContainerAsyncClient;
import com.azure.storage.blob.BlobServiceAsyncClient;
import com.azure.storage.blob.BlobServiceVersion;
import com.azure.storage.blob.implementation.models.AppendBlobsAppendBlockFromUrlHeaders;
import com.azure.storage.blob.implementation.models.AppendBlobsAppendBlockHeaders;
import com.azure.storage.blob.implementation.models.AppendBlobsCreateHeaders;
import com.azure.storage.blob.implementation.models.EncryptionScope;
import com.azure.storage.blob.models.AppendBlobItem;
import com.azure.storage.blob.models.AppendBlobRequestConditions;
import com.azure.storage.blob.models.BlobHttpHeaders;
import com.azure.storage.blob.models.BlobImmutabilityPolicy;
import com.azure.storage.blob.models.BlobRange;
import com.azure.storage.blob.models.BlobRequestConditions;
import com.azure.storage.blob.models.BlobStorageException;
import com.azure.storage.blob.models.CpkInfo;
import com.azure.storage.blob.models.CustomerProvidedKey;
import com.azure.storage.blob.options.AppendBlobAppendBlockFromUrlOptions;
import com.azure.storage.blob.options.AppendBlobCreateOptions;
import com.azure.storage.blob.options.AppendBlobSealOptions;
import com.azure.storage.common.implementation.Constants;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.ByteBuffer;
import java.util.Map;
import static com.azure.core.util.FluxUtil.monoError;
import static com.azure.core.util.FluxUtil.withContext;
/**
* Client to an append blob. It may only be instantiated through a
* {@link SpecializedBlobClientBuilder#buildAppendBlobAsyncClient()} or via the method
* {@link BlobAsyncClient#getAppendBlobAsyncClient()}. This class does not hold any state about a
* particular blob, but is instead a convenient way of sending appropriate requests to the resource on the service.
*
*
* This client contains operations on a blob. Operations on a container are available on {@link
* BlobContainerAsyncClient}, and operations on the service are available on {@link BlobServiceAsyncClient}.
*
*
* Please refer to the Azure
* Docs for more information.
*
*
* Note this client is an async client that returns reactive responses from Spring Reactor Core project
* (https://projectreactor.io/). Calling the methods in this client will NOT start the actual network
* operation, until {@code .subscribe()} is called on the reactive response. You can simply convert one of these
* responses to a {@link java.util.concurrent.CompletableFuture} object through {@link Mono#toFuture()}.
*/
@ServiceClient(builder = SpecializedBlobClientBuilder.class, isAsync = true)
public final class AppendBlobAsyncClient extends BlobAsyncClientBase {
private static final ClientLogger LOGGER = new ClientLogger(AppendBlobAsyncClient.class);
/**
* Indicates the maximum number of bytes that can be sent in a call to appendBlock.
* @deprecated use {@link AppendBlobAsyncClient#getMaxAppendBlockBytes()}.
*/
@Deprecated
public static final int MAX_APPEND_BLOCK_BYTES = 4 * Constants.MB;
/**
* Indicates the maximum number of blocks allowed in an append blob.
* @deprecated use {@link AppendBlobAsyncClient#getMaxBlocks()}.
*/
@Deprecated
public static final int MAX_BLOCKS = 50000;
/**
* Indicates the maximum number of bytes that can be sent in a call to appendBlock.
*/
static final int MAX_APPEND_BLOCK_BYTES_VERSIONS_2021_12_02_AND_BELOW = 4 * Constants.MB;
/**
* Indicates the maximum number of bytes that can be sent in a call to appendBlock.
* For versions 2022-11-02 and above.
*/
static final int MAX_APPEND_BLOCK_BYTES_VERSIONS_2022_11_02_AND_ABOVE = 100 * Constants.MB;
/**
* Indicates the maximum number of blocks allowed in an append blob.
*/
static final int MAX_APPEND_BLOCKS = 50000;
/**
* Package-private constructor for use by {@link SpecializedBlobClientBuilder}.
*
* @param pipeline The pipeline used to send and receive service requests.
* @param url The endpoint where to send service requests.
* @param serviceVersion The version of the service to receive requests.
* @param accountName The storage account name.
* @param containerName The container name.
* @param blobName The blob name.
* @param snapshot The snapshot identifier for the blob, pass {@code null} to interact with the blob directly.
* @param customerProvidedKey Customer provided key used during encryption of the blob's data on the server, pass
* {@code null} to allow the service to use its own encryption.
* @param encryptionScope Encryption scope used during encryption of the blob's data on the server, pass
* {@code null} to allow the service to use its own encryption.
* @param versionId The version identifier for the blob, pass {@code null} to interact with the latest blob version.
*/
AppendBlobAsyncClient(HttpPipeline pipeline, String url, BlobServiceVersion serviceVersion,
String accountName, String containerName, String blobName, String snapshot, CpkInfo customerProvidedKey,
EncryptionScope encryptionScope, String versionId) {
super(pipeline, url, serviceVersion, accountName, containerName, blobName, snapshot, customerProvidedKey,
encryptionScope, versionId);
}
/**
* Creates a new {@link AppendBlobAsyncClient} with the specified {@code encryptionScope}.
*
* @param encryptionScope the encryption scope for the blob, pass {@code null} to use no encryption scope.
* @return a {@link AppendBlobAsyncClient} with the specified {@code encryptionScope}.
*/
@Override
public AppendBlobAsyncClient getEncryptionScopeAsyncClient(String encryptionScope) {
EncryptionScope finalEncryptionScope = null;
if (encryptionScope != null) {
finalEncryptionScope = new EncryptionScope().setEncryptionScope(encryptionScope);
}
return new AppendBlobAsyncClient(getHttpPipeline(), getAccountUrl(), getServiceVersion(), getAccountName(),
getContainerName(), getBlobName(), getSnapshotId(), getCustomerProvidedKey(), finalEncryptionScope,
getVersionId());
}
/**
* Creates a new {@link AppendBlobAsyncClient} with the specified {@code customerProvidedKey}.
*
* @param customerProvidedKey the {@link CustomerProvidedKey} for the blob,
* pass {@code null} to use no customer provided key.
* @return a {@link AppendBlobAsyncClient} with the specified {@code customerProvidedKey}.
*/
@Override
public AppendBlobAsyncClient getCustomerProvidedKeyAsyncClient(CustomerProvidedKey customerProvidedKey) {
CpkInfo finalCustomerProvidedKey = null;
if (customerProvidedKey != null) {
finalCustomerProvidedKey = new CpkInfo()
.setEncryptionKey(customerProvidedKey.getKey())
.setEncryptionKeySha256(customerProvidedKey.getKeySha256())
.setEncryptionAlgorithm(customerProvidedKey.getEncryptionAlgorithm());
}
return new AppendBlobAsyncClient(getHttpPipeline(), getAccountUrl(), getServiceVersion(), getAccountName(),
getContainerName(), getBlobName(), getSnapshotId(), finalCustomerProvidedKey, encryptionScope,
getVersionId());
}
/**
* Creates a 0-length append blob. Call appendBlock to append data to an append blob. By default this method will
* not overwrite an existing blob.
*
*
Code Samples
*
*
*
* client.create().subscribe(response ->
* System.out.printf("Created AppendBlob at %s%n", response.getLastModified()));
*
*
*
* @return A {@link Mono} containing the information of the created appended blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono create() {
return create(false);
}
/**
* Creates a 0-length append blob. Call appendBlock to append data to an append blob.
*
* Code Samples
*
*
*
* boolean overwrite = false; // Default behavior
* client.create(overwrite).subscribe(response ->
* System.out.printf("Created AppendBlob at %s%n", response.getLastModified()));
*
*
*
* @param overwrite Whether to overwrite, should data exist on the blob.
*
* @return A {@link Mono} containing the information of the created appended blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono create(boolean overwrite) {
BlobRequestConditions blobRequestConditions = new BlobRequestConditions();
if (!overwrite) {
blobRequestConditions.setIfNoneMatch(Constants.HeaderConstants.ETAG_WILDCARD);
}
return createWithResponse(null, null, blobRequestConditions).flatMap(FluxUtil::toMono);
}
/**
* Creates a 0-length append blob. Call appendBlock to append data to an append blob.
*
* To avoid overwriting, pass "*" to {@link BlobRequestConditions#setIfNoneMatch(String)}.
*
*
Code Samples
*
*
*
* BlobHttpHeaders headers = new BlobHttpHeaders()
* .setContentType("binary")
* .setContentLanguage("en-US");
* Map<String, String> metadata = Collections.singletonMap("metadata", "value");
* BlobRequestConditions requestConditions = new BlobRequestConditions().setLeaseId(leaseId)
* .setIfUnmodifiedSince(OffsetDateTime.now().minusDays(3));
*
* client.createWithResponse(headers, metadata, requestConditions).subscribe(response ->
* System.out.printf("Created AppendBlob at %s%n", response.getValue().getLastModified()));
*
*
*
* @param headers {@link BlobHttpHeaders}
* @param metadata Metadata to associate with the blob. If there is leading or trailing whitespace in any
* metadata key or value, it must be removed or encoded.
* @param requestConditions {@link BlobRequestConditions}
* @return A {@link Mono} containing {@link Response} whose {@link Response#getValue() value} contains the created
* appended blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> createWithResponse(BlobHttpHeaders headers, Map metadata,
BlobRequestConditions requestConditions) {
return this.createWithResponse(new AppendBlobCreateOptions().setHeaders(headers).setMetadata(metadata)
.setRequestConditions(requestConditions));
}
/**
* Creates a 0-length append blob. Call appendBlock to append data to an append blob.
*
* To avoid overwriting, pass "*" to {@link BlobRequestConditions#setIfNoneMatch(String)}.
*
*
Code Samples
*
*
*
* BlobHttpHeaders headers = new BlobHttpHeaders()
* .setContentType("binary")
* .setContentLanguage("en-US");
* Map<String, String> metadata = Collections.singletonMap("metadata", "value");
* Map<String, String> tags = Collections.singletonMap("tag", "value");
* BlobRequestConditions requestConditions = new BlobRequestConditions().setLeaseId(leaseId)
* .setIfUnmodifiedSince(OffsetDateTime.now().minusDays(3));
*
* client.createWithResponse(new AppendBlobCreateOptions().setHeaders(headers).setMetadata(metadata)
* .setTags(tags).setRequestConditions(requestConditions)).subscribe(response ->
* System.out.printf("Created AppendBlob at %s%n", response.getValue().getLastModified()));
*
*
*
* @param options {@link AppendBlobCreateOptions}
* @return A {@link Mono} containing {@link Response} whose {@link Response#getValue() value} contains the created
* appended blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> createWithResponse(AppendBlobCreateOptions options) {
try {
return withContext(context -> createWithResponse(options, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> createWithResponse(AppendBlobCreateOptions options, Context context) {
options = (options == null) ? new AppendBlobCreateOptions() : options;
BlobRequestConditions requestConditions = options.getRequestConditions();
requestConditions = (requestConditions == null) ? new BlobRequestConditions() : requestConditions;
context = context == null ? Context.NONE : context;
BlobImmutabilityPolicy immutabilityPolicy = options.getImmutabilityPolicy() == null
? new BlobImmutabilityPolicy() : options.getImmutabilityPolicy();
return this.azureBlobStorage.getAppendBlobs().createWithResponseAsync(containerName, blobName, 0, null,
options.getMetadata(), requestConditions.getLeaseId(), requestConditions.getIfModifiedSince(),
requestConditions.getIfUnmodifiedSince(), requestConditions.getIfMatch(),
requestConditions.getIfNoneMatch(), requestConditions.getTagsConditions(), null,
tagsToString(options.getTags()), immutabilityPolicy.getExpiryTime(), immutabilityPolicy.getPolicyMode(),
options.hasLegalHold(), options.getHeaders(), getCustomerProvidedKey(),
encryptionScope, context)
.map(rb -> {
AppendBlobsCreateHeaders hd = rb.getDeserializedHeaders();
AppendBlobItem item = new AppendBlobItem(hd.getETag(), hd.getLastModified(), hd.getContentMD5(),
hd.isXMsRequestServerEncrypted(), hd.getXMsEncryptionKeySha256(), hd.getXMsEncryptionScope(),
null, null, hd.getXMsVersionId());
return new SimpleResponse<>(rb, item);
});
}
/**
* Creates a 0-length append blob if it does not exist. Call appendBlock to append data to an append blob.
*
* Code Samples
*
*
*
* client.createIfNotExists().subscribe(response ->
* System.out.printf("Created AppendBlob at %s%n", response.getLastModified()));
*
*
*
* @return A reactive response {@link Mono} signaling completion. {@link AppendBlobItem} contains the information of
* the created appended blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono createIfNotExists() {
return this.createIfNotExistsWithResponse(new AppendBlobCreateOptions()).flatMap(FluxUtil::toMono);
}
/**
* Creates a 0-length append blob if it does not exist. Call appendBlock to append data to an append blob.
*
* Code Samples
*
*
*
* BlobHttpHeaders headers = new BlobHttpHeaders()
* .setContentType("binary")
* .setContentLanguage("en-US");
* Map<String, String> metadata = Collections.singletonMap("metadata", "value");
* Map<String, String> tags = Collections.singletonMap("tag", "value");
*
* client.createIfNotExistsWithResponse(new AppendBlobCreateOptions().setHeaders(headers)
* .setMetadata(metadata).setTags(tags)).subscribe(response -> {
* if (response.getStatusCode() == 409) {
* System.out.println("Already exists.");
* } else {
* System.out.println("successfully created.");
* }
* });
*
*
*
* @param options {@link AppendBlobCreateOptions}
* @return A {@link Mono} containing {@link Response} signaling completion, whose {@link Response#getValue() value}
* contains a {@link AppendBlobItem} containing information about the append blob. If {@link Response}'s status code
* is 201, a new page blob was successfully created. If status code is 409, an append blob already existed at this
* location.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> createIfNotExistsWithResponse(AppendBlobCreateOptions options) {
try {
return withContext(context -> createIfNotExistsWithResponse(options, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> createIfNotExistsWithResponse(AppendBlobCreateOptions options, Context context) {
try {
options = options == null ? new AppendBlobCreateOptions() : options;
options.setRequestConditions(new AppendBlobRequestConditions()
.setIfNoneMatch(Constants.HeaderConstants.ETAG_WILDCARD));
return createWithResponse(options, context).onErrorResume(t -> t instanceof BlobStorageException
&& ((BlobStorageException) t).getStatusCode() == 409,
t -> {
HttpResponse response = ((BlobStorageException) t).getResponse();
return Mono.just(new SimpleResponse<>(response.getRequest(), response.getStatusCode(),
response.getHeaders(), null));
});
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
/**
* Commits a new block of data to the end of the existing append blob.
*
* Note that the data passed must be replayable if retries are enabled (the default). In other words, the
* {@code Flux} must produce the same data each time it is subscribed to.
*
* For service versions 2022-11-02 and later, the max block size is 100 MB. For previous versions, the max block
* size is 4 MB. For more information, see the
* Azure Docs.
*
*
Code Samples
*
*
*
* client.appendBlock(data, length).subscribe(response ->
* System.out.printf("AppendBlob has %d committed blocks%n", response.getBlobCommittedBlockCount()));
*
*
*
* @param data The data to write to the blob. Note that this {@code Flux} must be replayable if retries are enabled
* (the default). In other words, the Flux must produce the same data each time it is subscribed to.
* @param length The exact length of the data. It is important that this value match precisely the length of the
* data emitted by the {@code Flux}.
* @return {@link Mono} containing the information of the append blob operation.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono appendBlock(Flux data, long length) {
return appendBlockWithResponse(data, length, null, null).flatMap(FluxUtil::toMono);
}
/**
* Commits a new block of data to the end of the existing append blob.
*
* Note that the data passed must be replayable if retries are enabled (the default). In other words, the
* {@code Flux} must produce the same data each time it is subscribed to.
*
* For service versions 2022-11-02 and later, the max block size is 100 MB. For previous versions, the max block
* size is 4 MB. For more information, see the
* Azure Docs.
*
*
Code Samples
*
*
*
* byte[] md5 = MessageDigest.getInstance("MD5").digest("data".getBytes(StandardCharsets.UTF_8));
* AppendBlobRequestConditions requestConditions = new AppendBlobRequestConditions()
* .setAppendPosition(POSITION)
* .setMaxSize(maxSize);
*
* client.appendBlockWithResponse(data, length, md5, requestConditions).subscribe(response ->
* System.out.printf("AppendBlob has %d committed blocks%n", response.getValue().getBlobCommittedBlockCount()));
*
*
*
* @param data The data to write to the blob. Note that this {@code Flux} must be replayable if retries are enabled
* (the default). In other words, the Flux must produce the same data each time it is subscribed to.
* @param length The exact length of the data. It is important that this value match precisely the length of the
* data emitted by the {@code Flux}.
* @param contentMd5 An MD5 hash of the block content. This hash is used to verify the integrity of the block during
* transport. When this header is specified, the storage service compares the hash of the content that has arrived
* with this header value. Note that this MD5 hash is not stored with the blob. If the two hashes do not match, the
* operation will fail.
* @param appendBlobRequestConditions {@link AppendBlobRequestConditions}
* @return A {@link Mono} containing {@link Response} whose {@link Response#getValue() value} contains the append
* blob operation.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> appendBlockWithResponse(Flux data, long length, byte[] contentMd5,
AppendBlobRequestConditions appendBlobRequestConditions) {
try {
return withContext(context ->
appendBlockWithResponse(data, length, contentMd5, appendBlobRequestConditions, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> appendBlockWithResponse(Flux data, long length, byte[] contentMd5,
AppendBlobRequestConditions appendBlobRequestConditions, Context context) {
if (data == null) {
return Mono.error(new NullPointerException("'data' cannot be null."));
}
appendBlobRequestConditions = appendBlobRequestConditions == null ? new AppendBlobRequestConditions()
: appendBlobRequestConditions;
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getAppendBlobs().appendBlockWithResponseAsync(
containerName, blobName, length, data, null, contentMd5, null, appendBlobRequestConditions.getLeaseId(),
appendBlobRequestConditions.getMaxSize(), appendBlobRequestConditions.getAppendPosition(),
appendBlobRequestConditions.getIfModifiedSince(), appendBlobRequestConditions.getIfUnmodifiedSince(),
appendBlobRequestConditions.getIfMatch(), appendBlobRequestConditions.getIfNoneMatch(),
appendBlobRequestConditions.getTagsConditions(), null, getCustomerProvidedKey(), encryptionScope,
context)
.map(rb -> {
AppendBlobsAppendBlockHeaders hd = rb.getDeserializedHeaders();
AppendBlobItem item = new AppendBlobItem(hd.getETag(), hd.getLastModified(), hd.getContentMD5(),
hd.isXMsRequestServerEncrypted(), hd.getXMsEncryptionKeySha256(), hd.getXMsEncryptionScope(),
hd.getXMsBlobAppendOffset(), hd.getXMsBlobCommittedBlockCount());
return new SimpleResponse<>(rb, item);
});
}
/**
* Commits a new block of data from another blob to the end of this append blob.
*
* Code Samples
*
*
*
* client.appendBlockFromUrl(sourceUrl, new BlobRange(offset, count)).subscribe(response ->
* System.out.printf("AppendBlob has %d committed blocks%n", response.getBlobCommittedBlockCount()));
*
*
*
* @param sourceUrl The url to the blob that will be the source of the copy. A source blob in the same storage
* account can be authenticated via Shared Key. However, if the source is a blob in another account, the source blob
* must either be public or must be authenticated via a shared access signature. If the source blob is public, no
* authentication is required to perform the operation.
* @param sourceRange The source {@link BlobRange} to copy.
* @return {@link Mono} containing the information of the append blob operation.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono appendBlockFromUrl(String sourceUrl, BlobRange sourceRange) {
return appendBlockFromUrlWithResponse(sourceUrl, sourceRange, null, null, null).flatMap(FluxUtil::toMono);
}
/**
* Commits a new block of data from another blob to the end of this append blob.
*
* Code Samples
*
*
*
* AppendBlobRequestConditions appendBlobRequestConditions = new AppendBlobRequestConditions()
* .setAppendPosition(POSITION)
* .setMaxSize(maxSize);
*
* BlobRequestConditions modifiedRequestConditions = new BlobRequestConditions()
* .setIfUnmodifiedSince(OffsetDateTime.now().minusDays(3));
*
* client.appendBlockFromUrlWithResponse(sourceUrl, new BlobRange(offset, count), null,
* appendBlobRequestConditions, modifiedRequestConditions).subscribe(response ->
* System.out.printf("AppendBlob has %d committed blocks%n", response.getValue().getBlobCommittedBlockCount()));
*
*
*
* @param sourceUrl The url to the blob that will be the source of the copy. A source blob in the same storage
* account can be authenticated via Shared Key. However, if the source is a blob in another account, the source blob
* must either be public or must be authenticated via a shared access signature. If the source blob is public, no
* authentication is required to perform the operation.
* @param sourceRange {@link BlobRange}
* @param sourceContentMD5 An MD5 hash of the block content from the source blob. If specified, the service will
* calculate the MD5 of the received data and fail the request if it does not match the provided MD5.
* @param destRequestConditions {@link AppendBlobRequestConditions}
* @param sourceRequestConditions {@link BlobRequestConditions}
* @return A {@link Mono} containing {@link Response} whose {@link Response#getValue() value} contains the append
* blob operation.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> appendBlockFromUrlWithResponse(String sourceUrl, BlobRange sourceRange,
byte[] sourceContentMD5, AppendBlobRequestConditions destRequestConditions,
BlobRequestConditions sourceRequestConditions) {
return appendBlockFromUrlWithResponse(new AppendBlobAppendBlockFromUrlOptions(sourceUrl)
.setSourceRange(sourceRange).setSourceContentMd5(sourceContentMD5)
.setDestinationRequestConditions(destRequestConditions)
.setSourceRequestConditions(sourceRequestConditions));
}
/**
* Commits a new block of data from another blob to the end of this append blob.
*
* Code Samples
*
*
*
* AppendBlobRequestConditions appendBlobRequestConditions = new AppendBlobRequestConditions()
* .setAppendPosition(POSITION)
* .setMaxSize(maxSize);
*
* BlobRequestConditions modifiedRequestConditions = new BlobRequestConditions()
* .setIfUnmodifiedSince(OffsetDateTime.now().minusDays(3));
*
* client.appendBlockFromUrlWithResponse(new AppendBlobAppendBlockFromUrlOptions(sourceUrl)
* .setSourceRange(new BlobRange(offset, count))
* .setDestinationRequestConditions(appendBlobRequestConditions)
* .setSourceRequestConditions(modifiedRequestConditions)).subscribe(response ->
* System.out.printf("AppendBlob has %d committed blocks%n", response.getValue().getBlobCommittedBlockCount()));
*
*
*
* @param options Parameters for the operation.
* @return A {@link Mono} containing {@link Response} whose {@link Response#getValue() value} contains the append
* blob operation.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> appendBlockFromUrlWithResponse(AppendBlobAppendBlockFromUrlOptions options) {
try {
return withContext(context -> appendBlockFromUrlWithResponse(options, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> appendBlockFromUrlWithResponse(AppendBlobAppendBlockFromUrlOptions options, Context context) {
BlobRange sourceRange = (options.getSourceRange() == null) ? new BlobRange(0) : options.getSourceRange();
AppendBlobRequestConditions destRequestConditions = (options.getDestinationRequestConditions() == null)
? new AppendBlobRequestConditions() : options.getDestinationRequestConditions();
RequestConditions sourceRequestConditions = (options.getSourceRequestConditions() == null)
? new RequestConditions() : options.getSourceRequestConditions();
try {
new URL(options.getSourceUrl());
} catch (MalformedURLException ex) {
throw LOGGER.logExceptionAsError(new IllegalArgumentException("'sourceUrl' is not a valid url.", ex));
}
context = context == null ? Context.NONE : context;
String sourceAuth = options.getSourceAuthorization() == null
? null : options.getSourceAuthorization().toString();
return this.azureBlobStorage.getAppendBlobs().appendBlockFromUrlWithResponseAsync(containerName, blobName, options.getSourceUrl(), 0,
sourceRange.toString(), options.getSourceContentMd5(), null, null, null, destRequestConditions.getLeaseId(),
destRequestConditions.getMaxSize(), destRequestConditions.getAppendPosition(),
destRequestConditions.getIfModifiedSince(), destRequestConditions.getIfUnmodifiedSince(),
destRequestConditions.getIfMatch(), destRequestConditions.getIfNoneMatch(),
destRequestConditions.getTagsConditions(), sourceRequestConditions.getIfModifiedSince(),
sourceRequestConditions.getIfUnmodifiedSince(), sourceRequestConditions.getIfMatch(),
sourceRequestConditions.getIfNoneMatch(), null, sourceAuth, getCustomerProvidedKey(),
encryptionScope, context)
.map(rb -> {
AppendBlobsAppendBlockFromUrlHeaders hd = rb.getDeserializedHeaders();
AppendBlobItem item = new AppendBlobItem(hd.getETag(), hd.getLastModified(), hd.getContentMD5(),
hd.isXMsRequestServerEncrypted(), hd.getXMsEncryptionKeySha256(), hd.getXMsEncryptionScope(),
hd.getXMsBlobAppendOffset(), hd.getXMsBlobCommittedBlockCount());
return new SimpleResponse<>(rb, item);
});
}
/**
* Seals an append blob, making it read only. Any subsequent appends will fail.
*
* Code Samples
*
*
*
* client.seal().subscribe(response -> System.out.println("Sealed AppendBlob"));
*
*
*
* @return A reactive response signalling completion.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono seal() {
return sealWithResponse(new AppendBlobSealOptions()).flatMap(FluxUtil::toMono);
}
/**
* Seals an append blob, making it read only. Any subsequent appends will fail.
*
* Code Samples
*
*
*
* AppendBlobRequestConditions requestConditions = new AppendBlobRequestConditions().setLeaseId(leaseId)
* .setIfUnmodifiedSince(OffsetDateTime.now().minusDays(3));
*
* client.sealWithResponse(new AppendBlobSealOptions().setRequestConditions(requestConditions))
* .subscribe(response -> System.out.println("Sealed AppendBlob"));
*
*
*
* @param options {@link AppendBlobSealOptions}
* @return A reactive response signalling completion.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> sealWithResponse(AppendBlobSealOptions options) {
try {
return withContext(context -> sealWithResponse(options, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> sealWithResponse(AppendBlobSealOptions options, Context context) {
options = (options == null) ? new AppendBlobSealOptions() : options;
AppendBlobRequestConditions requestConditions = options.getRequestConditions();
requestConditions = (requestConditions == null) ? new AppendBlobRequestConditions() : requestConditions;
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getAppendBlobs().sealNoCustomHeadersWithResponseAsync(containerName, blobName,
null, null, requestConditions.getLeaseId(), requestConditions.getIfModifiedSince(),
requestConditions.getIfUnmodifiedSince(), requestConditions.getIfMatch(),
requestConditions.getIfNoneMatch(), requestConditions.getAppendPosition(), context);
}
/**
* Get the max number of append block bytes based on service version being used. Service versions 2022-11-02 and
* above support uploading block bytes up to 100MB, all older service versions support up to 4MB.
*
* @return the max number of block bytes that can be uploaded based on service version.
*/
public int getMaxAppendBlockBytes() {
if (getServiceVersion().ordinal() < BlobServiceVersion.V2022_11_02.ordinal()) {
return MAX_APPEND_BLOCK_BYTES_VERSIONS_2021_12_02_AND_BELOW;
} else {
return MAX_APPEND_BLOCK_BYTES_VERSIONS_2022_11_02_AND_ABOVE;
}
}
/**
* Get the maximum number of blocks allowed in an append blob.
*
* @return the max number of blocks that can be uploaded in an append blob.
*/
public int getMaxBlocks() {
return MAX_APPEND_BLOCKS;
}
}