kops/protokube/pkg/gossip/azure/client.go

199 lines
5.5 KiB
Go

/*
Copyright 2020 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package azure
import (
"context"
"encoding/json"
"fmt"
"io"
"net/http"
"strings"
"github.com/Azure/azure-sdk-for-go/sdk/azidentity"
compute "github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/compute/armcompute"
network "github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/network/armnetwork"
)
type instanceComputeMetadata struct {
Name string `json:"name"`
ResourceGroupName string `json:"resourceGroupName"`
SubscriptionID string `json:"subscriptionId"`
// Tags is a list of tags separated by ';'. Each tag
// is of the form "key:value".
Tags string `json:"tags"`
}
func (m *instanceComputeMetadata) GetTags() (map[string]string, error) {
tags := map[string]string{}
l := strings.Split(m.Tags, ";")
for _, t := range l {
tl := strings.SplitN(t, ":", 2)
if len(tl) != 2 {
return nil, fmt.Errorf("unexpected tag format: %s", tl)
}
tags[tl[0]] = tl[1]
}
return tags, nil
}
type ipAddress struct {
PrivateIPAddress string `json:"privateIpAddress"`
PublicIPAddress string `json:"publicIpAddress"`
}
type ipv4Interface struct {
IPAddresses []*ipAddress `json:"ipAddress"`
}
type networkInterface struct {
IPv4 *ipv4Interface `json:"ipv4"`
}
type instanceNetworkMetadata struct {
Interfaces []*networkInterface `json:"interface"`
}
type instanceMetadata struct {
Compute *instanceComputeMetadata `json:"compute"`
Network *instanceNetworkMetadata `json:"network"`
}
// Client is an Azure client.
type Client struct {
metadata *instanceMetadata
vmssesClient *compute.VirtualMachineScaleSetsClient
interfacesClient *network.InterfacesClient
}
// NewClient returns a new Client.
func NewClient() (*Client, error) {
m, err := queryInstanceMetadata()
if err != nil {
return nil, fmt.Errorf("error querying instance metadata: %s", err)
}
if m.Compute.SubscriptionID == "" {
return nil, fmt.Errorf("empty subscription name")
}
if m.Compute.ResourceGroupName == "" {
return nil, fmt.Errorf("empty resource group name")
}
cred, err := azidentity.NewDefaultAzureCredential(nil)
if err != nil {
return nil, fmt.Errorf("error creating an identity: %s", err)
}
vmssesClient, err := compute.NewVirtualMachineScaleSetsClient(m.Compute.SubscriptionID, cred, nil)
if err != nil {
return nil, fmt.Errorf("error creating VMSS client: %s", err)
}
interfacesClient, err := network.NewInterfacesClient(m.Compute.SubscriptionID, cred, nil)
if err != nil {
return nil, fmt.Errorf("error creating interfaces client: %s", err)
}
return &Client{
metadata: m,
vmssesClient: vmssesClient,
interfacesClient: interfacesClient,
}, nil
}
func (c *Client) resourceGroupName() string {
return c.metadata.Compute.ResourceGroupName
}
// GetName returns the name of the VM.
func (c *Client) GetName() string {
return c.metadata.Compute.Name
}
// GetTags returns the tags of the VM queried from Instance Metadata Service.
func (c *Client) GetTags() (map[string]string, error) {
return c.metadata.Compute.GetTags()
}
// ListVMScaleSets returns VM ScaleSets in the resource group.
func (c *Client) ListVMScaleSets(ctx context.Context) ([]*compute.VirtualMachineScaleSet, error) {
var l []*compute.VirtualMachineScaleSet
pager := c.vmssesClient.NewListPager(c.resourceGroupName(), nil)
for pager.More() {
resp, err := pager.NextPage(ctx)
if err != nil {
return nil, err
}
l = append(l, resp.Value...)
}
return l, nil
}
// ListVMSSNetworkInterfaces returns the interfaces that the specified VM ScaleSet has.
func (c *Client) ListVMSSNetworkInterfaces(ctx context.Context, vmScaleSetName string) ([]*network.Interface, error) {
var l []*network.Interface
pager := c.interfacesClient.NewListPager(c.resourceGroupName(), nil)
for pager.More() {
resp, err := pager.NextPage(ctx)
if err != nil {
return nil, err
}
l = append(l, resp.Value...)
}
return l, nil
}
// queryInstanceMetadata queries Azure Instance Metadata documented in
// https://docs.microsoft.com/en-us/azure/virtual-machines/windows/instance-metadata-service.
func queryInstanceMetadata() (*instanceMetadata, error) {
client := &http.Client{}
req, err := http.NewRequest("GET", "http://169.254.169.254/metadata/instance", nil)
if err != nil {
return nil, fmt.Errorf("error creating a new request: %s", err)
}
req.Header.Add("Metadata", "True")
q := req.URL.Query()
q.Add("format", "json")
q.Add("api-version", "2020-06-01")
req.URL.RawQuery = q.Encode()
resp, err := client.Do(req)
if err != nil {
return nil, fmt.Errorf("error sending request to the metadata server: %s", err)
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return nil, fmt.Errorf("error reading a response from the metadata server: %s", err)
}
metadata, err := unmarshalInstanceMetadata(body)
if err != nil {
return nil, fmt.Errorf("error unmarshalling metadata: %s", err)
}
return metadata, nil
}
func unmarshalInstanceMetadata(data []byte) (*instanceMetadata, error) {
m := &instanceMetadata{}
if err := json.Unmarshal(data, &m); err != nil {
return nil, err
}
return m, nil
}