Skip to content

Commit

Permalink
Add support for finding EBS devices on Xen instances (aws#3971)
Browse files Browse the repository at this point in the history
  • Loading branch information
amogh09 authored and timj-hh committed Oct 23, 2023
1 parent 996a72c commit dcf15bd
Show file tree
Hide file tree
Showing 5 changed files with 314 additions and 43 deletions.

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

24 changes: 21 additions & 3 deletions ecs-agent/api/resource/ebs_discovery.go
Original file line number Diff line number Diff line change
Expand Up @@ -35,13 +35,31 @@ var (
)

type EBSDiscoveryClient struct {
ctx context.Context
ctx context.Context
hasXenSupport bool
}

func NewDiscoveryClient(ctx context.Context) *EBSDiscoveryClient {
return &EBSDiscoveryClient{
type EBSDiscoveryClientOption func(*EBSDiscoveryClient)

// Enable Xen instances support for EBS Discovery Client
func WithXenSupport() EBSDiscoveryClientOption {
return func(ec *EBSDiscoveryClient) {
ec.hasXenSupport = true
}
}

func NewDiscoveryClient(ctx context.Context, opts ...EBSDiscoveryClientOption) *EBSDiscoveryClient {
client := &EBSDiscoveryClient{
ctx: ctx,
}
for _, opt := range opts {
opt(client)
}
return client
}

func (client *EBSDiscoveryClient) HasXenSupport() bool {
return client.hasXenSupport
}

// ScanEBSVolumes will iterate through the entire list of provided EBS volume attachments within the agent state and checks if it's attached on the host.
Expand Down
55 changes: 52 additions & 3 deletions ecs-agent/api/resource/ebs_discovery_linux.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,11 @@ import (
"strings"
)

const (
sdPrefix = "sd"
xvdPrefix = "xvd"
)

// LsblkOutput is used to manage and track the output of `lsblk`
type LsblkOutput struct {
BlockDevices []BlockDevice `json:"blockdevices"`
Expand Down Expand Up @@ -56,7 +61,7 @@ func (api *EBSDiscoveryClient) ConfirmEBSVolumeIsAttached(deviceName, volumeID s
}

expectedVolumeId := strings.ReplaceAll(volumeID, "-", "")
actualDeviceName, err := parseLsblkOutput(&lsblkOut, deviceName, expectedVolumeId)
actualDeviceName, err := parseLsblkOutput(&lsblkOut, deviceName, expectedVolumeId, api.HasXenSupport())
if err != nil {
return "", err
}
Expand All @@ -78,13 +83,57 @@ func (api *EBSDiscoveryClient) ConfirmEBSVolumeIsAttached(deviceName, volumeID s
// }
// ]
// }
func parseLsblkOutput(output *LsblkOutput, deviceName string, volumeId string) (string, error) {
//
// If hasXenSupport is true then, in addition to matching a device by its serial, this function
// matches devices by their names disregarding "sd" or "xvd" prefix if it exists in
// device name in parseLsblkOutput and deviceName both.
// This is because on Xen instances the device name received from upstream with "sd" or "xvd"
// prefix might appear on the instance with the prefix interchanged, that is "xvd" instead of "sd"
// and vice-versa.
func parseLsblkOutput(
output *LsblkOutput,
deviceName, volumeId string,
hasXenSupport bool,
) (string, error) {
actualDeviceName := deviceName[strings.LastIndex(deviceName, "/")+1:]
for _, block := range output.BlockDevices {
//TODO: Add edge case for Xen-based instances
if block.Serial == volumeId {
return block.Name, nil
}
if hasXenSupport {
if foundDeviceName, found := matchXenBlockDevice(block, actualDeviceName); found {
return foundDeviceName, nil
}
}
}
return "", fmt.Errorf("cannot find EBS volume with device name: %v and volume ID: %v", actualDeviceName, volumeId)
}

// Matches a block device against a device name by matching the block device's name and
// the device name after stripping "sd" or "xvd" prefixes (whichever is present) from both names
// if it is present in both names.
func matchXenBlockDevice(block BlockDevice, deviceName string) (string, bool) {
if hasXenPrefix(block.Name) && hasXenPrefix(deviceName) {
if trimXenPrefix(block.Name) == trimXenPrefix(deviceName) {
return block.Name, true
}
} else if block.Name == deviceName {
return block.Name, true
}
return "", false
}

// Checks if the device name has sd or xvd prefix.
func hasXenPrefix(name string) bool {
return strings.HasPrefix(name, sdPrefix) || strings.HasPrefix(name, xvdPrefix)
}

// Trims "sd" or "xvd" prefix (whichever is present) from the given name.
func trimXenPrefix(name string) string {
if strings.HasPrefix(name, sdPrefix) {
return strings.TrimPrefix(name, sdPrefix)
} else if strings.HasPrefix(name, xvdPrefix) {
return strings.TrimPrefix(name, xvdPrefix)
}
return name
}
Loading

0 comments on commit dcf15bd

Please sign in to comment.