ジャストほしい情報が日本語でなかったのでドキュメンテーションする。
AWS SDK for Javaを利用してS3にファイルをアップロードする際に、やや詳細な設定を付加する。
目次
- リトライ周りのケース概略
- とりあえず実装してみる
- 関連クラスの整理
- その他
- 参考
リトライ周りのケース概略
AWS SDK for Javaを使って、ローカルにあるS3にファイルをアップロードしたい。
そして、AWS側の都合で通信断となったり諸事情により正常稼働できないケースを見越して、AWSクライアントにリトライポリシーを仕込みたい。
今回は、AWS SDK for Javaに含まれる AmazonS3
オブジェクトのシングルトンを返すユーティリティクラスを用意する。
なお、本記事では自分が特に困った「クライアント生成時のオプション設定」にフォーカスを置く。そのため、ファイルアップロード前後の処理関連の説明は省略する。
とりあえず実装してみる
シンプルなクライアントオプションを詰めたクライアントを構築する。
流れとしては、リトライポリシーインスタンス、クライアント設定インスタンス、そしてクライアントの3つを準備していく。
まずは、ポリシーの基本設定やAWS定義のストラテジーなどを詰めた RetryPolicy
インスタンスを返すメソッドを用意。
/**
* ExponentialBackoffStrategyのコンストラクタにbaseDelayとmaxBackoffTimeの適切な値を入れる。
* 注意:ここではサンプルとして適当な値を入れてます。
*/
private static RetryPolicy getS3BaseRetryPolicy() {
return new RetryPolicy(
new PredefinedRetryPolicies.SDKDefaultRetryCondition(),
new PredefinedBackoffStrategies.ExponentialBackoffStrategy(0, 0),
PredefinedRetryPolicies.DEFAULT_MAX_ERROR_RETRY,
true
);
}
次に、↑で定義したRetryPolicy
インスタンスを返すメソッドを呼んで、クライアント設定を定義する。
private static final ClientConfiguration clientConfiguration = new ClientConfiguration()
.withRetryPolicy(getS3BaseRetryPolicy());
そして、他のクラスやメソッドが呼び出すS3クライアントのエントリーポイントとなる AmazonS3
オブジェクトを、クライアント設定を詰めて初期化。
private static final AmazonS3 amazonS3 = AmazonS3ClientBuilder
.standard()
.withClientConfiguration(clientConfiguration)
.withRegion(Regions.DEFAULT_REGION)
.build();
ここまでの設定をまとめると、
public final class AmazonS3Utils {
/**
* リトライポリシーを反映した、クライアント設定を初期化。
*/
private static final ClientConfiguration clientConfiguration = new ClientConfiguration()
.withRetryPolicy(getS3BaseRetryPolicy());
/**
* AmazonS3インスタンスを初期化。
*/
private static final AmazonS3 amazonS3 = AmazonS3ClientBuilder
.standard()
.withClientConfiguration(clientConfiguration)
.withRegion(Regions.DEFAULT_REGION).build();
/**
* ファイル操作を提供するオブジェクトを返す
*/
public static AmazonS3 getS3() {
return amazonS3;
}
/**
* リトライポリシーをインスタンス化。
*/
private static RetryPolicy getS3BaseRetryPolicy() {
return new RetryPolicy(
new PredefinedRetryPolicies.SDKDefaultRetryCondition(),
new PredefinedBackoffStrategies.ExponentialBackoffStrategy(0, 0),
PredefinedRetryPolicies.DEFAULT_MAX_ERROR_RETRY,
true
);
}
}
これで準備完了。あとは、getS3
メソッドで設定を反映したS3クライアントを呼び出せるようになる。
登場オブジェクトの整理
実装例に出てくる複数のクラスの役割を整理してみたい。
自分の調べ方が悪かったのか、やりたいことをさくっとできる感じではなかった...
AmazonS3(インターフェース)
S3のオブジェクトを操作するメソッドを定義するインターフェース。
具体的な実装はクライアントビルダークラスにより提供されるようだ。
AmazonS3
このオブジェクトはインターフェースであり、具はAmazonS3ClientBuilder
などにより提供されるようだ。
もともと、リトライ周りの設定を追加する前はAmazonS3ClientBuilder.standard().build();
で済ませていた。最小限の機能で済むなら、そのようなミニマルなクライアントで十分かと思う。
ClientConfiguration(クラス)
ClientConfigurationは、リトライやプロキシ、ユーザエージェント文字列などAWSクライアントに追加できるオプションの設定クラス。
今回の例では自分は多少カスタマイズを加えたRetryPolicy
を渡したが、PredefinedRetryPolicies
にあるstaticメソッドであるgetDefaultRetryPolicy()
を呼び出すことで、デフォルト設定をクライアント設定に詰められるようだ。
RetryPolicy(クラス)
RetryPolicyは、 ClientConfiguration
と組み合わせて、リトライポリシーを構築するクラス。
コンストラクタに必要なオブジェクトや値を詰めて初期化する。
コンストラクタには、RetryCondition
やBackoffStrategy
などのインターフェースが引数に指定されているのだが、引数のインターフェースに合うクラスに何を指定すればよいのか、最初悩んだ。いくつかコンストラクタがあるが、そのうちの一つを挙げてみると
public RetryPolicy(RetryCondition retryCondition,
BackoffStrategy backoffStrategy,
int maxErrorRetry,
boolean honorMaxErrorRetryInClientConfig) {
this(retryCondition, backoffStrategy, maxErrorRetry, honorMaxErrorRetryInClientConfig, false);
}
いろいろ調べていくうちに、どうやらSDKは事前にある程度定義したポリシーを詰められるクラスを用意していたとわかった。
RetryCondition
インターフェースには、今回の実装ではAWS SDKのデフォルトに従うことにした。PredefinedRetryPoliciesに、何種類か、使えそうな実装があるのでRetryPolicy
のコンストラクタに渡せばよい。
設定にこだわりがないのであれば、デフォルト設定を返してくれるstaticプロパティDEFAULT_RETRY_CONDITION
を指定すればよさそう。
BackoffStrategyインターフェースを拡張した抽象クラスV2CompatibleBackoffStrategyが用意されており、この抽象クラスを拡張したアダプターおよび各種具象クラスを使えばBackoffStrategy
インターフェース相当の引数を設定できる。
詳しくいうと、PredefinedBackoffStrategiesには V2CompatibleBackoffStrategyAdapter
を拡張した、
- FullJitterBackoffStrategy
- EqualJitterBackoffStrategy
- EqualJitterBackoffStrategy
などが用意されている。RetryPolicyを初期化する際に、コンストラクタになんとかBackoffStrategyを渡すことで細かい設定を施すことが可能である。
その他
本当は公式のドキュメントで、いい感じに説明してくれているサムシングがある気がしている...
自分は探し当てられなかったため、「そんな無理せんでもこれ読んだらええんやけど」ってのがあればご指摘いただければありがたいです。
参考
- AWS でのエラー再試行とエクスポネンシャルバックオフ
- AWS SDK Java でAPIコールをExponential Backoffでリトライする
- Class RetryPolicy
- Class PredefinedRetryPolicies
- Class ClientConfiguration
- Tuning AWS Java SDK HTTP request settings for latency-aware Amazon DynamoDB applications
- Exponential Backoff - wikipedia
- 大規模分散システム ”Amazon Web Services”の使い方(PDF)