com.azure.storage.blob.specialized.PageBlobAsyncClient 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.HttpRange;
import com.azure.core.http.HttpResponse;
import com.azure.core.http.RequestConditions;
import com.azure.core.http.rest.PagedFlux;
import com.azure.core.http.rest.PagedResponse;
import com.azure.core.http.rest.PagedResponseBase;
import com.azure.core.http.rest.Response;
import com.azure.core.http.rest.ResponseBase;
import com.azure.core.http.rest.SimpleResponse;
import com.azure.core.util.Context;
import com.azure.core.util.FluxUtil;
import com.azure.core.util.UrlBuilder;
import com.azure.core.util.logging.ClientLogger;
import com.azure.storage.blob.BlobAsyncClient;
import com.azure.storage.blob.BlobServiceVersion;
import com.azure.storage.blob.implementation.models.EncryptionScope;
import com.azure.storage.blob.implementation.models.PageBlobsClearPagesHeaders;
import com.azure.storage.blob.implementation.models.PageBlobsCreateHeaders;
import com.azure.storage.blob.implementation.models.PageBlobsGetPageRangesDiffHeaders;
import com.azure.storage.blob.implementation.models.PageBlobsGetPageRangesHeaders;
import com.azure.storage.blob.implementation.models.PageBlobsResizeHeaders;
import com.azure.storage.blob.implementation.models.PageBlobsUpdateSequenceNumberHeaders;
import com.azure.storage.blob.implementation.models.PageBlobsUploadPagesFromURLHeaders;
import com.azure.storage.blob.implementation.models.PageBlobsUploadPagesHeaders;
import com.azure.storage.blob.implementation.models.PageListHelper;
import com.azure.storage.blob.implementation.util.ModelHelper;
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.ClearRange;
import com.azure.storage.blob.models.CopyStatusType;
import com.azure.storage.blob.models.CpkInfo;
import com.azure.storage.blob.models.CustomerProvidedKey;
import com.azure.storage.blob.models.PageBlobCopyIncrementalRequestConditions;
import com.azure.storage.blob.models.PageBlobItem;
import com.azure.storage.blob.models.PageBlobRequestConditions;
import com.azure.storage.blob.models.PageList;
import com.azure.storage.blob.models.PageRange;
import com.azure.storage.blob.models.PageRangeItem;
import com.azure.storage.blob.models.SequenceNumberActionType;
import com.azure.storage.blob.options.ListPageRangesDiffOptions;
import com.azure.storage.blob.options.ListPageRangesOptions;
import com.azure.storage.blob.options.PageBlobCopyIncrementalOptions;
import com.azure.storage.blob.options.PageBlobCreateOptions;
import com.azure.storage.blob.options.PageBlobUploadPagesFromUrlOptions;
import com.azure.storage.common.implementation.Constants;
import com.azure.storage.common.implementation.StorageImplUtils;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.ByteBuffer;
import java.time.Duration;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.function.BiFunction;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import static com.azure.core.util.FluxUtil.monoError;
import static com.azure.core.util.FluxUtil.withContext;
/**
* Client to a page blob. It may only be instantiated through a {@link SpecializedBlobClientBuilder} or via the method
* {@link BlobAsyncClient#getPageBlobAsyncClient()}. 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.
*
*
* 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 PageBlobAsyncClient extends BlobAsyncClientBase {
/**
* Indicates the number of bytes in a page.
*/
public static final int PAGE_BYTES = 512;
/**
* Indicates the maximum number of bytes that may be sent in a call to putPage.
*/
public static final int MAX_PUT_PAGES_BYTES = 4 * Constants.MB;
private static final ClientLogger LOGGER = new ClientLogger(PageBlobAsyncClient.class);
/**
* 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.
*/
PageBlobAsyncClient(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 PageBlobAsyncClient} with the specified {@code encryptionScope}.
*
* @param encryptionScope the encryption scope for the blob, pass {@code null} to use no encryption scope.
* @return a {@link PageBlobAsyncClient} with the specified {@code encryptionScope}.
*/
@Override
public PageBlobAsyncClient getEncryptionScopeAsyncClient(String encryptionScope) {
EncryptionScope finalEncryptionScope = null;
if (encryptionScope != null) {
finalEncryptionScope = new EncryptionScope().setEncryptionScope(encryptionScope);
}
return new PageBlobAsyncClient(getHttpPipeline(), getAccountUrl(), getServiceVersion(), getAccountName(),
getContainerName(), getBlobName(), getSnapshotId(), getCustomerProvidedKey(), finalEncryptionScope,
getVersionId());
}
/**
* Creates a new {@link PageBlobAsyncClient} 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 PageBlobAsyncClient} with the specified {@code customerProvidedKey}.
*/
@Override
public PageBlobAsyncClient getCustomerProvidedKeyAsyncClient(CustomerProvidedKey customerProvidedKey) {
CpkInfo finalCustomerProvidedKey = null;
if (customerProvidedKey != null) {
finalCustomerProvidedKey = new CpkInfo()
.setEncryptionKey(customerProvidedKey.getKey())
.setEncryptionKeySha256(customerProvidedKey.getKeySha256())
.setEncryptionAlgorithm(customerProvidedKey.getEncryptionAlgorithm());
}
return new PageBlobAsyncClient(getHttpPipeline(), getAccountUrl(), getServiceVersion(), getAccountName(),
getContainerName(), getBlobName(), getSnapshotId(), finalCustomerProvidedKey, encryptionScope,
getVersionId());
}
private static String pageRangeToString(PageRange pageRange) {
if (pageRange.getStart() < 0 || pageRange.getEnd() <= 0) {
throw new IllegalArgumentException("PageRange's start and end values must be greater than or equal to "
+ "0 if specified.");
}
if (pageRange.getStart() % PAGE_BYTES != 0) {
throw new IllegalArgumentException("PageRange's start value must be a multiple of 512.");
}
if (pageRange.getEnd() % PAGE_BYTES != PAGE_BYTES - 1) {
throw new IllegalArgumentException("PageRange's end value must be 1 less than a multiple of 512.");
}
if (pageRange.getEnd() <= pageRange.getStart()) {
throw new IllegalArgumentException("PageRange's End value must be after the start.");
}
return "bytes=" + pageRange.getStart() + '-' + pageRange.getEnd();
}
/**
* Creates a page blob of the specified length. By default, this method will not overwrite an existing blob.
* Call PutPage to upload data to a page blob. For more information, see the
* Azure Docs.
*
*
Code Samples
*
*
*
* client.create(size).subscribe(response -> System.out.printf(
* "Created page blob with sequence number %s%n", response.getBlobSequenceNumber()));
*
*
*
* @param size Specifies the maximum size for the page blob, up to 8 TB. The page blob size must be aligned to a
* 512-byte boundary.
*
* @return A reactive response containing the information of the created page blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono create(long size) {
return create(size, false);
}
/**
* Creates a page blob of the specified length. Call PutPage to upload data to a page blob. For more
* information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* boolean overwrite = false; // Default behavior
* client.create(size, overwrite).subscribe(response -> System.out.printf(
* "Created page blob with sequence number %s%n", response.getBlobSequenceNumber()));
*
*
*
* @param size Specifies the maximum size for the page blob, up to 8 TB. The page blob size must be aligned to a
* 512-byte boundary.
* @param overwrite Whether to overwrite, should data exist on the blob.
* @return A reactive response containing the information of the created page blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono create(long size, boolean overwrite) {
BlobRequestConditions blobRequestConditions = new BlobRequestConditions();
if (!overwrite) {
blobRequestConditions.setIfNoneMatch(Constants.HeaderConstants.ETAG_WILDCARD);
}
return createWithResponse(size, null, null, null, blobRequestConditions).flatMap(FluxUtil::toMono);
}
/**
* Creates a page blob of the specified length. Call PutPage to upload data to a page blob. For more
* information, see the
* Azure Docs.
*
* To avoid overwriting, pass "*" to {@link BlobRequestConditions#setIfNoneMatch(String)}.
*
*
Code Samples
*
*
*
* BlobHttpHeaders headers = new BlobHttpHeaders()
* .setContentLanguage("en-US")
* .setContentType("binary");
* BlobRequestConditions blobRequestConditions = new BlobRequestConditions().setLeaseId(leaseId);
*
* client.createWithResponse(size, sequenceNumber, headers, metadata, blobRequestConditions)
* .subscribe(response -> System.out.printf(
* "Created page blob with sequence number %s%n", response.getValue().getBlobSequenceNumber()));
*
*
*
*
* @param size Specifies the maximum size for the page blob, up to 8 TB. The page blob size must be aligned to a
* 512-byte boundary.
* @param sequenceNumber A user-controlled value that you can use to track requests. The value of the sequence
* number must be between 0 and 2^63 - 1.The default value is 0.
* @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 reactive response containing the information of the created page blob.
*
* @throws IllegalArgumentException If {@code size} isn't a multiple of {@link PageBlobAsyncClient#PAGE_BYTES} or
* {@code sequenceNumber} isn't null and is less than 0.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> createWithResponse(long size, Long sequenceNumber, BlobHttpHeaders headers,
Map metadata, BlobRequestConditions requestConditions) {
return this.createWithResponse(new PageBlobCreateOptions(size).setSequenceNumber(sequenceNumber)
.setHeaders(headers).setMetadata(metadata).setRequestConditions(requestConditions));
}
/**
* Creates a page blob of the specified length. Call PutPage to upload data to a page blob. For more
* information, see the
* Azure Docs.
*
* To avoid overwriting, pass "*" to {@link BlobRequestConditions#setIfNoneMatch(String)}.
*
*
Code Samples
*
*
*
* BlobHttpHeaders headers = new BlobHttpHeaders()
* .setContentLanguage("en-US")
* .setContentType("binary");
* BlobRequestConditions blobRequestConditions = new BlobRequestConditions().setLeaseId(leaseId);
*
* client.createWithResponse(new PageBlobCreateOptions(size).setSequenceNumber(sequenceNumber)
* .setHeaders(headers).setMetadata(metadata).setTags(tags).setRequestConditions(blobRequestConditions))
* .subscribe(response -> System.out.printf(
* "Created page blob with sequence number %s%n", response.getValue().getBlobSequenceNumber()));
*
*
*
*
* @param options {@link PageBlobCreateOptions}
* @return A reactive response containing the information of the created page blob.
*
* @throws IllegalArgumentException If {@code size} isn't a multiple of {@link PageBlobAsyncClient#PAGE_BYTES} or
* {@code sequenceNumber} isn't null and is less than 0.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> createWithResponse(PageBlobCreateOptions options) {
try {
return withContext(context -> createWithResponse(options, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> createWithResponse(PageBlobCreateOptions options, Context context) {
StorageImplUtils.assertNotNull("options", options);
BlobRequestConditions requestConditions = options.getRequestConditions() == null ? new BlobRequestConditions()
: options.getRequestConditions();
if (options.getSize() % PAGE_BYTES != 0) {
// Throwing is preferred to Single.error because this will error out immediately instead of waiting until
// subscription.
throw LOGGER.logExceptionAsError(
new IllegalArgumentException("size must be a multiple of PageBlobAsyncClient.PAGE_BYTES."));
}
if (options.getSequenceNumber() != null && options.getSequenceNumber() < 0) {
// Throwing is preferred to Single.error because this will error out immediately instead of waiting until
// subscription.
throw LOGGER.logExceptionAsError(
new IllegalArgumentException("SequenceNumber must be greater than or equal to 0."));
}
context = context == null ? Context.NONE : context;
BlobImmutabilityPolicy immutabilityPolicy = options.getImmutabilityPolicy() == null
? new BlobImmutabilityPolicy() : options.getImmutabilityPolicy();
return this.azureBlobStorage.getPageBlobs().createWithResponseAsync(containerName, blobName, 0,
options.getSize(), null, null, options.getMetadata(), requestConditions.getLeaseId(),
requestConditions.getIfModifiedSince(), requestConditions.getIfUnmodifiedSince(),
requestConditions.getIfMatch(), requestConditions.getIfNoneMatch(), requestConditions.getTagsConditions(),
options.getSequenceNumber(), null, tagsToString(options.getTags()),
immutabilityPolicy.getExpiryTime(), immutabilityPolicy.getPolicyMode(), options.isLegalHold(),
options.getHeaders(), getCustomerProvidedKey(), encryptionScope, context)
.map(rb -> {
PageBlobsCreateHeaders hd = rb.getDeserializedHeaders();
PageBlobItem item = new PageBlobItem(hd.getETag(), hd.getLastModified(), hd.getContentMD5(),
hd.isXMsRequestServerEncrypted(), hd.getXMsEncryptionKeySha256(), hd.getXMsEncryptionScope(),
null, hd.getXMsVersionId());
return new SimpleResponse<>(rb, item);
});
}
/**
* Creates a page blob of the specified length if it does not exist.
* Call PutPage to upload data to a page blob. For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* client.createIfNotExists(size).subscribe(response ->
* System.out.printf("Created page blob with sequence number %s%n", response.getBlobSequenceNumber()));
*
*
*
* @param size Specifies the maximum size for the page blob, up to 8 TB. The page blob size must be aligned to a
* 512-byte boundary.
*
* @return A reactive response {@link Mono} signaling completion. {@link PageBlobItem} contains information of
* the newly created page blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono createIfNotExists(long size) {
return createIfNotExistsWithResponse(new PageBlobCreateOptions(size)).flatMap(FluxUtil::toMono);
}
/**
* Creates a page blob of the specified length if it does not exist. Call PutPage to upload data to a page blob.
* For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* BlobHttpHeaders headers = new BlobHttpHeaders()
* .setContentLanguage("en-US")
* .setContentType("binary");
*
* client.createIfNotExistsWithResponse(new PageBlobCreateOptions(size).setSequenceNumber(sequenceNumber)
* .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 PageBlobCreateOptions}
* @return A {@link Mono} containing {@link Response} signaling completion, whose {@link Response#getValue() value}
* contains a {@link PageBlobItem} containing information about the page blob. If {@link Response}'s status code is
* 201, a new page blob was successfully created. If status code is 409, a page blob already existed at this location.
*
* @throws IllegalArgumentException If {@code size} isn't a multiple of {@link PageBlobAsyncClient#PAGE_BYTES} or
* {@code sequenceNumber} isn't null and is less than 0.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> createIfNotExistsWithResponse(PageBlobCreateOptions options) {
try {
return createIfNotExistsWithResponse(options, null);
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> createIfNotExistsWithResponse(PageBlobCreateOptions options, Context context) {
try {
options.setRequestConditions(new BlobRequestConditions().setIfNoneMatch(Constants.HeaderConstants.ETAG_WILDCARD)
.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);
}
}
/**
* Writes one or more pages to the page blob. Write size must be a multiple of 512. For more information, see
* the Azure Docs.
*
* 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.
*
*
Code Samples
*
*
*
* PageRange pageRange = new PageRange()
* .setStart(0)
* .setEnd(511);
*
* client.uploadPages(pageRange, body).subscribe(response -> System.out.printf(
* "Uploaded page blob with sequence number %s%n", response.getBlobSequenceNumber()));
*
*
*
* @param pageRange A {@link PageRange} object. Given that pages must be aligned with 512-byte boundaries, the start
* offset must be a modulus of 512 and the end offset must be a modulus of 512 - 1. Examples of valid byte ranges
* are 0-511, 512-1023, etc.
* @param body The data to upload. Note that this {@code Flux} must be replayable if retries are enabled (the
* default). In other words, the Flowable must produce the same data each time it is subscribed to.
*
* @return A reactive response containing the information of the uploaded pages.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono uploadPages(PageRange pageRange, Flux body) {
return uploadPagesWithResponse(pageRange, body, null, null).flatMap(FluxUtil::toMono);
}
/**
* Writes one or more pages to the page blob. Write size must be a multiple of 512. For more information, see
* the Azure Docs.
*
* 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.
*
*
Code Samples
*
*
*
* PageRange pageRange = new PageRange()
* .setStart(0)
* .setEnd(511);
*
* byte[] md5 = MessageDigest.getInstance("MD5").digest("data".getBytes(StandardCharsets.UTF_8));
* PageBlobRequestConditions pageBlobRequestConditions = new PageBlobRequestConditions().setLeaseId(leaseId);
*
* client.uploadPagesWithResponse(pageRange, body, md5, pageBlobRequestConditions)
* .subscribe(response -> System.out.printf(
* "Uploaded page blob with sequence number %s%n", response.getValue().getBlobSequenceNumber()));
*
*
*
* @param pageRange A {@link PageRange} object. Given that pages must be aligned with 512-byte boundaries, the start
* offset must be a modulus of 512 and the end offset must be a modulus of 512 - 1. Examples of valid byte ranges
* are 0-511, 512-1023, etc.
* @param body The data to upload. Note that this {@code Flux} must be replayable if retries are enabled (the
* default). In other words, the Flowable must produce the same data each time it is subscribed to.
* @param contentMd5 An MD5 hash of the page content. This hash is used to verify the integrity of the page 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 pageBlobRequestConditions {@link PageBlobRequestConditions}
* @return A reactive response containing the information of the uploaded pages.
*
* @throws IllegalArgumentException If {@code pageRange} is {@code null}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> uploadPagesWithResponse(PageRange pageRange, Flux body,
byte[] contentMd5, PageBlobRequestConditions pageBlobRequestConditions) {
if (body == null) {
return Mono.error(new NullPointerException("'body' cannot be null."));
}
try {
return withContext(context -> uploadPagesWithResponse(pageRange, body, contentMd5,
pageBlobRequestConditions, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> uploadPagesWithResponse(PageRange pageRange, Flux body, byte[] contentMd5,
PageBlobRequestConditions pageBlobRequestConditions, Context context) {
pageBlobRequestConditions = pageBlobRequestConditions == null
? new PageBlobRequestConditions()
: pageBlobRequestConditions;
if (pageRange == null) {
// Throwing is preferred to Single.error because this will error out immediately instead of waiting until
// subscription.
throw LOGGER.logExceptionAsError(new IllegalArgumentException("pageRange cannot be null."));
}
String pageRangeStr = pageRangeToString(pageRange);
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getPageBlobs().uploadPagesWithResponseAsync(containerName, blobName,
pageRange.getEnd() - pageRange.getStart() + 1, body, contentMd5, null, null, pageRangeStr,
pageBlobRequestConditions.getLeaseId(),
pageBlobRequestConditions.getIfSequenceNumberLessThanOrEqualTo(),
pageBlobRequestConditions.getIfSequenceNumberLessThan(),
pageBlobRequestConditions.getIfSequenceNumberEqualTo(), pageBlobRequestConditions.getIfModifiedSince(),
pageBlobRequestConditions.getIfUnmodifiedSince(), pageBlobRequestConditions.getIfMatch(),
pageBlobRequestConditions.getIfNoneMatch(), pageBlobRequestConditions.getTagsConditions(), null,
getCustomerProvidedKey(), encryptionScope, context)
.map(rb -> {
PageBlobsUploadPagesHeaders hd = rb.getDeserializedHeaders();
PageBlobItem item = new PageBlobItem(hd.getETag(), hd.getLastModified(), hd.getContentMD5(),
hd.isXMsRequestServerEncrypted(), hd.getXMsEncryptionKeySha256(), hd.getXMsEncryptionScope(),
hd.getXMsBlobSequenceNumber());
return new SimpleResponse<>(rb, item);
});
}
/**
* Writes one or more pages from the source page blob to this page blob. Write size must be a multiple of 512.
* For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* PageRange pageRange = new PageRange()
* .setStart(0)
* .setEnd(511);
*
* client.uploadPagesFromUrl(pageRange, url, sourceOffset)
* .subscribe(response -> System.out.printf(
* "Uploaded page blob from URL with sequence number %s%n", response.getBlobSequenceNumber()));
*
*
*
* @param range A {@link PageRange} object. Given that pages must be aligned with 512-byte boundaries, the start
* offset must be a modulus of 512 and the end offset must be a modulus of 512 - 1. Examples of valid byte ranges
* are 0-511, 512-1023, etc.
* @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 sourceOffset The source offset to copy from. Pass null or 0 to copy from the beginning of source page
* blob.
*
* @return A reactive response containing the information of the uploaded pages.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono uploadPagesFromUrl(PageRange range, String sourceUrl, Long sourceOffset) {
return uploadPagesFromUrlWithResponse(range, sourceUrl, sourceOffset, null, null, null)
.flatMap(FluxUtil::toMono);
}
/**
* Writes one or more pages from the source page blob to this page blob. Write size must be a multiple of 512.
* For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* PageRange pageRange = new PageRange()
* .setStart(0)
* .setEnd(511);
* InputStream dataStream = new ByteArrayInputStream(data.getBytes(StandardCharsets.UTF_8));
* byte[] sourceContentMD5 = new byte[512];
* PageBlobRequestConditions pageBlobRequestConditions = new PageBlobRequestConditions().setLeaseId(leaseId);
* BlobRequestConditions sourceRequestConditions = new BlobRequestConditions()
* .setIfUnmodifiedSince(OffsetDateTime.now().minusDays(3));
*
* client.uploadPagesFromUrlWithResponse(pageRange, url, sourceOffset, sourceContentMD5, pageBlobRequestConditions,
* sourceRequestConditions)
* .subscribe(response -> System.out.printf(
* "Uploaded page blob from URL with sequence number %s%n", response.getValue().getBlobSequenceNumber()));
*
*
*
* @param range The destination {@link PageRange} range. Given that pages must be aligned with 512-byte boundaries,
* the start offset must be a modulus of 512 and the end offset must be a modulus of 512 - 1. Examples of valid byte
* ranges are 0-511, 512-1023, etc.
* @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 sourceOffset The source offset to copy from. Pass null or 0 to copy from the beginning of source blob.
* @param sourceContentMd5 An MD5 hash of the page content. This hash is used to verify the integrity of the page
* 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 destRequestConditions {@link PageBlobRequestConditions}
* @param sourceRequestConditions {@link BlobRequestConditions}
* @return A reactive response containing the information of the uploaded pages.
*
* @throws IllegalArgumentException If {@code range} is {@code null}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> uploadPagesFromUrlWithResponse(PageRange range, String sourceUrl,
Long sourceOffset, byte[] sourceContentMd5, PageBlobRequestConditions destRequestConditions,
BlobRequestConditions sourceRequestConditions) {
return uploadPagesFromUrlWithResponse(new PageBlobUploadPagesFromUrlOptions(range, sourceUrl)
.setSourceOffset(sourceOffset).setSourceContentMd5(sourceContentMd5)
.setDestinationRequestConditions(destRequestConditions)
.setSourceRequestConditions(sourceRequestConditions));
}
/**
* Writes one or more pages from the source page blob to this page blob. Write size must be a multiple of 512.
* For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* PageRange pageRange = new PageRange()
* .setStart(0)
* .setEnd(511);
* InputStream dataStream = new ByteArrayInputStream(data.getBytes(StandardCharsets.UTF_8));
* byte[] sourceContentMD5 = new byte[512];
* PageBlobRequestConditions pageBlobRequestConditions = new PageBlobRequestConditions().setLeaseId(leaseId);
* BlobRequestConditions sourceRequestConditions = new BlobRequestConditions()
* .setIfUnmodifiedSince(OffsetDateTime.now().minusDays(3));
*
* client.uploadPagesFromUrlWithResponse(new PageBlobUploadPagesFromUrlOptions(pageRange, url)
* .setSourceOffset(sourceOffset).setSourceContentMd5(sourceContentMD5)
* .setDestinationRequestConditions(pageBlobRequestConditions)
* .setSourceRequestConditions(sourceRequestConditions))
* .subscribe(response -> System.out.printf(
* "Uploaded page blob from URL with sequence number %s%n", response.getValue().getBlobSequenceNumber()));
*
*
*
* @param options Parameters for the operation.
* @return A reactive response containing the information of the uploaded pages.
*
* @throws IllegalArgumentException If {@code range} is {@code null}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> uploadPagesFromUrlWithResponse(PageBlobUploadPagesFromUrlOptions options) {
try {
return withContext(context -> uploadPagesFromUrlWithResponse(options, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> uploadPagesFromUrlWithResponse(PageBlobUploadPagesFromUrlOptions options, Context context) {
if (options.getRange() == null) {
// Throwing is preferred to Single.error because this will error out immediately instead of waiting until
// subscription.
throw LOGGER.logExceptionAsError(new IllegalArgumentException("range cannot be null."));
}
String rangeString = pageRangeToString(options.getRange());
long sourceOffset = options.getSourceOffset() == null ? 0L : options.getSourceOffset();
String sourceRangeString = pageRangeToString(new PageRange()
.setStart(sourceOffset)
.setEnd(sourceOffset + (options.getRange().getEnd() - options.getRange().getStart())));
PageBlobRequestConditions destRequestConditions = (options.getDestinationRequestConditions() == null)
? new PageBlobRequestConditions() : options.getDestinationRequestConditions();
BlobRequestConditions sourceRequestConditions = (options.getSourceRequestConditions() == null)
? new BlobRequestConditions() : 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.getPageBlobs().uploadPagesFromURLWithResponseAsync(
containerName, blobName, options.getSourceUrl(), sourceRangeString, 0, rangeString, options.getSourceContentMd5(), null, null,
destRequestConditions.getLeaseId(), destRequestConditions.getIfSequenceNumberLessThanOrEqualTo(),
destRequestConditions.getIfSequenceNumberLessThan(), destRequestConditions.getIfSequenceNumberEqualTo(),
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 -> {
PageBlobsUploadPagesFromURLHeaders hd = rb.getDeserializedHeaders();
PageBlobItem item = new PageBlobItem(hd.getETag(), hd.getLastModified(), hd.getContentMD5(),
hd.isXMsRequestServerEncrypted(), hd.getXMsEncryptionKeySha256(), hd.getXMsEncryptionScope(), null);
return new SimpleResponse<>(rb, item);
});
}
/**
* Frees the specified pages from the page blob. The size of the range must be a multiple of 512. For more
* information, see the Azure Docs.
*
* Code Samples
*
*
*
* PageRange pageRange = new PageRange()
* .setStart(0)
* .setEnd(511);
*
* client.clearPages(pageRange).subscribe(response -> System.out.printf(
* "Cleared page blob with sequence number %s%n", response.getBlobSequenceNumber()));
*
*
*
* @param pageRange A {@link PageRange} object. Given that pages must be aligned with 512-byte boundaries, the start
* offset must be a modulus of 512 and the end offset must be a modulus of 512 - 1. Examples of valid byte ranges
* are 0-511, 512-1023, etc.
*
* @return A reactive response containing the information of the cleared pages.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono clearPages(PageRange pageRange) {
return clearPagesWithResponse(pageRange, null).flatMap(FluxUtil::toMono);
}
/**
* Frees the specified pages from the page blob. The size of the range must be a multiple of 512. For more
* information, see the Azure Docs.
*
* Code Samples
*
*
*
* PageRange pageRange = new PageRange()
* .setStart(0)
* .setEnd(511);
* PageBlobRequestConditions pageBlobRequestConditions = new PageBlobRequestConditions().setLeaseId(leaseId);
*
* client.clearPagesWithResponse(pageRange, pageBlobRequestConditions)
* .subscribe(response -> System.out.printf(
* "Cleared page blob with sequence number %s%n", response.getValue().getBlobSequenceNumber()));
*
*
*
* @param pageRange A {@link PageRange} object. Given that pages must be aligned with 512-byte boundaries, the start
* offset must be a modulus of 512 and the end offset must be a modulus of 512 - 1. Examples of valid byte ranges
* are 0-511, 512-1023, etc.
* @param pageBlobRequestConditions {@link PageBlobRequestConditions}
* @return A reactive response containing the information of the cleared pages.
*
* @throws IllegalArgumentException If {@code pageRange} is {@code null}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> clearPagesWithResponse(PageRange pageRange,
PageBlobRequestConditions pageBlobRequestConditions) {
try {
return withContext(context -> clearPagesWithResponse(pageRange, pageBlobRequestConditions, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> clearPagesWithResponse(PageRange pageRange,
PageBlobRequestConditions pageBlobRequestConditions, Context context) {
pageBlobRequestConditions = pageBlobRequestConditions == null
? new PageBlobRequestConditions()
: pageBlobRequestConditions;
if (pageRange == null) {
// Throwing is preferred to Single.error because this will error out immediately instead of waiting until
// subscription.
throw LOGGER.logExceptionAsError(new IllegalArgumentException("pageRange cannot be null."));
}
String pageRangeStr = pageRangeToString(pageRange);
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getPageBlobs().clearPagesWithResponseAsync(containerName, blobName, 0,
null, pageRangeStr, pageBlobRequestConditions.getLeaseId(),
pageBlobRequestConditions.getIfSequenceNumberLessThanOrEqualTo(),
pageBlobRequestConditions.getIfSequenceNumberLessThan(),
pageBlobRequestConditions.getIfSequenceNumberEqualTo(), pageBlobRequestConditions.getIfModifiedSince(),
pageBlobRequestConditions.getIfUnmodifiedSince(), pageBlobRequestConditions.getIfMatch(),
pageBlobRequestConditions.getIfNoneMatch(), pageBlobRequestConditions.getTagsConditions(), null,
getCustomerProvidedKey(), encryptionScope, context)
.map(rb -> {
PageBlobsClearPagesHeaders hd = rb.getDeserializedHeaders();
PageBlobItem item = new PageBlobItem(hd.getETag(), hd.getLastModified(), hd.getContentMD5(),
hd.isXMsRequestServerEncrypted(), hd.getXMsEncryptionKeySha256(), null,
hd.getXMsBlobSequenceNumber());
return new SimpleResponse<>(rb, item);
});
}
/**
* Returns the list of valid page ranges for a page blob or snapshot of a page blob. For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* BlobRange blobRange = new BlobRange(offset);
*
* client.getPageRanges(blobRange).subscribe(response -> {
* System.out.println("Valid Page Ranges are:");
* for (PageRange pageRange : response.getPageRange()) {
* System.out.printf("Start: %s, End: %s%n", pageRange.getStart(), pageRange.getEnd());
* }
* });
*
*
*
* @param blobRange {@link BlobRange}
*
* @return A reactive response containing the information of the cleared pages.
* @deprecated Use {@link #listPageRanges(BlobRange)}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
@Deprecated
public Mono getPageRanges(BlobRange blobRange) {
return getPageRangesWithResponse(blobRange, null).flatMap(FluxUtil::toMono);
}
/**
* Returns the list of valid page ranges for a page blob or snapshot of a page blob. For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* BlobRange blobRange = new BlobRange(offset);
* BlobRequestConditions blobRequestConditions = new BlobRequestConditions().setLeaseId(leaseId);
*
* client.getPageRangesWithResponse(blobRange, blobRequestConditions)
* .subscribe(response -> {
* System.out.println("Valid Page Ranges are:");
* for (PageRange pageRange : response.getValue().getPageRange()) {
* System.out.printf("Start: %s, End: %s%n", pageRange.getStart(), pageRange.getEnd());
* }
* });
*
*
*
* @param blobRange {@link BlobRange}
* @param requestConditions {@link BlobRequestConditions}
* @return A reactive response emitting all the page ranges.
* @deprecated Use {@link #listPageRanges(ListPageRangesOptions)}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
@Deprecated
public Mono> getPageRangesWithResponse(BlobRange blobRange,
BlobRequestConditions requestConditions) {
try {
return withContext(context -> getPageRangesWithResponse(blobRange, requestConditions, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> getPageRangesWithResponse(BlobRange blobRange, BlobRequestConditions requestConditions,
Context context) {
blobRange = blobRange == null ? new BlobRange(0) : blobRange;
requestConditions = requestConditions == null ? new BlobRequestConditions() : requestConditions;
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getPageBlobs().getPageRangesWithResponseAsync(containerName, blobName,
getSnapshotId(), null, blobRange.toHeaderValue(), requestConditions.getLeaseId(),
requestConditions.getIfModifiedSince(), requestConditions.getIfUnmodifiedSince(),
requestConditions.getIfMatch(), requestConditions.getIfNoneMatch(),
requestConditions.getTagsConditions(), null, null, null, context)
.map(response -> new SimpleResponse<>(response.getRequest(), response.getStatusCode(),
response.getHeaders(), response.getValue()));
}
/**
* Returns the list of valid page ranges for a page blob or snapshot of a page blob. For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* BlobRange blobRange = new BlobRange(offset);
*
* System.out.println("Valid Page Ranges are:");
* client.listPageRanges(blobRange).subscribe(rangeItem -> System.out.printf("Offset: %s, Length: %s%n",
* rangeItem.getRange().getOffset(), rangeItem.getRange().getLength()));
*
*
*
* @param blobRange {@link BlobRange}
*
* @return A reactive response containing the information of the cleared pages.
*/
@ServiceMethod(returns = ReturnType.COLLECTION)
public PagedFlux listPageRanges(BlobRange blobRange) {
return listPageRanges(new ListPageRangesOptions(blobRange));
}
/**
* Returns the list of valid page ranges for a page blob or snapshot of a page blob. For more information, see the
* Azure Docs.
*
* Code Samples
*
*
*
* ListPageRangesOptions options = new ListPageRangesOptions(new BlobRange(offset))
* .setMaxResultsPerPage(1000).setRequestConditions(new BlobRequestConditions().setLeaseId(leaseId));
*
* client.listPageRanges(options)
* .subscribe(rangeItem -> System.out.printf("Offset: %s, Length: %s%n", rangeItem.getRange().getOffset(),
* rangeItem.getRange().getLength()));
*
*
*
* @param options {@link ListPageRangesOptions}
* @return A reactive response emitting all the page ranges.
*/
@ServiceMethod(returns = ReturnType.COLLECTION)
public PagedFlux listPageRanges(ListPageRangesOptions options) {
return new PagedFlux<>(
pageSize -> withContext(context ->
listPageRangesWithOptionalTimeout(options, null, context).apply(null, pageSize)),
(continuationToken, pageSize) -> withContext(context ->
listPageRangesWithOptionalTimeout(options, null, context).apply(continuationToken, pageSize)));
}
/*
* Implementation for this paged listing operation, supporting an optional timeout provided by the synchronous
* ContainerClient. Applies the given timeout to each Mono backing the
* PagedFlux.
*
* @param delimiter The delimiter for blob hierarchy, "/" for hierarchy based on directories
* @param options {@link PageBlobGetPageRangesOptions}
* @param timeout An optional timeout to be applied to the network asynchronous operations.
* @return A reactive response emitting the listed blobs, flattened.
*/
BiFunction>> listPageRangesWithOptionalTimeout(
ListPageRangesOptions options, Duration timeout, Context context) {
return (marker, pageSize) -> {
ListPageRangesOptions finalOptions;
/*
If pageSize was not set in a .byPage(int) method, the page size from options will be preserved.
Otherwise, prefer the new value.
*/
if (pageSize != null) {
finalOptions = new ListPageRangesOptions(options.getRange()).setMaxResultsPerPage(pageSize);
} else {
finalOptions = options;
}
return getPageRangesSegment(marker, finalOptions, timeout, context)
.map(response -> {
List value = response.getValue() == null
? Collections.emptyList()
: Stream.concat(
response.getValue().getPageRange().stream().map(PageBlobAsyncClient::toPageBlobRange),
response.getValue().getClearRange().stream().map(PageBlobAsyncClient::toPageBlobRange)
).collect(Collectors.toList());
return new PagedResponseBase<>(
response.getRequest(),
response.getStatusCode(),
response.getHeaders(),
value,
PageListHelper.getNextMarker(response.getValue()),
response.getDeserializedHeaders());
});
};
}
private Mono> getPageRangesSegment(String marker,
ListPageRangesOptions options, Duration timeout, Context context) {
BlobRequestConditions requestConditions = options.getRequestConditions() == null ? new BlobRequestConditions()
: options.getRequestConditions();
context = context == null ? Context.NONE : context;
return StorageImplUtils.applyOptionalTimeout(
this.azureBlobStorage.getPageBlobs().getPageRangesWithResponseAsync(containerName, blobName,
getSnapshotId(), null, options.getRange().toHeaderValue(), requestConditions.getLeaseId(),
requestConditions.getIfModifiedSince(), requestConditions.getIfUnmodifiedSince(),
requestConditions.getIfMatch(), requestConditions.getIfNoneMatch(),
requestConditions.getTagsConditions(), null, marker, options.getMaxResultsPerPage(),
context),
timeout);
}
private static PageRangeItem toPageBlobRange(PageRange range) {
return new PageRangeItem(new HttpRange(range.getStart(), range.getEnd() - range.getStart() + 1), false);
}
private static PageRangeItem toPageBlobRange(ClearRange range) {
return new PageRangeItem(new HttpRange(range.getStart(), range.getEnd() - range.getStart() + 1), true);
}
/**
* Gets the collection of page ranges that differ between a specified snapshot and this page blob. For more
* information, see the Azure
* Docs.
*
* Code Samples
*
*
*
* BlobRange blobRange = new BlobRange(offset);
* final String prevSnapshot = "previous snapshot";
*
* client.getPageRangesDiff(blobRange, prevSnapshot).subscribe(response -> {
* System.out.println("Valid Page Ranges are:");
* for (PageRange pageRange : response.getPageRange()) {
* System.out.printf("Start: %s, End: %s%n", pageRange.getStart(), pageRange.getEnd());
* }
* });
*
*
*
* @param blobRange {@link BlobRange}
* @param prevSnapshot Specifies that the response will contain only pages that were changed between target blob and
* previous snapshot. Changed pages include both updated and cleared pages. The target blob may be a snapshot, as
* long as the snapshot specified by prevsnapshot is the older of the two.
*
* @return A reactive response emitting all the different page ranges.
* @deprecated See {@link #listPageRangesDiff(BlobRange, String)}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
@Deprecated
public Mono getPageRangesDiff(BlobRange blobRange, String prevSnapshot) {
return getPageRangesDiffWithResponse(blobRange, prevSnapshot, null).flatMap(FluxUtil::toMono);
}
/**
* Gets the collection of page ranges that differ between a specified snapshot and this page blob. For more
* information, see the Azure
* Docs.
*
* Code Samples
*
*
*
* BlobRange blobRange = new BlobRange(offset);
* final String prevSnapshot = "previous snapshot";
* BlobRequestConditions blobRequestConditions = new BlobRequestConditions().setLeaseId(leaseId);
*
* client.getPageRangesDiffWithResponse(blobRange, prevSnapshot, blobRequestConditions)
* .subscribe(response -> {
* System.out.println("Valid Page Ranges are:");
* for (PageRange pageRange : response.getValue().getPageRange()) {
* System.out.printf("Start: %s, End: %s%n", pageRange.getStart(), pageRange.getEnd());
* }
* });
*
*
*
* @param blobRange {@link BlobRange}
* @param prevSnapshot Specifies that the response will contain only pages that were changed between target blob and
* previous snapshot. Changed pages include both updated and cleared pages. The target blob may be a snapshot, as
* long as the snapshot specified by prevsnapshot is the older of the two.
* @param requestConditions {@link BlobRequestConditions}
* @return A reactive response emitting all the different page ranges.
*
* @throws IllegalArgumentException If {@code prevSnapshot} is {@code null}
* @deprecated See {@link #listPageRangesDiff(ListPageRangesDiffOptions)}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
@Deprecated
public Mono> getPageRangesDiffWithResponse(BlobRange blobRange, String prevSnapshot,
BlobRequestConditions requestConditions) {
try {
return withContext(context -> getPageRangesDiffWithResponse(blobRange, prevSnapshot, null,
requestConditions, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
/**
* Gets the collection of page ranges that differ between a specified snapshot and this page blob. For more
* information, see the Azure
* Docs.
*
* Code Samples
*
*
*
* BlobRange blobRange = new BlobRange(offset);
* String prevSnapshot = "previous snapshot";
*
* System.out.println("Valid Page Ranges are:");
* client.listPageRangesDiff(blobRange, prevSnapshot).subscribe(rangeItem ->
* System.out.printf("Offset: %s, Length: %s, isClear: %s%n",
* rangeItem.getRange().getOffset(), rangeItem.getRange().getLength(), rangeItem.isClear()));
*
*
*
* @param blobRange {@link BlobRange}
* @param prevSnapshot Specifies that the response will contain only pages that were changed between target blob and
* previous snapshot. Changed pages include both updated and cleared pages. The target blob may be a snapshot, as
* long as the snapshot specified by prevsnapshot is the older of the two.
*
* @return A reactive response emitting all the different page ranges.
*/
@ServiceMethod(returns = ReturnType.COLLECTION)
public PagedFlux listPageRangesDiff(BlobRange blobRange, String prevSnapshot) {
return listPageRangesDiff(new ListPageRangesDiffOptions(blobRange, prevSnapshot));
}
/**
* Gets the collection of page ranges that differ between a specified snapshot and this page blob. For more
* information, see the Azure
* Docs.
*
* Code Samples
*
*
*
* ListPageRangesDiffOptions options = new ListPageRangesDiffOptions(new BlobRange(offset), "previous snapshot")
* .setRequestConditions(new BlobRequestConditions().setLeaseId(leaseId))
* .setMaxResultsPerPage(1000);
*
* client.listPageRangesDiff(options)
* .subscribe(rangeItem -> System.out.printf("Offset: %s, Length: %s, isClear: %s%n",
* rangeItem.getRange().getOffset(), rangeItem.getRange().getLength(), rangeItem.isClear()));
*
*
*
* @param options {@link ListPageRangesDiffOptions}.
* @return A reactive response emitting all the different page ranges.
*
* @throws IllegalArgumentException If {@code prevSnapshot} is {@code null}
*/
@ServiceMethod(returns = ReturnType.COLLECTION)
public PagedFlux listPageRangesDiff(ListPageRangesDiffOptions options) {
return new PagedFlux<>(
pageSize -> withContext(context ->
listPageRangesDiffWithOptionalTimeout(options, null, context).apply(null, pageSize)),
(continuationToken, pageSize) -> withContext(context ->
listPageRangesDiffWithOptionalTimeout(options, null, context).apply(continuationToken, pageSize)));
}
/**
* This API only works for managed disk accounts.
* Gets the collection of page ranges that differ between a specified snapshot and this page blob. For more
* information, see the Azure
* Docs.
*
* Code Samples
*
*
*
* BlobRange blobRange = new BlobRange(offset);
* final String prevSnapshotUrl = "previous snapshot url";
*
* client.getPageRangesDiff(blobRange, prevSnapshotUrl).subscribe(response -> {
* System.out.println("Valid Page Ranges are:");
* for (PageRange pageRange : response.getPageRange()) {
* System.out.printf("Start: %s, End: %s%n", pageRange.getStart(), pageRange.getEnd());
* }
* });
*
*
*
* @param blobRange {@link BlobRange}
* @param prevSnapshotUrl Specifies the URL of a previous snapshot of the target blob. Specifies that the
* response will contain only pages that were changed between target blob and previous snapshot. Changed pages
* include both updated and cleared pages. The target blob may be a snapshot, as long as the snapshot specified by
* prevsnapshot is the older of the two.
*
* @return A reactive response emitting all the different page ranges.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono getManagedDiskPageRangesDiff(BlobRange blobRange, String prevSnapshotUrl) {
return getManagedDiskPageRangesDiffWithResponse(blobRange, prevSnapshotUrl, null).flatMap(FluxUtil::toMono);
}
/**
* This API only works for managed disk accounts.
* Gets the collection of page ranges that differ between a specified snapshot and this page blob. For more
* information, see the Azure
* Docs.
*
* Code Samples
*
*
*
* BlobRange blobRange = new BlobRange(offset);
* final String prevSnapshotUrl = "previous snapshot url";
* BlobRequestConditions blobRequestConditions = new BlobRequestConditions().setLeaseId(leaseId);
*
* client.getPageRangesDiffWithResponse(blobRange, prevSnapshotUrl, blobRequestConditions)
* .subscribe(response -> {
* System.out.println("Valid Page Ranges are:");
* for (PageRange pageRange : response.getValue().getPageRange()) {
* System.out.printf("Start: %s, End: %s%n", pageRange.getStart(), pageRange.getEnd());
* }
* });
*
*
*
* @param blobRange {@link BlobRange}
* @param prevSnapshotUrl Specifies the URL of a previous snapshot of the target blob. Specifies that the
* response will contain only pages that were changed between target blob and previous snapshot. Changed pages
* include both updated and cleared pages. The target blob may be a snapshot, as long as the snapshot specified by
* prevsnapshot is the older of the two.
* @param requestConditions {@link BlobRequestConditions}
* @return A reactive response emitting all the different page ranges.
*
* @throws IllegalArgumentException If {@code prevSnapshot} is {@code null}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> getManagedDiskPageRangesDiffWithResponse(BlobRange blobRange,
String prevSnapshotUrl, BlobRequestConditions requestConditions) {
try {
return withContext(context ->
getPageRangesDiffWithResponse(blobRange, null, prevSnapshotUrl, requestConditions, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> getPageRangesDiffWithResponse(BlobRange blobRange, String prevSnapshot,
String prevSnapshotUrl, BlobRequestConditions requestConditions, Context context) {
blobRange = blobRange == null ? new BlobRange(0) : blobRange;
requestConditions = requestConditions == null ? new BlobRequestConditions() : requestConditions;
if (prevSnapshotUrl == null && prevSnapshot == null) {
throw LOGGER.logExceptionAsError(new IllegalArgumentException("prevSnapshot cannot be null"));
}
if (prevSnapshotUrl != null) {
try {
new URL(prevSnapshotUrl);
} catch (MalformedURLException ex) {
throw LOGGER.logExceptionAsError(new IllegalArgumentException("'prevSnapshotUrl' is not a valid url.", ex));
}
}
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getPageBlobs().getPageRangesDiffWithResponseAsync(containerName, blobName,
getSnapshotId(), null, prevSnapshot, prevSnapshotUrl, blobRange.toHeaderValue(),
requestConditions.getLeaseId(), requestConditions.getIfModifiedSince(),
requestConditions.getIfUnmodifiedSince(), requestConditions.getIfMatch(),
requestConditions.getIfNoneMatch(), requestConditions.getTagsConditions(), null, null, null, context)
.map(response -> new SimpleResponse<>(response.getRequest(), response.getStatusCode(),
response.getHeaders(), response.getValue()));
}
/*
* Implementation for this paged listing operation, supporting an optional timeout provided by the synchronous
* ContainerClient. Applies the given timeout to each Mono backing the
* PagedFlux.
*
* @param delimiter The delimiter for blob hierarchy, "/" for hierarchy based on directories
* @param options {@link PageBlobGetPageRangesDiffOptions}
* @param timeout An optional timeout to be applied to the network asynchronous operations.
* @return A reactive response emitting the listed blobs, flattened.
*/
BiFunction>> listPageRangesDiffWithOptionalTimeout(ListPageRangesDiffOptions options,
Duration timeout, Context context) {
return (marker, pageSize) -> {
ListPageRangesDiffOptions finalOptions;
/*
If pageSize was not set in a .byPage(int) method, the page size from options will be preserved.
Otherwise, prefer the new value.
*/
if (pageSize != null) {
finalOptions =
new ListPageRangesDiffOptions(options.getRange(), options.getPreviousSnapshot())
.setRequestConditions(options.getRequestConditions()).setMaxResultsPerPage(pageSize);
} else {
finalOptions = options;
}
return getPageRangesDiffSegment(marker, finalOptions, timeout, context)
.map(response -> {
List value = response.getValue() == null
? Collections.emptyList()
: Stream.concat(
response.getValue().getPageRange().stream().map(PageBlobAsyncClient::toPageBlobRange),
response.getValue().getClearRange().stream().map(PageBlobAsyncClient::toPageBlobRange))
.collect(Collectors.toList());
return new PagedResponseBase<>(
response.getRequest(),
response.getStatusCode(),
response.getHeaders(),
value,
PageListHelper.getNextMarker(response.getValue()),
response.getDeserializedHeaders());
});
};
}
private Mono> getPageRangesDiffSegment(String marker,
ListPageRangesDiffOptions options, Duration timeout, Context context) {
BlobRequestConditions requestConditions = options.getRequestConditions() == null ? new BlobRequestConditions()
: options.getRequestConditions();
context = context == null ? Context.NONE : context;
return StorageImplUtils.applyOptionalTimeout(
this.azureBlobStorage.getPageBlobs().getPageRangesDiffWithResponseAsync(containerName, blobName,
getSnapshotId(), null, options.getPreviousSnapshot(), null,
options.getRange().toHeaderValue(), requestConditions.getLeaseId(),
requestConditions.getIfModifiedSince(), requestConditions.getIfUnmodifiedSince(),
requestConditions.getIfMatch(), requestConditions.getIfNoneMatch(),
requestConditions.getTagsConditions(), null, marker, options.getMaxResultsPerPage(), context),
timeout);
}
/**
* Resizes the page blob to the specified size (which must be a multiple of 512). For more information, see the Azure Docs.
*
* Code Samples
*
*
*
* client.resize(size).subscribe(response -> System.out.printf(
* "Page blob resized with sequence number %s%n", response.getBlobSequenceNumber()));
*
*
*
* @param size Resizes a page blob to the specified size. If the specified value is less than the current size of
* the blob, then all pages above the specified value are cleared.
*
* @return A reactive response emitting the resized page blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono resize(long size) {
return resizeWithResponse(size, null).flatMap(FluxUtil::toMono);
}
/**
* Resizes the page blob to the specified size (which must be a multiple of 512). For more information, see the Azure Docs.
*
* Code Samples
*
*
*
* BlobRequestConditions blobRequestConditions = new BlobRequestConditions().setLeaseId(leaseId);
*
* client.resizeWithResponse(size, blobRequestConditions)
* .subscribe(response -> System.out.printf(
* "Page blob resized with sequence number %s%n", response.getValue().getBlobSequenceNumber()));
*
*
*
* @param size Resizes a page blob to the specified size. If the specified value is less than the current size of
* the blob, then all pages above the specified value are cleared.
* @param requestConditions {@link BlobRequestConditions}
* @return A reactive response emitting the resized page blob.
*
* @throws IllegalArgumentException If {@code size} isn't a multiple of {@link PageBlobAsyncClient#PAGE_BYTES}
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> resizeWithResponse(long size, BlobRequestConditions requestConditions) {
try {
return withContext(context -> resizeWithResponse(size, requestConditions, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> resizeWithResponse(long size, BlobRequestConditions requestConditions,
Context context) {
if (size % PAGE_BYTES != 0) {
// Throwing is preferred to Single.error because this will error out immediately instead of waiting until
// subscription.
throw LOGGER.logExceptionAsError(
new IllegalArgumentException("size must be a multiple of PageBlobAsyncClient.PAGE_BYTES."));
}
requestConditions = requestConditions == null ? new BlobRequestConditions() : requestConditions;
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getPageBlobs().resizeWithResponseAsync(containerName, blobName, size, null,
requestConditions.getLeaseId(), requestConditions.getIfModifiedSince(),
requestConditions.getIfUnmodifiedSince(), requestConditions.getIfMatch(),
requestConditions.getIfNoneMatch(), requestConditions.getTagsConditions(), null,
getCustomerProvidedKey(), encryptionScope, context)
.map(rb -> {
PageBlobsResizeHeaders hd = rb.getDeserializedHeaders();
PageBlobItem item = new PageBlobItem(hd.getETag(), hd.getLastModified(), null, null, null, null,
hd.getXMsBlobSequenceNumber());
return new SimpleResponse<>(rb, item);
});
}
/**
* Sets the page blob's sequence number. For more information, see the
* Azure
* Docs.
*
* Code Samples
*
*
*
* client.updateSequenceNumber(SequenceNumberActionType.INCREMENT, size)
* .subscribe(response -> System.out.printf(
* "Page blob updated to sequence number %s%n", response.getBlobSequenceNumber()));
*
*
*
* @param action Indicates how the service should modify the blob's sequence number.
* @param sequenceNumber The blob's sequence number. The sequence number is a user-controlled property that you can
* use to track requests and manage concurrency issues.
*
* @return A reactive response emitting the updated page blob.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono updateSequenceNumber(SequenceNumberActionType action, Long sequenceNumber) {
return updateSequenceNumberWithResponse(action, sequenceNumber, null).flatMap(FluxUtil::toMono);
}
/**
* Sets the page blob's sequence number. For more information, see the Azure
* Docs.
*
* Code Samples
*
*
*
* BlobRequestConditions blobRequestConditions = new BlobRequestConditions().setLeaseId(leaseId);
*
* client.updateSequenceNumberWithResponse(SequenceNumberActionType.INCREMENT, size, blobRequestConditions)
* .subscribe(response -> System.out.printf(
* "Page blob updated to sequence number %s%n", response.getValue().getBlobSequenceNumber()));
*
*
*
* @param action Indicates how the service should modify the blob's sequence number.
* @param sequenceNumber The blob's sequence number. The sequence number is a user-controlled property that you can
* use to track requests and manage concurrency issues.
* @param requestConditions {@link BlobRequestConditions}
* @return A reactive response emitting the updated page blob.
*
* @throws IllegalArgumentException If {@code sequenceNumber} isn't null and is less than 0
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> updateSequenceNumberWithResponse(SequenceNumberActionType action,
Long sequenceNumber, BlobRequestConditions requestConditions) {
try {
return withContext(context -> updateSequenceNumberWithResponse(action, sequenceNumber, requestConditions,
context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> updateSequenceNumberWithResponse(SequenceNumberActionType action, Long sequenceNumber,
BlobRequestConditions requestConditions, Context context) {
if (sequenceNumber != null && sequenceNumber < 0) {
// Throwing is preferred to Single.error because this will error out immediately instead of waiting until
// subscription.
throw LOGGER.logExceptionAsError(
new IllegalArgumentException("SequenceNumber must be greater than or equal to 0."));
}
requestConditions = requestConditions == null ? new BlobRequestConditions() : requestConditions;
sequenceNumber = action == SequenceNumberActionType.INCREMENT ? null : sequenceNumber;
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getPageBlobs().updateSequenceNumberWithResponseAsync(containerName, blobName, action, null,
requestConditions.getLeaseId(), requestConditions.getIfModifiedSince(),
requestConditions.getIfUnmodifiedSince(), requestConditions.getIfMatch(),
requestConditions.getIfNoneMatch(), requestConditions.getTagsConditions(), sequenceNumber, null, context)
.map(rb -> {
PageBlobsUpdateSequenceNumberHeaders hd = rb.getDeserializedHeaders();
PageBlobItem item = new PageBlobItem(hd.getETag(), hd.getLastModified(), null, null, null, null,
hd.getXMsBlobSequenceNumber());
return new SimpleResponse<>(rb, item);
});
}
/**
* Begins an operation to start an incremental copy from one page blob's snapshot to this page blob. The snapshot
* is
* copied such that only the differential changes between the previously copied snapshot are transferred to the
* destination. The copied snapshots are complete copies of the original snapshot and can be read or copied from as
* usual. For more information, see the Azure Docs
* here
* and
* here.
*
* Code Samples
*
*
*
* final String snapshot = "copy snapshot";
* client.copyIncremental(url, snapshot).subscribe(statusType -> {
* switch (statusType) {
* case SUCCESS:
* System.out.println("Page blob copied successfully");
* break;
* case FAILED:
* System.out.println("Page blob copied failed");
* break;
* case ABORTED:
* System.out.println("Page blob copied aborted");
* break;
* case PENDING:
* System.out.println("Page blob copied pending");
* break;
* default:
* break;
* }
* });
*
*
*
* @param source The source page blob.
* @param snapshot The snapshot on the copy source.
*
* @return A reactive response emitting the copy status.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono copyIncremental(String source, String snapshot) {
return copyIncrementalWithResponse(source, snapshot, null).flatMap(FluxUtil::toMono);
}
/**
* Begins an operation to start an incremental copy from one page blob's snapshot to this page blob. The snapshot
* is
* copied such that only the differential changes between the previously copied snapshot are transferred to the
* destination. The copied snapshots are complete copies of the original snapshot and can be read or copied from as
* usual. For more information, see the Azure Docs
* here
* and
* here.
*
* Code Samples
*
*
*
* final String snapshot = "copy snapshot";
* RequestConditions modifiedRequestConditions = new RequestConditions()
* .setIfNoneMatch("snapshotMatch");
*
* client.copyIncrementalWithResponse(url, snapshot, modifiedRequestConditions)
* .subscribe(response -> {
* CopyStatusType statusType = response.getValue();
*
* switch (statusType) {
* case SUCCESS:
* System.out.println("Page blob copied successfully");
* break;
* case FAILED:
* System.out.println("Page blob copied failed");
* break;
* case ABORTED:
* System.out.println("Page blob copied aborted");
* break;
* case PENDING:
* System.out.println("Page blob copied pending");
* break;
* default:
* break;
* }
* });
*
*
*
* @param source The source page blob.
* @param snapshot The snapshot on the copy source.
* @param modifiedRequestConditions Standard HTTP Access conditions related to the modification of data. ETag and
* LastModifiedTime are used to construct conditions related to when the blob was changed relative to the given
* request. The request will fail if the specified condition is not satisfied.
*
* @return A reactive response emitting the copy status.
*
* @throws IllegalStateException If {@code source} and {@code snapshot} form a malformed URL.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> copyIncrementalWithResponse(String source, String snapshot,
RequestConditions modifiedRequestConditions) {
return copyIncrementalWithResponse(new PageBlobCopyIncrementalOptions(source, snapshot)
.setRequestConditions(ModelHelper.populateBlobDestinationRequestConditions(modifiedRequestConditions)));
}
/**
* Begins an operation to start an incremental copy from one page blob's snapshot to this page blob. The snapshot
* is
* copied such that only the differential changes between the previously copied snapshot are transferred to the
* destination. The copied snapshots are complete copies of the original snapshot and can be read or copied from as
* usual. For more information, see the Azure Docs
* here
* and
* here.
*
* Code Samples
*
*
*
* final String snapshot = "copy snapshot";
* PageBlobCopyIncrementalRequestConditions destinationRequestConditions = new PageBlobCopyIncrementalRequestConditions()
* .setIfNoneMatch("snapshotMatch");
*
* client.copyIncrementalWithResponse(new PageBlobCopyIncrementalOptions(url, snapshot)
* .setRequestConditions(destinationRequestConditions))
* .subscribe(response -> {
* CopyStatusType statusType = response.getValue();
*
* switch (statusType) {
* case SUCCESS:
* System.out.println("Page blob copied successfully");
* break;
* case FAILED:
* System.out.println("Page blob copied failed");
* break;
* case ABORTED:
* System.out.println("Page blob copied aborted");
* break;
* case PENDING:
* System.out.println("Page blob copied pending");
* break;
* default:
* break;
* }
* });
*
*
*
* @param options {@link PageBlobCopyIncrementalOptions}
*
* @return A reactive response emitting the copy status.
*
* @throws IllegalStateException If {@code source} and {@code snapshot} form a malformed URL.
*/
@ServiceMethod(returns = ReturnType.SINGLE)
public Mono> copyIncrementalWithResponse(PageBlobCopyIncrementalOptions options) {
try {
return withContext(context -> copyIncrementalWithResponse(options, context));
} catch (RuntimeException ex) {
return monoError(LOGGER, ex);
}
}
Mono> copyIncrementalWithResponse(PageBlobCopyIncrementalOptions options,
Context context) {
StorageImplUtils.assertNotNull("options", options);
UrlBuilder builder = UrlBuilder.parse(options.getSource());
builder.setQueryParameter(Constants.UrlConstants.SNAPSHOT_QUERY_PARAMETER, options.getSnapshot());
PageBlobCopyIncrementalRequestConditions modifiedRequestConditions = (options.getRequestConditions() == null)
? new PageBlobCopyIncrementalRequestConditions() : options.getRequestConditions();
try {
builder.toUrl();
} catch (MalformedURLException e) {
// We are parsing a valid url and adding a query parameter. If this fails, we can't recover.
throw LOGGER.logExceptionAsError(new IllegalArgumentException(e));
}
context = context == null ? Context.NONE : context;
return this.azureBlobStorage.getPageBlobs().copyIncrementalWithResponseAsync(containerName, blobName,
builder.toString(), null, modifiedRequestConditions.getIfModifiedSince(),
modifiedRequestConditions.getIfUnmodifiedSince(), modifiedRequestConditions.getIfMatch(),
modifiedRequestConditions.getIfNoneMatch(), modifiedRequestConditions.getTagsConditions(), null,
context)
.map(rb -> new SimpleResponse<>(rb, rb.getDeserializedHeaders().getXMsCopyStatus()));
}
}