oci-java-sdk
oci-java-sdk copied to clipboard
Retries for operations that upload binary data without request-level retries do not retry in OCI Java SDK versions 3.0.0 to 3.31.0
If you are using any of the OCI Java SDK synchronous clients that upload streams of data, e.g. ObjectStorageClient
or DataSafeClient
, and you do not define the RetryConfiguration
at request level, your requests will not be automatically retried. However, there is no chance of silent data corruption.
Description
When using OCI Java SDK (versions 3.0.0
to 3.31.0
) for operations that have retries enabled by default, and you do not define the RetryConfiguration
at request level, and the request fails with a retry-able error, the OCI Java SDK should automatically retry the request. In this situation, the clients fail to reset the stream position for requests that upload streams. As a result, retries cannot be attempted, and the operation fails with a BmcException
.
The upload of the stream is likely incomplete at this time and needs to be re-attempted. Fortunately, this failure is visible and therefore cannot lead to silent data corruption.
Affected requests
This happens only for synchronous clients in versions 3.0.0
to 3.31.0
:
-
com.oracle.bmc.objectstorage.requests.PutObjectRequest
-
com.oracle.bmc.objectstorage.requests.UploadPartRequest
-
com.oracle.bmc.datasafe.requests.UploadSensitiveDataModelRequest
-
com.oracle.bmc.datasafe.requests.UploadMaskingPolicyRequest
-
com.oracle.bmc.loganalytics.requests.RegisterLookupRequest
-
com.oracle.bmc.loganalytics.requests.UpdateLookupDataRequest
-
com.oracle.bmc.loganalytics.requests.AppendLookupDataRequest
-
com.oracle.bmc.loganalytics.requests.ImportCustomContentRequest
-
com.oracle.bmc.loganalytics.requests.UploadDiscoveryDataRequest
-
com.oracle.bmc.loganalytics.requests.UploadLogFileRequest
-
com.oracle.bmc.loganalytics.requests.UploadLogEventsFileRequest
-
com.oracle.bmc.keymanagement.requests.RestoreVaultFromFileRequest
-
com.oracle.bmc.keymanagement.requests.RestoreKeyFromFileRequest
-
com.oracle.bmc.datascience.requests.CreateModelArtifactRequest
-
com.oracle.bmc.datascience.requests.CreateJobArtifactRequest
-
com.oracle.bmc.datascience.requests.CreateStepArtifactRequest
-
com.oracle.bmc.marketplacepublisher.requests.UpdateTermVersionContentRequest
-
com.oracle.bmc.marketplacepublisher.requests.UpdateListingRevisionIconContentRequest
-
com.oracle.bmc.marketplacepublisher.requests.CreateTermVersionRequest
-
com.oracle.bmc.marketplacepublisher.requests.UpdateListingRevisionAttachmentContentRequest
-
com.oracle.bmc.genericartifactscontent.requests.PutGenericArtifactContentByPathRequest
-
com.oracle.bmc.dns.requests.CreateZoneFromZoneFileRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadSecurityRulesRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadServicesRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadServiceListsRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadApplicationsRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadAddressListsRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadDecryptionRulesRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadUrlListsRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadMappedSecretsRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadApplicationGroupsRequest
-
com.oracle.bmc.networkfirewall.requests.BulkUploadDecryptionProfilesRequest
-
com.oracle.bmc.cloudmigrations.requests.ImportMigrationPlanRequest
-
com.oracle.bmc.opsi.requests.PutAwrHubObjectRequest
-
com.oracle.bmc.functions.requests.InvokeFunctionRequest
You are also affected if you use the Object Storage Upload Manager:
Operations that do not upload streams are not affected.
If the stream that is being uploaded is a ByteArrayInputStream
, operations are not affected.
To summarize, you are affected if you
- use versions
3.0.0
to3.31.0
- and upload streams using the above operations,
- and do not set a retry configuration at request level
Workarounds
This problem was fixed in version 3.31.1
. If you are using any of the affected versions, we recommend that you upgrade to version 3.31.1
or later.
If, for some reason, you cannot upgrade to version 3.31.1
or later, here are some other possible workarounds:
- Set the
RetryConfiguration
at request level by following this example.- The easiest way to do this is to insert the default retry policy at request level:
final PutObjectRequest putObjectRequest = PutObjectRequest.builder() .retryConfiguration( com.oracle.bmc.retrier.RetryConfiguration.SDK_DEFAULT_RETRY_CONFIGURATION) // other request parameters .build();
- The easiest way to do this is to insert the default retry policy at request level:
- Disable default retries by following any of the methods described here.
- Buffer the data in memory yourself by first copying it into a
ByteArrayInputSteam
, and then use thatByteArrayInputStream
in your upload operation.
**Update
We had previously stated that there may be a potential data corruption issue. Our initial fear was that the first attempt may fail, and subsequent retries fail to reset the stream and therefore do not upload the entire stream again, leading to missing data. During careful evaluation, we determined that data corruption does not occur.
Instead, the first retry fails with an exception:
Caused by: java.lang.RuntimeException: Stream {} does not support mark/reset, retries do not work
This is still far from ideal, but a hard, visible failure is preferable to silent data corruption.
**Update 2
This problem was fixed in version 3.31.1
.