/* Copyright (c) Edgeless Systems GmbH SPDX-License-Identifier: AGPL-3.0-only */ package azure import ( "context" "errors" "fmt" "strings" armcomputev2 "github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/compute/armcompute/v2" "github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/network/armnetwork" ) // getVMInterfaces retrieves all network interfaces referenced by a virtual machine. func (m *Metadata) getVMInterfaces(ctx context.Context, vm armcomputev2.VirtualMachine, resourceGroup string) ([]armnetwork.Interface, error) { if vm.Properties == nil || vm.Properties.NetworkProfile == nil { return []armnetwork.Interface{}, nil } interfaceNames := extractInterfaceNamesFromInterfaceReferences(vm.Properties.NetworkProfile.NetworkInterfaces) networkInterfaces := []armnetwork.Interface{} for _, interfaceName := range interfaceNames { networkInterfacesResp, err := m.networkInterfacesAPI.Get(ctx, resourceGroup, interfaceName, nil) if err != nil { return nil, fmt.Errorf("retrieving network interface %v: %w", interfaceName, err) } networkInterfaces = append(networkInterfaces, networkInterfacesResp.Interface) } return networkInterfaces, nil } // getScaleSetVMInterfaces retrieves all network interfaces referenced by a scale set virtual machine. func (m *Metadata) getScaleSetVMInterfaces(ctx context.Context, vm armcomputev2.VirtualMachineScaleSetVM, resourceGroup, scaleSet, instanceID string) ([]armnetwork.Interface, error) { if vm.Properties == nil || vm.Properties.NetworkProfile == nil { return []armnetwork.Interface{}, nil } interfaceNames := extractInterfaceNamesFromInterfaceReferences(vm.Properties.NetworkProfile.NetworkInterfaces) networkInterfaces := []armnetwork.Interface{} for _, interfaceName := range interfaceNames { networkInterfacesResp, err := m.networkInterfacesAPI.GetVirtualMachineScaleSetNetworkInterface(ctx, resourceGroup, scaleSet, instanceID, interfaceName, nil) if err != nil { return nil, fmt.Errorf("retrieving network interface %v: %w", interfaceName, err) } networkInterfaces = append(networkInterfaces, networkInterfacesResp.Interface) } return networkInterfaces, nil } // getScaleSetVMPublicIPAddress retrieves the primary public IP address from a network interface which is referenced by a scale set virtual machine. func (m *Metadata) getScaleSetVMPublicIPAddress(ctx context.Context, resourceGroup, scaleSet, instanceID string, networkInterfaces []armnetwork.Interface, ) (string, error) { for _, networkInterface := range networkInterfaces { if networkInterface.Properties == nil || networkInterface.Name == nil { continue } for _, config := range networkInterface.Properties.IPConfigurations { if config == nil || config.Name == nil || config.Properties == nil || config.Properties.PublicIPAddress == nil || config.Properties.Primary == nil || !*config.Properties.Primary { continue } publicIPAddressName := *config.Properties.PublicIPAddress.ID publicIPAddressNameParts := strings.Split(publicIPAddressName, "/") publicIPAddressName = publicIPAddressNameParts[len(publicIPAddressNameParts)-1] publicIPAddress, err := m.publicIPAddressesAPI.GetVirtualMachineScaleSetPublicIPAddress(ctx, resourceGroup, scaleSet, instanceID, *networkInterface.Name, *config.Name, publicIPAddressName, nil) if err != nil { return "", fmt.Errorf("failed to retrieve public ip address %v: %w", publicIPAddressName, err) } if publicIPAddress.Properties == nil || publicIPAddress.Properties.IPAddress == nil { return "", errors.New("retrieved public ip address has invalid ip address") } return *publicIPAddress.Properties.IPAddress, nil } } // instances may have no public IP, in that case we don't return an error. return "", nil } // extractVPCIP extracts the primary VPC IP from a list of network interface IP configurations. func extractVPCIP(networkInterfaces []armnetwork.Interface) string { for _, networkInterface := range networkInterfaces { if networkInterface.Properties == nil || len(networkInterface.Properties.IPConfigurations) == 0 { continue } for _, config := range networkInterface.Properties.IPConfigurations { if config == nil || config.Properties == nil || config.Properties.PrivateIPAddress == nil || config.Properties.Primary == nil { continue } if *config.Properties.Primary { return *config.Properties.PrivateIPAddress } } } return "" } // extractInterfaceNamesFromInterfaceReferences extracts the name of a network interface from a reference id. // Format: // - "/subscriptions/<subscription>/resourceGroups/<resource-group>/providers/Microsoft.Network/networkInterfaces/<interface-name>" // - "/subscriptions/<subscription>/resourceGroups/<resource-group>/providers/Microsoft.Compute/virtualMachineScaleSets/<scale-set-name>/virtualMachines/<instanceID>/networkInterfaces/<interface-name>". func extractInterfaceNamesFromInterfaceReferences(references []*armcomputev2.NetworkInterfaceReference) []string { interfaceNames := []string{} for _, interfaceReference := range references { if interfaceReference == nil || interfaceReference.ID == nil { continue } interfaceIDParts := strings.Split(*interfaceReference.ID, "/") if len(interfaceIDParts) < 1 { continue } interfaceName := interfaceIDParts[len(interfaceIDParts)-1] interfaceNames = append(interfaceNames, interfaceName) } return interfaceNames }