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

com.azure.storage.blob.specialized.AppendBlobAsyncClient Maven / Gradle / Ivy

There is a newer version: 12.29.0
Show newest version
// 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; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy