S3 client
??? warning “Experimental module”
**Experimental** module is fully working and tested, but requires additional approbation and usage analytics,
for this reason, API may potentially undergo minor changes before fully stable.
Module provides a thin layer of abstraction for creating S3-clients using declarative-style annotations, or using imperative-style clients to work with S3 storage.
AWS¶
S3 client implementation based on the AWS library.
Components available for injection:
- Imperative Kora S3 clients
S3Client
synchronous AWS S3 clientS3AsyncClient
asynchronous AWS S3 clientS3AsyncClient
with tag@Tag(MultipartUpload.class)
asynchronous AWS S3 client for batch uploading
Dependency¶
Dependency build.gradle
:
Module:
Dependency build.gradle.kts
:
Module:
Requires any HTTP client module to be added.
Configuration¶
Complete configuration described in the AwsS3ClientConfig
and S3Config
classes (example values or default values are specified):
s3client {
aws {
addressStyle = "PATH" //(1)!
requestTimeout = "45s" //(2)!
checksumValidationEnabled = false //(3)!
chunkedEncodingEnabled = true //(4)!
upload {
bufferSize = "32MiB" //(5)!
partSize = "8MiB" //(6)!
}
}
url = "http://localhost:9000" //(7)!
accessKey = "someKey" //(8)!
secretKey = "someSecret" //(9)!
region = "aws-global" //(10)!
telemetry {
logging {
enabled = false //(11)!
}
metrics {
enabled = true //(12)!
slo = [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] //(13)!
}
tracing {
enabled = true //(14)!
}
}
}
- Which type of file access to use, can have values
PATH
orVIRTUAL_HOSTED
- The maximum execution time of the operation
- Whether to check the checksum MD5 of files before uploading and when retrieving from AWS
- Whether to encode in chunks when signing file data when uploading to AWS
- Maximum buffer size when loading files (specified as a number in bytes / or as
4MiB
/4MB
/1000Kb
etc.) - Maximum file chunk size when loading a file at a time (specified as a number in bytes / or as
4MiB
/4MB
/1000Kb
etc.) - S3 storage URL
- S3 access key
- S3 access secret
- S3 storage region
- Enables module logging (default is
false
) - Enables module metrics (default is
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default is
true
)
s3client:
aws:
addressStyle: "PATH" #(1)!
requestTimeout: "45s" #(2)!
checksumValidationEnabled: false #(3)!
chunkedEncodingEnabled: true #(4)!
upload:
bufferSize: "32MiB" #(5)!
partSize: "8MiB" #(6)!
url: "http://localhost:9000" #(7)!
accessKey: "someKey" #(8)!
secretKey: "someSecret" #(9)!
region: "aws-global" #(10)!
telemetry:
logging:
enabled: false #(11)!
metrics:
enabled: true #(12)!
slo: [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] #(13)!
telemetry:
enabled: true #(14)!
- Which type of file access to use, can have values
PATH
orVIRTUAL_HOSTED
- The maximum execution time of the operation
- Whether to check the checksum MD5 of files before uploading and when retrieving from AWS
- Whether to encode in chunks when signing file data when uploading to AWS
- Maximum buffer size when loading files (specified as a number in bytes / or as
4MiB
/4MB
/1000Kb
etc.) - Maximum file chunk size when loading a file at a time (specified as a number in bytes / or as
4MiB
/4MB
/1000Kb
etc.) - S3 storage URL
- S3 access key
- S3 access secret
- S3 storage region
- Enables module logging (default is
false
) - Enables module metrics (default is
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default is
true
)
Response format¶
When using AWS module, it is possible to return special response formats specific only to AWS library:
Операция | Формат ответа |
---|---|
Get file | GetObjectResponse / ResponseInputStream |
List files | ListObjectsV2Response |
Add file | PutObjectResponse |
Delete file | DeleteObjectResponse / DeleteObjectsResponse |
Minio¶
S3 client implementation based on Minio library. Note that the implementation uses OkHttp written in Kotlin and uses appropriate dependencies.
Available components for injection:
- Imperative Kora S3 clients
MinioClient
synchronous Minio S3 clientMinioAsyncClient
asynchronous Minio S3 client
Dependency¶
Dependency build.gradle
:
Module:
Dependency build.gradle.kts
:
Module:
You can add OkHttp module dependency or a standard HTTP client will be created automatically.
Configuration¶
Complete configuration described in the MinioS3ClientConfig
and S3Config
classes (example values or default values are specified):
s3client {
minio {
addressStyle = "PATH" //(1)!
requestTimeout = "45s" //(2)!
upload {
partSize = "8MiB" //(3)!
}
}
url = "http://localhost:9000" //(4)!
accessKey = "someKey" //(5)!
secretKey = "someSecret" //(6)!
region = "aws-global" //(7)!
telemetry {
logging {
enabled = false //(8)!
}
metrics {
enabled = true //(9)!
slo = [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] //(10)!
}
tracing {
enabled = true //(11)!
}
}
}
- Which type of file access to use, can have values
PATH
orVIRTUAL_HOSTED
- Maximum execution time of the operation
- Maximum file chunk size for a single file upload (specified as a number in bytes / or as
4MiB
/4MB
/1000Kb
etc.) - S3 storage URL
- S3 access key
- S3 access secret
- S3 storage region
- Enables module logging (default is
false
) - Enables module metrics (default is
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default is
true
)
s3client:
minio:
addressStyle: "PATH" #(1)!
requestTimeout: "45s" #(1)!
upload:
partSize: "8MiB" #(2)!
url: "http://localhost:9000" #(3)!
accessKey: "someKey" #(4)!
secretKey: "someSecret" #(5)!
region: "aws-global" #(6)!
telemetry:
logging:
enabled: false #(7)!
metrics:
enabled: true #(8)!
slo: [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] #(9)!
telemetry:
enabled: true #(10)!
- Which type of file access to use, can have values
PATH
orVIRTUAL_HOSTED
- Maximum execution time of the operation
- Maximum file chunk size for a single file upload (specified as a number in bytes / or as
4MiB
/4MB
/1000Kb
etc.) - S3 storage URL
- S3 access key
- S3 access secret
- S3 storage region
- Enables module logging (default is
false
) - Enables module metrics (default is
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default is
true
)
Client declarative¶
It is suggested to use special annotations to create a declarative client:
@S3.Client
- indicates that the interface is a declarative S3 client@S3.Get
- indicates that the method performs the get file/metadata operation@S3.List
- indicates that the method performs the get file/metadata list operation@S3.Put
- indicates that the method performs the add file operation@S3.Delete
- indicates that the method performs the delete file operation
Client Configuration¶
Configuration of a particular implementation of @S3.Client
:
Configuration in the case of the s3client.someClient
path described in the S3ClientConfig
class:
Get file¶
Section describes the operation of getting a file/metadata using a declarative S3 client.
It is suggested to use the @S3.Get
annotation to specify the operation.
Metadata¶
Get file by key operation can return either a complete file S3Object
along with data,
or a lightweight version in the form of file metadata S3ObjectMeta
without data,
this method is much faster because it does not return file data.
Key template¶
You can also specify a key as a template and substitute method arguments there as part of the template, all method arguments must be part of the compound key.
@S3.Client("s3client.someClient")
public interface SomeClient {
@S3.Get("prefix-{key1}-{key2}-suffix") //(1)!
S3Object operation(String key1, int key2); //(2)!
}
- template to build the key template, each template argument will be substituted via
toString()
, the arguments in the template are specified as method argument names in{covens}
. - All method arguments must be part of the key template
@S3.Client("s3client.someClient")
interface SomeClient {
@S3.Get("prefix-{key1}-{key2}-suffix") //(1)!
fun operation(key1: String, key2: Int): S3Object //(2)!
}
- template to build the key template, each template argument will be substituted via
toString()
, the arguments in the template are specified as method argument names in{covens}
. - All method arguments must be part of the key template
Multiple keys¶
It is also possible to retrieve multiple files at once by key, either as a complete file along with the S3Object
data,
or a lighter version as a set of metadata files without S3ObjectMeta
data.
List files¶
The section describes the operation to get a list of files/metadata using a declarative S3 client.
It is suggested that the @S3.List
annotation be used to specify the operation.
You can specify key prefix to sample the desired keys matching the prefix,
you can also specify a file selection limit, the maximum number of files for the operation is 1000
:
@S3.Client("s3client.someClient")
public interface SomeClient {
@S3.List
S3ObjectList operation1(String prefix); //(1)!
@S3.List("some-prefix-") //(2)!
S3ObjectList operation2();
@S3.List(limit = 100) //(3)!
S3ObjectList operation3();
}
- prefix can be passed as a method argument if it is not specified in the annotation
- prefix can be specified in the annotation
- you can specify the file selection limit for the enumeration operation, the maximum number of files for the
1000
operation:
@S3.Client("s3client.someClient")
interface SomeClient {
@S3.List
fun operation1(prefix: String): S3ObjectList //(1)!
@S3.List("some-prefix-") //(2)!
fun operation2(): S3ObjectList
@S3.List(limit = 100) //(3)!
fun operation3(): S3ObjectList
}
- prefix can be passed as a method argument if it is not specified in the annotation
- prefix can be specified in the annotation
- you can specify the file selection limit for the enumeration operation, the maximum number of files for the
1000
operation:
Metadata¶
Get file by key operation can return either a complete file S3ObjectList
along with data,
or a lightweight version in the form of file metadata S3ObjectMetaList
without data,
this method is much faster because it does not return file data.
Prefix template¶
A prefix can also be specified as a template and method arguments can be substituted there as part of the template, all method arguments must be part of a compound key.
@S3.Client("s3client.someClient")
public interface SomeClient {
@S3.List("prefix-{key1}-{key2}-") //(1)!
S3ObjectList operation(String key1, int key2);
}
- template to build the prefix template, each template argument will be substituted via
toString()
, the arguments in the template are specified as method argument names in{covens}
.
@S3.Client("s3client.someClient")
interface SomeClient {
@S3.List("prefix-{key1}-{key2}-") //(1)!
fun operation(key1: String, key2: Int): S3ObjectList
}
- template to build the prefix template, each template argument will be substituted via
toString()
, the arguments in the template are specified as method argument names in{covens}
.
Separator¶
You can specify a delimiter for key prefix, to exclude required files from the sample:
Add file¶
Section describes the operation of adding a file using a declarative S3 client.
It is suggested to use the @S3.Put
annotation for the operation.
It is required to specify the key and body of the file to be added:
@S3.Client("s3client.someClient")
public interface SomeClient {
@S3.Put
void operation1(String key, //(1)!
S3Body body); //(2)!
@S3.Put("some-key") //(3)!
S3ObjectUpload operation2(S3Body body);
}
- File key by which it will be added to the repository
- file body itself, which will be added to the repository
- key can also be specified in the annotation if it is static
@S3.Client("s3client.someClient")
interface SomeClient {
@S3.Put
fun operation(key: String, body: S3Body)
@S3.Put("some-key")
fun operation(body: S3Body): S3ObjectUpload
}
- File key by which it will be added to the repository
- file body itself, which will be added to the repository
- key can also be specified in the annotation if it is static
File body¶
File body (S3Body
) can be created from byte[]
/ ByteBuffer
/ InputStream
/ Flow.Publisher<ByteBuffer>
via the corresponding static constructor methods.
If the file is very large or its length is unknown and streaming is required, it is recommended to create the body using S3Body.ofPublisher()
or S3Body.ofInputStreamUnbound()
.
If no file type is specified, it will be set as application/octet-stream
.
Key template¶
Key can also be specified as a template and method arguments can be substituted there as part of the template, all method arguments must be part of a compound key.
@S3.Client("s3client.someClient")
public interface SomeClient {
@S3.Put("prefix-{key1}-{key2}-suffix") //(1)!
void operation(String key1, int key2, S3Body body); //(2)!
}
- template to build the key template, each template argument will be substituted via
toString()
, the arguments in the template are specified as method argument names in{covens}
. - All method arguments must be part of the key template either
S3Body
@S3.Client("s3client.someClient")
interface SomeClient {
@S3.Put("prefix-{key1}-{key2}-suffix") //(1)!
fun operation(key1: String, key2: Int, body: S3Body) //(2)!
}
- template to build the key template, each template argument will be substituted via
toString()
, the arguments in the template are specified as method argument names in{covens}
. - All method arguments must be part of the key template either
S3Body
Delete file¶
Section describes the operation of deleting a file using a declarative S3 client.
It is suggested to use the @S3.Delete
annotation for the operation.
Key template¶
Key can also be specified as a template and method arguments can be substituted there as part of the template, all method arguments must be part of the composite key.
@S3.Client("s3client.someClient")
public interface SomeClient {
@S3.Delete("prefix-{key1}-{key2}-suffix") //(1)!
void operation(String key1, int key2); //(2)!
}
- template to build the key template, each template argument will be substituted via
toString()
, the arguments in the template are specified as method argument names in{covens}
. - All method arguments must be part of the key template
@S3.Client("s3client.someClient")
interface SomeClient {
@S3.Delete("prefix-{key1}-{key2}-suffix") //(1)!
fun operation(key1: String, key2: Int) //(2)!
}
- template to build the key template, each template argument will be substituted via
toString()
, the arguments in the template are specified as method argument names in{covens}
. - All method arguments must be part of the key template
Multiple keys¶
It is also possible to retrieve multiple files at once by key, either as a complete file along with the S3Object
data,
or a lighter version as a set of metadata files without S3ObjectMeta
data.
Signatures¶
Available signatures for repository methods out of the box:
The T
refers to the type of the return value.
T myMethod()
CompletionStage<T> myMethod()
CompletionStageMono<T> myMethod()
Project Reactor (require dependency)
By T
we mean the type of the return value, either T?
, or Unit
.
myMethod(): T
suspend myMethod(): T
Kotlin Coroutine (require dependency asimplementation
)
Client imperative¶
It is possible to implement an imperative Kora client to work with S3, both synchronous and asynchronous clients are provided:
S3KoraClient
- client for synchronous operationS3KoraAsyncClient
- client for asynchronous operation.
Exceptions¶
Special errors will be thrown if a client operation error occurs:
S3NotFoundException
- in case it does not find a file by the specified keyS3DeleteException
- in case of file deletion errorS3Exception
- in any other case