mirror of
https://github.com/edgelesssys/constellation.git
synced 2025-01-07 05:38:03 -05:00
118 lines
3.9 KiB
Go
118 lines
3.9 KiB
Go
|
/*
|
||
|
Copyright (c) Edgeless Systems GmbH
|
||
|
|
||
|
SPDX-License-Identifier: AGPL-3.0-only
|
||
|
*/
|
||
|
|
||
|
package idkeydigest
|
||
|
|
||
|
import (
|
||
|
"encoding/hex"
|
||
|
"encoding/json"
|
||
|
"fmt"
|
||
|
|
||
|
"github.com/edgelesssys/constellation/v2/internal/cloud/cloudprovider"
|
||
|
"go.uber.org/multierr"
|
||
|
)
|
||
|
|
||
|
// IDKeyDigests is a list of trusted digest values for the ID key.
|
||
|
type IDKeyDigests [][]byte
|
||
|
|
||
|
type encodedIDKeyDigests []string
|
||
|
|
||
|
// encodedDigestLength is the length of a digest in hex encoding.
|
||
|
const encodedDigestLength = 2 * 48
|
||
|
|
||
|
// NewIDKeyDigests creates a new IDKeyDigests from a list of digests.
|
||
|
func NewIDKeyDigests(digests [][]byte) IDKeyDigests {
|
||
|
idKeyDigests := make(IDKeyDigests, len(digests))
|
||
|
copy(idKeyDigests, digests)
|
||
|
return idKeyDigests
|
||
|
}
|
||
|
|
||
|
// DefaultsFor returns the default IDKeyDigests for the given cloud provider.
|
||
|
func DefaultsFor(csp cloudprovider.Provider) IDKeyDigests {
|
||
|
switch csp {
|
||
|
case cloudprovider.Azure:
|
||
|
return IDKeyDigests{
|
||
|
{0x57, 0x48, 0x6a, 0x44, 0x7e, 0xc0, 0xf1, 0x95, 0x80, 0x02, 0xa2, 0x2a, 0x06, 0xb7, 0x67, 0x3b, 0x9f, 0xd2, 0x7d, 0x11, 0xe1, 0xc6, 0x52, 0x74, 0x98, 0x05, 0x60, 0x54, 0xc5, 0xfa, 0x92, 0xd2, 0x3c, 0x50, 0xf9, 0xde, 0x44, 0x07, 0x27, 0x60, 0xfe, 0x2b, 0x6f, 0xb8, 0x97, 0x40, 0xb6, 0x96},
|
||
|
{0x03, 0x56, 0x21, 0x58, 0x82, 0xa8, 0x25, 0x27, 0x9a, 0x85, 0xb3, 0x00, 0xb0, 0xb7, 0x42, 0x93, 0x1d, 0x11, 0x3b, 0xf7, 0xe3, 0x2d, 0xde, 0x2e, 0x50, 0xff, 0xde, 0x7e, 0xc7, 0x43, 0xca, 0x49, 0x1e, 0xcd, 0xd7, 0xf3, 0x36, 0xdc, 0x28, 0xa6, 0xe0, 0xb2, 0xbb, 0x57, 0xaf, 0x7a, 0x44, 0xa3},
|
||
|
}
|
||
|
default:
|
||
|
return nil
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// MarshalYAML implements the yaml.Marshaler interface.
|
||
|
func (d IDKeyDigests) MarshalYAML() (any, error) {
|
||
|
encodedIDKeyDigests := []string{}
|
||
|
for _, digest := range d {
|
||
|
encodedIDKeyDigests = append(encodedIDKeyDigests, hex.EncodeToString(digest))
|
||
|
}
|
||
|
return encodedIDKeyDigests, nil
|
||
|
}
|
||
|
|
||
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
||
|
func (d *IDKeyDigests) UnmarshalYAML(unmarshal func(any) error) error {
|
||
|
var encodedDigests encodedIDKeyDigests
|
||
|
if err := unmarshal(&encodedDigests); err != nil {
|
||
|
// Unmarshalling failed, IDKeyDigests might be a simple string instead of IDKeyDigests struct.
|
||
|
var unmarshalledString string
|
||
|
if legacyErr := unmarshal(&unmarshalledString); legacyErr != nil {
|
||
|
return multierr.Append(
|
||
|
err,
|
||
|
fmt.Errorf("trying legacy format: %w", legacyErr),
|
||
|
)
|
||
|
}
|
||
|
encodedDigests = append(encodedDigests, unmarshalledString)
|
||
|
}
|
||
|
if err := d.unmarshal(encodedDigests); err != nil {
|
||
|
return fmt.Errorf("unmarshalling yaml: %w", err)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// MarshalJSON implements the json.Marshaler interface.
|
||
|
func (d IDKeyDigests) MarshalJSON() ([]byte, error) {
|
||
|
encodedIDKeyDigests := []string{}
|
||
|
for _, digest := range d {
|
||
|
encodedIDKeyDigests = append(encodedIDKeyDigests, hex.EncodeToString(digest))
|
||
|
}
|
||
|
return json.Marshal(encodedIDKeyDigests)
|
||
|
}
|
||
|
|
||
|
// UnmarshalJSON implements the json.Unmarshaler interface.
|
||
|
func (d *IDKeyDigests) UnmarshalJSON(b []byte) error {
|
||
|
var encodedDigests encodedIDKeyDigests
|
||
|
if err := json.Unmarshal(b, &encodedDigests); err != nil {
|
||
|
// Unmarshalling failed, IDKeyDigests might be a simple string instead of IDKeyDigests struct.
|
||
|
var unmarshalledString string
|
||
|
if legacyErr := json.Unmarshal(b, &unmarshalledString); legacyErr != nil {
|
||
|
return multierr.Append(
|
||
|
err,
|
||
|
fmt.Errorf("trying legacy format: %w", legacyErr),
|
||
|
)
|
||
|
}
|
||
|
encodedDigests = []string{unmarshalledString}
|
||
|
}
|
||
|
if err := d.unmarshal(encodedDigests); err != nil {
|
||
|
return fmt.Errorf("unmarshalling json: %w", err)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// unmarshal is a helper function for unmarshalling encodedIDKeyDigests into IDKeyDigests.
|
||
|
func (d *IDKeyDigests) unmarshal(encodedDigests encodedIDKeyDigests) error {
|
||
|
for _, encodedDigest := range encodedDigests {
|
||
|
if len(encodedDigest) != encodedDigestLength {
|
||
|
return fmt.Errorf("invalid digest length: %d", len(encodedDigest))
|
||
|
}
|
||
|
digest, err := hex.DecodeString(encodedDigest)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("decoding digest: %w", err)
|
||
|
}
|
||
|
*d = append(*d, digest)
|
||
|
}
|
||
|
return nil
|
||
|
}
|