diff --git a/.changes/next-release/feature-AWSSDKforJavav2-0e79eba.json b/.changes/next-release/feature-AWSSDKforJavav2-0e79eba.json new file mode 100644 index 000000000000..3342d8419699 --- /dev/null +++ b/.changes/next-release/feature-AWSSDKforJavav2-0e79eba.json @@ -0,0 +1,6 @@ +{ + "type": "feature", + "category": "AWS SDK for Java v2", + "contributor": "", + "description": "Allow users to configure `subscribeTimeout` for BlockingOutputStreamAsyncRequestBody. See [#4893](https://github.com/aws/aws-sdk-java-v2/issues/4893)" +} diff --git a/core/sdk-core/src/main/java/software/amazon/awssdk/core/async/AsyncRequestBody.java b/core/sdk-core/src/main/java/software/amazon/awssdk/core/async/AsyncRequestBody.java index 7f9d4be67b2d..cde394b6ed42 100644 --- a/core/sdk-core/src/main/java/software/amazon/awssdk/core/async/AsyncRequestBody.java +++ b/core/sdk-core/src/main/java/software/amazon/awssdk/core/async/AsyncRequestBody.java @@ -418,6 +418,8 @@ static BlockingInputStreamAsyncRequestBody forBlockingInputStream(Long contentLe *
The caller is responsible for calling {@link OutputStream#close()} on the * {@link BlockingOutputStreamAsyncRequestBody#outputStream()} when writing is complete. * + *
By default, it will time out if streaming hasn't started within 10 seconds, and you can configure the timeout + * via {@link BlockingOutputStreamAsyncRequestBody#builder()} *
Example Usage *
* {@snippet :
@@ -440,9 +442,12 @@ static BlockingInputStreamAsyncRequestBody forBlockingInputStream(Long contentLe
* // Wait for the service to respond.
* PutObjectResponse response = responseFuture.join();
* }
+ * @see BlockingOutputStreamAsyncRequestBody
*/
static BlockingOutputStreamAsyncRequestBody forBlockingOutputStream(Long contentLength) {
- return new BlockingOutputStreamAsyncRequestBody(contentLength);
+ return BlockingOutputStreamAsyncRequestBody.builder()
+ .contentLength(contentLength)
+ .build();
}
/**
diff --git a/core/sdk-core/src/main/java/software/amazon/awssdk/core/async/BlockingOutputStreamAsyncRequestBody.java b/core/sdk-core/src/main/java/software/amazon/awssdk/core/async/BlockingOutputStreamAsyncRequestBody.java
index 872e25306cc8..c0a044ffa3ba 100644
--- a/core/sdk-core/src/main/java/software/amazon/awssdk/core/async/BlockingOutputStreamAsyncRequestBody.java
+++ b/core/sdk-core/src/main/java/software/amazon/awssdk/core/async/BlockingOutputStreamAsyncRequestBody.java
@@ -27,6 +27,7 @@
import software.amazon.awssdk.core.exception.NonRetryableException;
import software.amazon.awssdk.core.internal.util.NoopSubscription;
import software.amazon.awssdk.utils.CancellableOutputStream;
+import software.amazon.awssdk.utils.Validate;
import software.amazon.awssdk.utils.async.OutputStreamPublisher;
/**
@@ -46,13 +47,11 @@ public final class BlockingOutputStreamAsyncRequestBody implements AsyncRequestB
private final Long contentLength;
private final Duration subscribeTimeout;
- BlockingOutputStreamAsyncRequestBody(Long contentLength) {
- this(contentLength, Duration.ofSeconds(10));
- }
-
- BlockingOutputStreamAsyncRequestBody(Long contentLength, Duration subscribeTimeout) {
- this.contentLength = contentLength;
- this.subscribeTimeout = subscribeTimeout;
+ private BlockingOutputStreamAsyncRequestBody(Builder builder) {
+ this.contentLength = builder.contentLength;
+ this.subscribeTimeout = Validate.isPositiveOrNull(builder.subscribeTimeout, "subscribeTimeout") != null ?
+ builder.subscribeTimeout :
+ Duration.ofSeconds(10);
}
/**
@@ -69,6 +68,13 @@ public CancellableOutputStream outputStream() {
return delegate;
}
+ /**
+ * Creates a default builder for {@link BlockingOutputStreamAsyncRequestBody}.
+ */
+ public static Builder builder() {
+ return new Builder();
+ }
+
@Override
public Optional You may want to increase it if the request may not be executed right away.
+ *
+ * @param subscribeTimeout the timeout
+ * @return Returns a reference to this object so that method calls can be chained together.
+ */
+ public Builder subscribeTimeout(Duration subscribeTimeout) {
+ this.subscribeTimeout = subscribeTimeout;
+ return this;
+ }
+
+ /**
+ * The content length of the output stream.
+ *
+ * @param contentLength the content length
+ * @return Returns a reference to this object so that method calls can be chained together.
+ */
+ public Builder contentLength(Long contentLength) {
+ this.contentLength = contentLength;
+ return this;
+ }
+
+ public BlockingOutputStreamAsyncRequestBody build() {
+ return new BlockingOutputStreamAsyncRequestBody(this);
+ }
+ }
}
diff --git a/core/sdk-core/src/test/java/software/amazon/awssdk/core/async/BlockingOutputStreamAsyncRequestBodyTest.java b/core/sdk-core/src/test/java/software/amazon/awssdk/core/async/BlockingOutputStreamAsyncRequestBodyTest.java
index 5cfcbc593916..ed2aa9b1a7ca 100644
--- a/core/sdk-core/src/test/java/software/amazon/awssdk/core/async/BlockingOutputStreamAsyncRequestBodyTest.java
+++ b/core/sdk-core/src/test/java/software/amazon/awssdk/core/async/BlockingOutputStreamAsyncRequestBodyTest.java
@@ -50,9 +50,9 @@ public void outputStream_waitsForSubscription() throws IOException {
@Test
@Timeout(10)
- public void outputStream_failsIfSubscriptionNeverComes() {
+ public void outputStream_overrideSubscribeTimeout_failsIfSubscriptionNeverComes() {
BlockingOutputStreamAsyncRequestBody requestBody =
- new BlockingOutputStreamAsyncRequestBody(0L, Duration.ofSeconds(1));
+ BlockingOutputStreamAsyncRequestBody.builder().contentLength(0L).subscribeTimeout(Duration.ofSeconds(1)).build();
assertThatThrownBy(requestBody::outputStream).hasMessageContaining("The service request was not made");
}