2023-06-07 10:16:32 -04:00
|
|
|
/*
|
|
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
|
|
|
package attestationconfigapi
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
apifetcher "github.com/edgelesssys/constellation/v2/internal/api/fetcher"
|
2023-11-14 04:03:01 -05:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/attestation/variant"
|
2023-06-07 10:16:32 -04:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/constants"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/sigstore"
|
|
|
|
)
|
|
|
|
|
|
|
|
const cosignPublicKey = constants.CosignPublicKeyReleases
|
|
|
|
|
|
|
|
// Fetcher fetches config API resources without authentication.
|
|
|
|
type Fetcher interface {
|
2024-09-05 03:59:26 -04:00
|
|
|
FetchLatestVersion(ctx context.Context, attestation fmt.Stringer) (Entry, error)
|
2023-06-07 10:16:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// fetcher fetches AttestationCfg API resources without authentication.
|
|
|
|
type fetcher struct {
|
|
|
|
apifetcher.HTTPClient
|
2023-09-25 05:53:02 -04:00
|
|
|
cdnURL string
|
2023-06-09 06:48:12 -04:00
|
|
|
verifier sigstore.Verifier
|
2023-06-07 10:16:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewFetcher returns a new apifetcher.
|
|
|
|
func NewFetcher() Fetcher {
|
2023-09-25 05:53:02 -04:00
|
|
|
return NewFetcherWithClient(apifetcher.NewHTTPClient(), constants.CDNRepositoryURL)
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewFetcherWithCustomCDNAndCosignKey returns a new fetcher with custom CDN URL.
|
|
|
|
func NewFetcherWithCustomCDNAndCosignKey(cdnURL, cosignKey string) Fetcher {
|
|
|
|
verifier, err := sigstore.NewCosignVerifier([]byte(cosignKey))
|
|
|
|
if err != nil {
|
|
|
|
// This relies on an embedded public key. If this key can not be validated, there is no way to recover from this.
|
|
|
|
panic(fmt.Errorf("creating cosign verifier: %w", err))
|
|
|
|
}
|
|
|
|
return newFetcherWithClientAndVerifier(apifetcher.NewHTTPClient(), verifier, cdnURL)
|
2023-06-07 10:16:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewFetcherWithClient returns a new fetcher with custom http client.
|
2023-09-25 05:53:02 -04:00
|
|
|
func NewFetcherWithClient(client apifetcher.HTTPClient, cdnURL string) Fetcher {
|
2023-08-01 10:48:13 -04:00
|
|
|
verifier, err := sigstore.NewCosignVerifier([]byte(cosignPublicKey))
|
|
|
|
if err != nil {
|
|
|
|
// This relies on an embedded public key. If this key can not be validated, there is no way to recover from this.
|
|
|
|
panic(fmt.Errorf("creating cosign verifier: %w", err))
|
|
|
|
}
|
2023-09-25 05:53:02 -04:00
|
|
|
return newFetcherWithClientAndVerifier(client, verifier, cdnURL)
|
2023-06-09 06:48:12 -04:00
|
|
|
}
|
|
|
|
|
2023-09-25 05:53:02 -04:00
|
|
|
func newFetcherWithClientAndVerifier(client apifetcher.HTTPClient, cosignVerifier sigstore.Verifier, url string) Fetcher {
|
|
|
|
return &fetcher{HTTPClient: client, verifier: cosignVerifier, cdnURL: url}
|
2023-06-07 10:16:32 -04:00
|
|
|
}
|
|
|
|
|
2024-06-11 09:12:05 -04:00
|
|
|
// FetchLatestVersion returns the latest versions of the given type.
|
2024-09-05 03:59:26 -04:00
|
|
|
func (f *fetcher) FetchLatestVersion(ctx context.Context, variant fmt.Stringer) (Entry, error) {
|
2024-06-11 10:25:24 -04:00
|
|
|
list, err := f.fetchVersionList(ctx, variant)
|
2024-06-11 09:12:05 -04:00
|
|
|
if err != nil {
|
2024-06-12 10:30:03 -04:00
|
|
|
return Entry{}, err
|
2024-06-11 09:12:05 -04:00
|
|
|
}
|
|
|
|
|
2024-06-11 10:25:24 -04:00
|
|
|
// latest version is first in list
|
|
|
|
return f.fetchVersion(ctx, list.List[0], variant)
|
2024-06-11 09:12:05 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// fetchVersionList fetches the version list information from the config API.
|
2024-09-05 03:59:26 -04:00
|
|
|
func (f *fetcher) fetchVersionList(ctx context.Context, attestationVariant fmt.Stringer) (List, error) {
|
|
|
|
parsedVariant, err := variant.FromString(attestationVariant.String())
|
|
|
|
if err != nil {
|
|
|
|
return List{}, fmt.Errorf("parsing variant: %w", err)
|
|
|
|
}
|
|
|
|
fetchedList, err := apifetcher.FetchAndVerify(ctx, f.HTTPClient, f.cdnURL, List{Variant: parsedVariant}, f.verifier)
|
2023-11-14 04:03:01 -05:00
|
|
|
if err != nil {
|
2024-06-12 10:30:03 -04:00
|
|
|
return List{}, fmt.Errorf("fetching version list: %w", err)
|
2023-11-14 04:03:01 -05:00
|
|
|
}
|
|
|
|
|
2024-06-11 10:25:24 -04:00
|
|
|
// Set the attestation variant of the list as it is not part of the marshalled JSON retrieved by Fetch
|
2024-09-05 03:59:26 -04:00
|
|
|
fetchedList.Variant = parsedVariant
|
2023-11-14 04:03:01 -05:00
|
|
|
|
|
|
|
return fetchedList, nil
|
2023-06-07 10:16:32 -04:00
|
|
|
}
|
|
|
|
|
2024-06-11 09:12:05 -04:00
|
|
|
// fetchVersion fetches the version information from the config API.
|
2024-09-05 03:59:26 -04:00
|
|
|
func (f *fetcher) fetchVersion(ctx context.Context, version string, attestationVariant fmt.Stringer) (Entry, error) {
|
|
|
|
parsedVariant, err := variant.FromString(attestationVariant.String())
|
|
|
|
if err != nil {
|
|
|
|
return Entry{}, fmt.Errorf("parsing variant: %w", err)
|
|
|
|
}
|
2024-06-12 10:30:03 -04:00
|
|
|
obj := Entry{
|
2024-06-11 10:25:24 -04:00
|
|
|
Version: version,
|
2024-09-05 03:59:26 -04:00
|
|
|
Variant: parsedVariant,
|
2024-06-11 10:25:24 -04:00
|
|
|
}
|
|
|
|
fetchedVersion, err := apifetcher.FetchAndVerify(ctx, f.HTTPClient, f.cdnURL, obj, f.verifier)
|
2023-06-07 10:16:32 -04:00
|
|
|
if err != nil {
|
2024-06-12 10:30:03 -04:00
|
|
|
return Entry{}, fmt.Errorf("fetching version %q: %w", version, err)
|
2023-06-07 10:16:32 -04:00
|
|
|
}
|
2023-11-14 04:03:01 -05:00
|
|
|
|
2024-06-11 10:25:24 -04:00
|
|
|
// Set the attestation variant of the list as it is not part of the marshalled JSON retrieved by FetchAndVerify
|
2024-09-05 03:59:26 -04:00
|
|
|
fetchedVersion.Variant = parsedVariant
|
2023-11-14 04:03:01 -05:00
|
|
|
|
2023-06-07 10:16:32 -04:00
|
|
|
return fetchedVersion, nil
|
|
|
|
}
|