constellation/internal/cloud/openstack/imds.go

321 lines
8.8 KiB
Go

/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package openstack
import (
"context"
"encoding/json"
"errors"
"fmt"
"io"
"net/http"
"time"
"github.com/edgelesssys/constellation/v2/internal/cloud"
"github.com/edgelesssys/constellation/v2/internal/role"
)
// documentation of OpenStack Metadata Service: https://docs.openstack.org/nova/rocky/user/metadata-service.html
const (
imdsMetaDataURL = "http://169.254.169.254/openstack/2018-08-27/meta_data.json"
imdsUserDataURL = "http://169.254.169.254/openstack/2018-08-27/user_data"
ec2ImdsBaseURL = "http://169.254.169.254/1.0/meta-data"
maxCacheAge = 12 * time.Hour
)
type imdsClient struct {
client httpClient
vpcIPCache string
vpcIPCacheTime time.Time
cache metadataResponse
userDataCache userDataResponse
cacheTime time.Time
}
// providerID returns the provider ID of the instance the function is called from.
func (c *imdsClient) providerID(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.cache.UUID == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.cache.UUID == "" {
return "", errors.New("unable to get provider id")
}
return c.cache.UUID, nil
}
func (c *imdsClient) name(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.cache.Name == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.cache.Name == "" {
return "", errors.New("unable to get name")
}
return c.cache.Name, nil
}
// projectID returns the project ID of the instance the function
// is called from.
func (c *imdsClient) projectID(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.cache.ProjectID == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.cache.ProjectID == "" {
return "", errors.New("unable to get project id")
}
return c.cache.ProjectID, nil
}
// uid returns the UID of the cluster, based on the tags on the instance
// the function is called from, which are inherited from the scale set.
func (c *imdsClient) uid(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || len(c.cache.Tags.UID) == 0 {
if err := c.update(ctx); err != nil {
return "", err
}
}
if len(c.cache.Tags.UID) == 0 {
return "", fmt.Errorf("unable to get uid from metadata tags %v", c.cache.Tags)
}
return c.cache.Tags.UID, nil
}
// initSecretHash returns the hash of the init secret of the cluster, based on the tags on the instance
// the function is called from, which are inherited from the scale set.
func (c *imdsClient) initSecretHash(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || len(c.cache.Tags.InitSecretHash) == 0 {
if err := c.update(ctx); err != nil {
return "", err
}
}
if len(c.cache.Tags.InitSecretHash) == 0 {
return "", fmt.Errorf("unable to get tag %s from metadata tags %v", cloud.TagInitSecretHash, c.cache.Tags)
}
return c.cache.Tags.InitSecretHash, nil
}
// role returns the role of the instance the function is called from.
func (c *imdsClient) role(ctx context.Context) (role.Role, error) {
if c.timeForUpdate(c.cacheTime) || len(c.cache.Tags.Role) == 0 {
if err := c.update(ctx); err != nil {
return role.Unknown, err
}
}
if len(c.cache.Tags.Role) == 0 {
return role.Unknown, fmt.Errorf("unable to get role from metadata tags %v", c.cache.Tags)
}
return role.FromString(c.cache.Tags.Role), nil
}
func (c *imdsClient) loadBalancerEndpoint(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.userDataCache.LoadBalancerEndpoint == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.userDataCache.LoadBalancerEndpoint == "" {
return "", errors.New("unable to get load balancer endpoint")
}
return c.userDataCache.LoadBalancerEndpoint, nil
}
func (c *imdsClient) authURL(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.userDataCache.AuthURL == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.userDataCache.AuthURL == "" {
return "", errors.New("unable to get auth url")
}
return c.userDataCache.AuthURL, nil
}
func (c *imdsClient) userDomainName(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.userDataCache.UserDomainName == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.userDataCache.UserDomainName == "" {
return "", errors.New("unable to get user domain name")
}
return c.userDataCache.UserDomainName, nil
}
func (c *imdsClient) regionName(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.userDataCache.RegionName == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.userDataCache.RegionName == "" {
return "", errors.New("unable to get user domain name")
}
return c.userDataCache.RegionName, nil
}
func (c *imdsClient) username(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.userDataCache.Username == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.userDataCache.Username == "" {
return "", errors.New("unable to get token name")
}
return c.userDataCache.Username, nil
}
func (c *imdsClient) password(ctx context.Context) (string, error) {
if c.timeForUpdate(c.cacheTime) || c.userDataCache.Password == "" {
if err := c.update(ctx); err != nil {
return "", err
}
}
if c.userDataCache.Password == "" {
return "", errors.New("unable to get token password")
}
return c.userDataCache.Password, nil
}
// timeForUpdate checks whether an update is needed due to cache age.
func (c *imdsClient) timeForUpdate(t time.Time) bool {
return time.Since(t) > maxCacheAge
}
func (c *imdsClient) update(ctx context.Context) error {
if err := c.updateInstanceMetadata(ctx); err != nil {
return fmt.Errorf("updating instance metadata: %w", err)
}
if err := c.updateUserData(ctx); err != nil {
return fmt.Errorf("updating user data: %w", err)
}
c.cacheTime = time.Now()
return nil
}
// update updates instance metadata from the azure imds API.
func (c *imdsClient) updateInstanceMetadata(ctx context.Context) error {
resp, err := httpGet(ctx, c.client, imdsMetaDataURL)
if err != nil {
return err
}
var metadataResp metadataResponse
if err := json.Unmarshal(resp, &metadataResp); err != nil {
return fmt.Errorf("unmarshalling IMDS metadata response %q: %w", resp, err)
}
c.cache = metadataResp
return nil
}
func (c *imdsClient) updateUserData(ctx context.Context) error {
resp, err := httpGet(ctx, c.client, imdsUserDataURL)
if err != nil {
return err
}
var userdataResp userDataResponse
if err := json.Unmarshal(resp, &userdataResp); err != nil {
return fmt.Errorf("unmarshalling IMDS user_data response %q: %w", resp, err)
}
c.userDataCache = userdataResp
return nil
}
func (c *imdsClient) vpcIP(ctx context.Context) (string, error) {
const path = "local-ipv4"
if c.timeForUpdate(c.vpcIPCacheTime) || c.vpcIPCache == "" {
resp, err := httpGet(ctx, c.client, ec2ImdsBaseURL+"/"+path)
if err != nil {
return "", err
}
c.vpcIPCache = string(resp)
c.vpcIPCacheTime = time.Now()
}
if c.vpcIPCache == "" {
return "", errors.New("unable to get vpc ip")
}
return c.vpcIPCache, nil
}
func httpGet(ctx context.Context, c httpClient, url string) ([]byte, error) {
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, http.NoBody)
if err != nil {
return nil, err
}
resp, err := c.Do(req)
if err != nil {
return nil, fmt.Errorf("querying the OpenStack IMDS api failed for %q: %w", url, err)
}
if resp.StatusCode < 200 || resp.StatusCode >= 300 {
return nil, fmt.Errorf("IMDS api might be broken for this server. Recreate the cluster if this issue persists. Querying the OpenStack IMDS api failed for %q with error code %d", url, resp.StatusCode)
}
defer resp.Body.Close()
return io.ReadAll(resp.Body)
}
// metadataResponse contains metadataResponse with only the required values.
type metadataResponse struct {
UUID string `json:"uuid,omitempty"`
ProjectID string `json:"project_id,omitempty"`
Name string `json:"name,omitempty"`
Tags metadataTags `json:"meta,omitempty"`
}
type metadataTags struct {
InitSecretHash string `json:"constellation-init-secret-hash,omitempty"`
Role string `json:"constellation-role,omitempty"`
UID string `json:"constellation-uid,omitempty"`
}
type userDataResponse struct {
AuthURL string `json:"openstack-auth-url,omitempty"`
UserDomainName string `json:"openstack-user-domain-name,omitempty"`
RegionName string `json:"openstack-region-name,omitempty"`
Username string `json:"openstack-username,omitempty"`
Password string `json:"openstack-password,omitempty"`
LoadBalancerEndpoint string `json:"openstack-load-balancer-endpoint,omitempty"`
}
type httpClient interface {
Do(req *http.Request) (*http.Response, error)
}