Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for finding EBS devices on Xen instances #3971

Merged
merged 5 commits into from
Oct 20, 2023
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view

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
53 changes: 50 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,55 @@ 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 a device by its name disregarding "sd" or "xvd" prefix (whichever is present).
fierlion marked this conversation as resolved.
Show resolved Hide resolved
// 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.
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