diff --git a/pkg/cloudmeta/aws.go b/pkg/cloudmeta/aws.go new file mode 100644 index 000000000..eeb38283b --- /dev/null +++ b/pkg/cloudmeta/aws.go @@ -0,0 +1,50 @@ +// Copyright (C) 2024 ScyllaDB + +package cloudmeta + +import ( + "context" + + "github.com/aws/aws-sdk-go/aws" + "github.com/aws/aws-sdk-go/aws/ec2metadata" + "github.com/aws/aws-sdk-go/aws/session" + "github.com/pkg/errors" +) + +// AWSMetadata is a wrapper around ec2 metadata client. +type AWSMetadata struct { + ec2meta *ec2metadata.EC2Metadata +} + +// NewAWSMetadata is a constructor for AWSMetadata service. +// testEndpoint can be provided if you want to overwrite the default metadata endpoint, otherwise leave it empty. +func NewAWSMetadata(testEndpoint string) (*AWSMetadata, error) { + session, err := session.NewSession() + if err != nil { + return nil, errors.Wrap(err, "session.NewSession") + } + cfg := aws.NewConfig() + if testEndpoint != "" { + cfg = cfg.WithEndpoint(testEndpoint) + } + return &AWSMetadata{ + ec2meta: ec2metadata.New(session, cfg), + }, nil +} + +// Metadata return InstanceMetadata from aws if available. +func (aws *AWSMetadata) Metadata(ctx context.Context) (InstanceMetadata, error) { + if !aws.ec2meta.AvailableWithContext(ctx) { + return InstanceMetadata{}, errors.New("metadata is not available") + } + + instanceData, err := aws.ec2meta.GetInstanceIdentityDocumentWithContext(ctx) + if err != nil { + return InstanceMetadata{}, errors.Wrap(err, "aws.metadataClient.GetInstanceIdentityDocument") + } + + return InstanceMetadata{ + CloudProvider: CloudProviderAWS, + InstanceType: instanceData.InstanceType, + }, nil +} diff --git a/pkg/cloudmeta/metadata.go b/pkg/cloudmeta/metadata.go index 23822b708..0fc1e9796 100644 --- a/pkg/cloudmeta/metadata.go +++ b/pkg/cloudmeta/metadata.go @@ -31,8 +31,15 @@ type CloudMeta struct { // NewCloudMeta creates new CloudMeta provider. func NewCloudMeta() (*CloudMeta, error) { - // providers will initialized here and added to CloudMeta.providers. - return &CloudMeta{}, nil + awsMeta, err := NewAWSMetadata("") + if err != nil { + return nil, err + } + return &CloudMeta{ + providers: []CloudMetadataProvider{ + awsMeta, + }, + }, nil } // GetInstanceMetadata tries to fetch instance metadata from AWS, GCP, Azure providers in order.