This repository was archived by the owner on Mar 6, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 85
/
Copy pathclient_setup.go
116 lines (100 loc) · 3.14 KB
/
client_setup.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
package cloudaws
import (
"context"
"errors"
"fmt"
"github.com/aws/aws-sdk-go-v2/aws"
"github.com/aws/aws-sdk-go-v2/config"
"github.com/aws/aws-sdk-go-v2/service/iam"
iamtypes "github.com/aws/aws-sdk-go-v2/service/iam/types"
"github.com/aws/aws-sdk-go-v2/service/s3"
s3types "github.com/aws/aws-sdk-go-v2/service/s3/types"
"time"
)
const (
// DefaultInstanceRegions is the default regions to use for spot instances
DefaultInstanceRegions = "us-west-2,us-west-1,us-east-2"
)
type SetupClient struct {
cfg aws.Config
name string
region string
}
func NewSetupClient(name, region string) (*SetupClient, error) {
cfg, err := config.LoadDefaultConfig(context.Background(), config.WithRegion(region))
if err != nil {
return nil, fmt.Errorf("failed to load default aws config: %w", err)
}
if err := checkS3Access(cfg); err != nil {
return nil, err
}
return &SetupClient{
cfg: cfg,
name: name,
region: region,
}, nil
}
func (c *SetupClient) Setup(ctx context.Context) error {
if err := c.s3BucketSetup(ctx); err != nil {
return err
}
if err := c.serviceLinkedRolesSetup(ctx); err != nil {
return err
}
return nil
}
func (c *SetupClient) serviceLinkedRolesSetup(ctx context.Context) error {
iamClient := iam.NewFromConfig(c.cfg)
_, err := iamClient.CreateServiceLinkedRole(ctx, &iam.CreateServiceLinkedRoleInput{
AWSServiceName: aws.String("spot.amazonaws.com"),
})
if err != nil {
var invalidInputException *iamtypes.InvalidInputException
if errors.Is(err, invalidInputException) {
return fmt.Errorf("failed to create spot.amazonaws.com service linked role: %w", err)
}
}
_, err = iamClient.CreateServiceLinkedRole(ctx, &iam.CreateServiceLinkedRoleInput{
AWSServiceName: aws.String("spotfleet.amazonaws.com"),
})
if err != nil {
var invalidInputException *iamtypes.InvalidInputException
if errors.Is(err, invalidInputException) {
return fmt.Errorf("failed to create spotfleet.amazonaws.com service linked role: %w", err)
}
}
return nil
}
func (c *SetupClient) s3BucketSetup(ctx context.Context) error {
s3Client := s3.NewFromConfig(c.cfg)
_, err := s3Client.HeadBucket(ctx, &s3.HeadBucketInput{Bucket: &c.name})
if err != nil {
var noSuchBucket *s3types.NoSuchBucket
if !errors.As(err, &noSuchBucket) {
return fmt.Errorf("unknown S3 error: %w", err)
}
bucketInput := &s3.CreateBucketInput{
Bucket: &c.name,
}
if c.region != "us-east-1" {
bucketInput.CreateBucketConfiguration = &s3types.CreateBucketConfiguration{
LocationConstraint: s3types.BucketLocationConstraint(c.region),
}
}
output, err := s3Client.CreateBucket(ctx, bucketInput)
if err != nil {
return fmt.Errorf("failed to create bucket %v - note that this bucket name must be globally unique: output:%v err:%w", c.name, output, err)
}
}
return nil
}
func checkS3Access(cfg aws.Config) error {
ctx, cancel := context.WithTimeout(context.Background(), time.Second * 10)
defer cancel()
s3Client := s3.NewFromConfig(cfg)
_, err := s3Client.ListBuckets(ctx, &s3.ListBucketsInput{})
if err != nil {
return fmt.Errorf("unable to list S3 buckets - make sure you have valid admin AWS credentials: %w", err)
}
return nil
}