2022-05-12 19:37:38 +00:00
|
|
|
package s3
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
2022-09-02 19:23:33 +00:00
|
|
|
"crypto/sha256"
|
|
|
|
"encoding/hex"
|
2023-07-17 13:33:51 +00:00
|
|
|
"fmt"
|
2022-06-17 10:37:43 +00:00
|
|
|
"strconv"
|
2022-05-12 19:37:38 +00:00
|
|
|
"time"
|
|
|
|
|
2023-03-07 14:53:17 +00:00
|
|
|
"git.frostfs.info/TrueCloudLab/xk6-frostfs/internal/stats"
|
2022-05-12 19:37:38 +00:00
|
|
|
"github.com/aws/aws-sdk-go-v2/aws"
|
2023-07-17 13:19:52 +00:00
|
|
|
"github.com/aws/aws-sdk-go-v2/feature/s3/manager"
|
2023-07-17 15:32:18 +00:00
|
|
|
"github.com/aws/aws-sdk-go-v2/service/s3"
|
2022-06-17 10:37:43 +00:00
|
|
|
"github.com/aws/aws-sdk-go-v2/service/s3/types"
|
2022-05-20 11:14:50 +00:00
|
|
|
"github.com/dop251/goja"
|
2022-05-12 19:37:38 +00:00
|
|
|
"go.k6.io/k6/js/modules"
|
|
|
|
"go.k6.io/k6/metrics"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
Client struct {
|
|
|
|
vu modules.VU
|
|
|
|
cli *s3.Client
|
|
|
|
}
|
2022-05-20 09:34:34 +00:00
|
|
|
|
|
|
|
PutResponse struct {
|
|
|
|
Success bool
|
|
|
|
Error string
|
|
|
|
}
|
|
|
|
|
2022-09-27 16:01:43 +00:00
|
|
|
DeleteResponse struct {
|
|
|
|
Success bool
|
|
|
|
Error string
|
|
|
|
}
|
|
|
|
|
2022-05-20 09:34:34 +00:00
|
|
|
GetResponse struct {
|
|
|
|
Success bool
|
|
|
|
Error string
|
|
|
|
}
|
2022-06-17 10:37:43 +00:00
|
|
|
|
|
|
|
CreateBucketResponse struct {
|
|
|
|
Success bool
|
|
|
|
Error string
|
|
|
|
}
|
2022-09-02 19:23:33 +00:00
|
|
|
|
|
|
|
VerifyHashResponse struct {
|
|
|
|
Success bool
|
|
|
|
Error string
|
|
|
|
}
|
2022-05-12 19:37:38 +00:00
|
|
|
)
|
|
|
|
|
2022-05-20 11:14:50 +00:00
|
|
|
func (c *Client) Put(bucket, key string, payload goja.ArrayBuffer) PutResponse {
|
|
|
|
rdr := bytes.NewReader(payload.Bytes())
|
2022-05-12 19:37:38 +00:00
|
|
|
sz := rdr.Size()
|
|
|
|
|
|
|
|
start := time.Now()
|
|
|
|
_, err := c.cli.PutObject(c.vu.Context(), &s3.PutObjectInput{
|
|
|
|
Bucket: aws.String(bucket),
|
|
|
|
Key: aws.String(key),
|
|
|
|
Body: rdr,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
stats.Report(c.vu, objPutFails, 1)
|
2022-05-20 09:34:34 +00:00
|
|
|
return PutResponse{Success: false, Error: err.Error()}
|
2022-05-12 19:37:38 +00:00
|
|
|
}
|
|
|
|
|
2022-12-30 10:47:28 +00:00
|
|
|
stats.Report(c.vu, objPutTotal, 1)
|
2022-05-12 19:37:38 +00:00
|
|
|
stats.ReportDataSent(c.vu, float64(sz))
|
|
|
|
stats.Report(c.vu, objPutDuration, metrics.D(time.Since(start)))
|
2022-05-20 09:34:34 +00:00
|
|
|
return PutResponse{Success: true}
|
2022-05-12 19:37:38 +00:00
|
|
|
}
|
|
|
|
|
2023-07-17 13:33:51 +00:00
|
|
|
const multipartUploadMinPartSize = 5 * 1024 * 1024 // 5MB
|
|
|
|
|
2023-07-17 12:46:53 +00:00
|
|
|
func (c *Client) Multipart(bucket, key string, objPartSize int, payload goja.ArrayBuffer, concurrency int) PutResponse {
|
2023-07-17 15:32:18 +00:00
|
|
|
if objPartSize < multipartUploadMinPartSize {
|
2023-07-17 12:46:53 +00:00
|
|
|
stats.Report(c.vu, objPutFails, 1)
|
|
|
|
return PutResponse{Success: false, Error: fmt.Sprintf("part size '%d' must be greater than '%d'(5 MB)", objPartSize, multipartUploadMinPartSize)}
|
|
|
|
}
|
2023-07-18 09:21:41 +00:00
|
|
|
if concurrency < 1 {
|
2023-07-18 09:03:22 +00:00
|
|
|
stats.Report(c.vu, objPutFails, 1)
|
|
|
|
return PutResponse{Success: false, Error: fmt.Sprintf("number of parts to upload in parallel must be greater than 0")}
|
|
|
|
}
|
2023-07-17 12:46:53 +00:00
|
|
|
start := time.Now()
|
2023-07-17 13:25:14 +00:00
|
|
|
uploader := manager.NewUploader(c.cli, func(u *manager.Uploader) {
|
2023-07-17 15:32:18 +00:00
|
|
|
u.PartSize = int64(objPartSize)
|
|
|
|
u.Concurrency = concurrency
|
|
|
|
})
|
2023-07-17 14:59:21 +00:00
|
|
|
_, err := uploader.Upload(c.vu.Context(), &s3.PutObjectInput{
|
2023-07-17 15:32:18 +00:00
|
|
|
Bucket: aws.String(bucket),
|
|
|
|
Key: aws.String(key),
|
|
|
|
Body: bytes.NewReader(payload.Bytes()),
|
|
|
|
})
|
|
|
|
if err != nil {
|
2023-07-17 12:46:53 +00:00
|
|
|
stats.Report(c.vu, objPutFails, 1)
|
|
|
|
return PutResponse{Success: false, Error: err.Error()}
|
|
|
|
}
|
2023-07-17 13:26:59 +00:00
|
|
|
size := len(payload.Bytes())
|
2023-07-17 13:17:32 +00:00
|
|
|
stats.Report(c.vu, objPutTotal, 1)
|
2023-07-17 13:26:59 +00:00
|
|
|
stats.ReportDataSent(c.vu, float64(size))
|
2023-07-17 13:17:32 +00:00
|
|
|
stats.Report(c.vu, objPutDuration, metrics.D(time.Since(start)))
|
|
|
|
return PutResponse{Success: true}
|
2023-07-17 12:46:53 +00:00
|
|
|
}
|
|
|
|
|
2022-09-27 16:01:43 +00:00
|
|
|
func (c *Client) Delete(bucket, key string) DeleteResponse {
|
2022-11-29 13:08:58 +00:00
|
|
|
start := time.Now()
|
|
|
|
|
2022-09-27 16:01:43 +00:00
|
|
|
_, err := c.cli.DeleteObject(c.vu.Context(), &s3.DeleteObjectInput{
|
|
|
|
Bucket: aws.String(bucket),
|
|
|
|
Key: aws.String(key),
|
|
|
|
})
|
|
|
|
if err != nil {
|
2022-11-29 13:08:58 +00:00
|
|
|
stats.Report(c.vu, objDeleteFails, 1)
|
2022-09-27 16:01:43 +00:00
|
|
|
return DeleteResponse{Success: false, Error: err.Error()}
|
|
|
|
}
|
|
|
|
|
2022-12-30 10:47:28 +00:00
|
|
|
stats.Report(c.vu, objDeleteTotal, 1)
|
2022-11-29 13:08:58 +00:00
|
|
|
stats.Report(c.vu, objDeleteDuration, metrics.D(time.Since(start)))
|
2022-09-27 16:01:43 +00:00
|
|
|
return DeleteResponse{Success: true}
|
|
|
|
}
|
|
|
|
|
2022-05-20 09:34:34 +00:00
|
|
|
func (c *Client) Get(bucket, key string) GetResponse {
|
2022-05-12 19:37:38 +00:00
|
|
|
start := time.Now()
|
2022-09-02 19:23:33 +00:00
|
|
|
|
|
|
|
var objSize = 0
|
|
|
|
err := get(c.cli, bucket, key, func(chunk []byte) {
|
|
|
|
objSize += len(chunk)
|
2022-05-12 19:37:38 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
stats.Report(c.vu, objGetFails, 1)
|
2022-05-20 09:34:34 +00:00
|
|
|
return GetResponse{Success: false, Error: err.Error()}
|
2022-05-12 19:37:38 +00:00
|
|
|
}
|
2022-09-02 19:23:33 +00:00
|
|
|
|
2022-12-30 10:47:28 +00:00
|
|
|
stats.Report(c.vu, objGetTotal, 1)
|
2022-05-12 19:37:38 +00:00
|
|
|
stats.Report(c.vu, objGetDuration, metrics.D(time.Since(start)))
|
2022-09-02 19:23:33 +00:00
|
|
|
stats.ReportDataReceived(c.vu, float64(objSize))
|
|
|
|
return GetResponse{Success: true}
|
|
|
|
}
|
|
|
|
|
|
|
|
func get(
|
|
|
|
c *s3.Client,
|
|
|
|
bucket string,
|
|
|
|
key string,
|
|
|
|
onDataChunk func(chunk []byte),
|
|
|
|
) error {
|
|
|
|
var buf = make([]byte, 4*1024)
|
|
|
|
|
|
|
|
obj, err := c.GetObject(context.Background(), &s3.GetObjectInput{
|
|
|
|
Bucket: aws.String(bucket),
|
|
|
|
Key: aws.String(key),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-05-12 19:37:38 +00:00
|
|
|
for {
|
|
|
|
n, err := obj.Body.Read(buf)
|
|
|
|
if n > 0 {
|
2022-09-02 19:23:33 +00:00
|
|
|
onDataChunk(buf[:n])
|
2022-05-12 19:37:38 +00:00
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2022-09-02 19:23:33 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) VerifyHash(bucket, key, expectedHash string) VerifyHashResponse {
|
|
|
|
hasher := sha256.New()
|
|
|
|
err := get(c.cli, bucket, key, func(data []byte) {
|
|
|
|
hasher.Write(data)
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return VerifyHashResponse{Success: false, Error: err.Error()}
|
|
|
|
}
|
2022-09-22 16:57:21 +00:00
|
|
|
actualHash := hex.EncodeToString(hasher.Sum(nil))
|
2022-09-02 19:23:33 +00:00
|
|
|
if actualHash != expectedHash {
|
|
|
|
return VerifyHashResponse{Success: true, Error: "hash mismatch"}
|
|
|
|
}
|
|
|
|
|
|
|
|
return VerifyHashResponse{Success: true}
|
2022-05-12 19:37:38 +00:00
|
|
|
}
|
2022-06-17 10:37:43 +00:00
|
|
|
|
|
|
|
func (c *Client) CreateBucket(bucket string, params map[string]string) CreateBucketResponse {
|
|
|
|
var err error
|
|
|
|
var lockEnabled bool
|
|
|
|
if lockEnabledStr, ok := params["lock_enabled"]; ok {
|
|
|
|
if lockEnabled, err = strconv.ParseBool(lockEnabledStr); err != nil {
|
|
|
|
stats.Report(c.vu, createBucketFails, 1)
|
|
|
|
return CreateBucketResponse{Success: false, Error: "invalid lock_enabled params"}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var bucketConfiguration *types.CreateBucketConfiguration
|
|
|
|
if locationConstraint, ok := params["location_constraint"]; ok {
|
|
|
|
bucketConfiguration = &types.CreateBucketConfiguration{
|
|
|
|
LocationConstraint: types.BucketLocationConstraint(locationConstraint),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
start := time.Now()
|
|
|
|
_, err = c.cli.CreateBucket(c.vu.Context(), &s3.CreateBucketInput{
|
|
|
|
Bucket: aws.String(bucket),
|
|
|
|
ACL: types.BucketCannedACL(params["acl"]),
|
|
|
|
CreateBucketConfiguration: bucketConfiguration,
|
|
|
|
ObjectLockEnabledForBucket: lockEnabled,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
stats.Report(c.vu, createBucketFails, 1)
|
|
|
|
return CreateBucketResponse{Success: false, Error: err.Error()}
|
|
|
|
}
|
|
|
|
|
2022-12-30 10:47:28 +00:00
|
|
|
stats.Report(c.vu, createBucketTotal, 1)
|
2022-06-17 10:37:43 +00:00
|
|
|
stats.Report(c.vu, createBucketDuration, metrics.D(time.Since(start)))
|
|
|
|
return CreateBucketResponse{Success: true}
|
|
|
|
}
|