2023-11-27 09:00:08 +01:00
/ *
Copyright ( c ) Edgeless Systems GmbH
SPDX - License - Identifier : AGPL - 3.0 - only
* /
package provider
import (
"context"
2023-12-20 15:56:48 +01:00
"errors"
2023-11-27 09:00:08 +01:00
"fmt"
2023-12-11 15:55:44 +01:00
"regexp"
2023-12-18 10:15:54 +01:00
"strings"
2023-11-27 09:00:08 +01:00
2023-12-18 10:15:54 +01:00
"github.com/edgelesssys/constellation/v2/internal/api/versionsapi"
2023-11-27 09:00:08 +01:00
"github.com/edgelesssys/constellation/v2/internal/attestation/variant"
"github.com/edgelesssys/constellation/v2/internal/cloud/cloudprovider"
"github.com/edgelesssys/constellation/v2/internal/imagefetcher"
2023-12-20 15:56:48 +01:00
"github.com/edgelesssys/constellation/v2/internal/semver"
2023-12-14 15:47:55 +01:00
"github.com/edgelesssys/constellation/v2/terraform-provider-constellation/internal/data"
2023-11-27 09:00:08 +01:00
"github.com/hashicorp/terraform-plugin-framework/datasource"
"github.com/hashicorp/terraform-plugin-framework/datasource/schema"
"github.com/hashicorp/terraform-plugin-framework/path"
"github.com/hashicorp/terraform-plugin-framework/types"
2023-12-14 15:47:55 +01:00
"github.com/hashicorp/terraform-plugin-log/tflog"
2023-11-27 09:00:08 +01:00
)
2023-12-11 15:55:44 +01:00
var (
// Ensure provider defined types fully satisfy framework interfaces.
2023-12-20 15:56:48 +01:00
_ datasource . DataSource = & ImageDataSource { }
_ datasource . DataSourceWithValidateConfig = & ImageDataSource { }
_ datasource . DataSourceWithConfigure = & ImageDataSource { }
caseInsensitiveCommunityGalleriesRegexp = regexp . MustCompile ( ` (?i)\/communitygalleries\/ ` )
caseInsensitiveImagesRegExp = regexp . MustCompile ( ` (?i)\/images\/ ` )
caseInsensitiveVersionsRegExp = regexp . MustCompile ( ` (?i)\/versions\/ ` )
2023-12-11 15:55:44 +01:00
)
2023-11-27 09:00:08 +01:00
// NewImageDataSource creates a new data source for fetching Constellation OS images
// from the Versions-API.
func NewImageDataSource ( ) datasource . DataSource {
return & ImageDataSource { }
}
// ImageDataSource defines the data source implementation for the image data source.
// It is used to retrieve the Constellation OS image reference for a given CSP and Attestation Variant.
type ImageDataSource struct {
imageFetcher imageFetcher
2023-12-14 15:47:55 +01:00
version string
2023-11-27 09:00:08 +01:00
}
// imageFetcher gets an image reference from the versionsapi.
type imageFetcher interface {
FetchReference ( ctx context . Context ,
provider cloudprovider . Provider , attestationVariant variant . Variant ,
2023-12-08 14:40:31 +01:00
image , region string , useMarketplaceImage bool ,
2023-11-27 09:00:08 +01:00
) ( string , error )
}
// ImageDataSourceModel defines the image data source's data model.
type ImageDataSourceModel struct {
AttestationVariant types . String ` tfsdk:"attestation_variant" `
2023-12-18 10:15:54 +01:00
Version types . String ` tfsdk:"version" `
2023-11-27 09:00:08 +01:00
CSP types . String ` tfsdk:"csp" `
2023-12-08 14:40:31 +01:00
MarketplaceImage types . Bool ` tfsdk:"marketplace_image" `
2023-11-27 09:00:08 +01:00
Region types . String ` tfsdk:"region" `
2023-12-18 10:15:54 +01:00
Image types . Object ` tfsdk:"image" `
2023-11-27 09:00:08 +01:00
}
// Metadata returns the metadata for the image data source.
func ( d * ImageDataSource ) Metadata ( _ context . Context , req datasource . MetadataRequest , resp * datasource . MetadataResponse ) {
resp . TypeName = req . ProviderTypeName + "_image"
}
// Schema returns the schema for the image data source.
func ( d * ImageDataSource ) Schema ( _ context . Context , _ datasource . SchemaRequest , resp * datasource . SchemaResponse ) {
resp . Schema = schema . Schema {
2023-12-14 15:47:55 +01:00
Description : "The data source to resolve the CSP-specific OS image reference for a given version and attestation variant." ,
MarkdownDescription : "Data source to resolve the CSP-specific OS image reference for a given version and attestation variant." ,
2023-11-27 09:00:08 +01:00
Attributes : map [ string ] schema . Attribute {
2023-12-18 10:15:54 +01:00
// Input Attributes
"attestation_variant" : newAttestationVariantAttributeSchema ( attributeInput ) ,
"version" : schema . StringAttribute {
2023-12-14 15:47:55 +01:00
Description : "Version of the Constellation OS image to use. (e.g. `v2.13.0`). If not set, the provider version is used." ,
MarkdownDescription : "Version of the Constellation OS image to use. (e.g. `v2.13.0`). If not set, the provider version value is used." ,
Optional : true ,
2023-11-27 09:00:08 +01:00
} ,
2023-12-18 10:15:54 +01:00
"csp" : newCSPAttributeSchema ( ) ,
2023-12-08 14:40:31 +01:00
"marketplace_image" : schema . BoolAttribute {
2024-02-06 12:13:59 +01:00
Description : "Whether a marketplace image should be used." ,
MarkdownDescription : "Whether a marketplace image should be used." ,
2023-12-08 14:40:31 +01:00
Optional : true ,
} ,
2023-11-27 09:00:08 +01:00
"region" : schema . StringAttribute {
Description : "Region to retrieve the image for. Only required for AWS." ,
MarkdownDescription : "Region to retrieve the image for. Only required for AWS.\n" +
"The Constellation OS image must be [replicated to the region](https://docs.edgeless.systems/constellation/workflows/config)," +
"and the region must [support AMD SEV-SNP](https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/snp-requirements.html), if it is used for Attestation." ,
Optional : true ,
} ,
2023-12-18 10:15:54 +01:00
// Output Attributes
"image" : newImageAttributeSchema ( attributeOutput ) ,
2023-11-27 09:00:08 +01:00
} ,
}
}
2023-12-14 15:47:55 +01:00
// ValidateConfig validates the configuration for the image data source.
func ( d * ImageDataSource ) ValidateConfig ( ctx context . Context , req datasource . ValidateConfigRequest , resp * datasource . ValidateConfigResponse ) {
var data ImageDataSourceModel
resp . Diagnostics . Append ( req . Config . Get ( ctx , & data ) ... )
if resp . Diagnostics . HasError ( ) {
return
}
2023-12-20 15:56:48 +01:00
// Region must be set for AWS
2023-12-14 15:47:55 +01:00
if data . CSP . Equal ( types . StringValue ( "aws" ) ) && data . Region . IsNull ( ) {
resp . Diagnostics . AddAttributeError (
path . Root ( "region" ) ,
"Region must be set for AWS" , "When csp is set to 'aws', 'region' must be specified." ,
)
2023-12-20 15:56:48 +01:00
}
// Setting Region for non-AWS CSPs has no effect
if ! data . CSP . Equal ( types . StringValue ( "aws" ) ) && ! data . Region . IsNull ( ) {
resp . Diagnostics . AddAttributeWarning (
path . Root ( "region" ) ,
"Region should only be set for AWS" , "When another CSP than AWS is used, setting 'region' has no effect." ,
)
}
// Version should be a valid semver or short path, if set
if ! data . Version . IsNull ( ) {
_ , semverErr := semver . New ( data . Version . ValueString ( ) )
_ , shortpathErr := versionsapi . NewVersionFromShortPath ( data . Version . ValueString ( ) , versionsapi . VersionKindImage )
if semverErr != nil && shortpathErr != nil {
resp . Diagnostics . AddAttributeError (
path . Root ( "version" ) ,
"Invalid Version" ,
fmt . Sprintf ( "When parsing the version (%s), an error occurred: %s" , data . Version . ValueString ( ) , errors . Join ( semverErr , shortpathErr ) ) ,
)
}
2023-12-14 15:47:55 +01:00
}
}
2023-11-27 09:00:08 +01:00
// Configure configures the data source.
2023-12-14 15:47:55 +01:00
func ( d * ImageDataSource ) Configure ( _ context . Context , req datasource . ConfigureRequest , resp * datasource . ConfigureResponse ) {
2023-11-27 09:00:08 +01:00
d . imageFetcher = imagefetcher . New ( )
2023-12-14 15:47:55 +01:00
// Prevent panic if the provider has not been configured. is necessary!
if req . ProviderData == nil {
return
}
providerData , ok := req . ProviderData . ( data . ProviderData )
if ! ok {
resp . Diagnostics . AddError (
"Unexpected Data Source Configure Type" ,
fmt . Sprintf ( "Expected data.ProviderData, got: %T. Please report this issue to the provider developers." , req . ProviderData ) ,
)
return
}
2023-12-18 14:21:19 +01:00
d . version = providerData . Version . String ( )
2023-11-27 09:00:08 +01:00
}
// Read reads from the data source.
func ( d * ImageDataSource ) Read ( ctx context . Context , req datasource . ReadRequest , resp * datasource . ReadResponse ) {
// Retrieve the configuration values for this data source instance.
var data ImageDataSourceModel
resp . Diagnostics . Append ( req . Config . Get ( ctx , & data ) ... )
// Check configuration for errors.
csp := cloudprovider . FromString ( data . CSP . ValueString ( ) )
if csp == cloudprovider . Unknown {
resp . Diagnostics . AddAttributeError (
path . Root ( "csp" ) ,
"Invalid CSP" ,
fmt . Sprintf ( "Invalid CSP: %s" , data . CSP . ValueString ( ) ) ,
)
}
attestationVariant , err := variant . FromString ( data . AttestationVariant . ValueString ( ) )
if err != nil {
resp . Diagnostics . AddAttributeError (
path . Root ( "attestation_variant" ) ,
"Invalid Attestation Variant" ,
fmt . Sprintf ( "When parsing the Attestation Variant (%s), an error occurred: %s" , data . AttestationVariant . ValueString ( ) , err ) ,
)
}
if resp . Diagnostics . HasError ( ) {
return
}
2023-12-18 10:15:54 +01:00
// lock-step with the provider
imageVersion := data . Version . ValueString ( )
2023-12-14 15:47:55 +01:00
if imageVersion == "" {
tflog . Info ( ctx , fmt . Sprintf ( "No image version specified, using provider version %s" , d . version ) )
imageVersion = d . version // Use provider version as default.
}
2023-12-18 10:15:54 +01:00
// determine semver from version string
var imageSemver string
var apiCompatibleVer versionsapi . Version
if strings . HasPrefix ( imageVersion , "v" ) {
// If the version is a release version, it should look like vX.Y.Z
imageSemver = imageVersion
apiCompatibleVer , err = versionsapi . NewVersion (
versionsapi . ReleaseRef ,
"stable" ,
imageVersion ,
versionsapi . VersionKindImage ,
)
if err != nil {
resp . Diagnostics . AddAttributeError (
path . Root ( "version" ) ,
"Invalid Version" ,
fmt . Sprintf ( "When parsing the version (%s), an error occurred: %s" , imageVersion , err ) ,
)
return
}
} else {
// otherwise, it should be a versionsapi short path
apiCompatibleVer , err = versionsapi . NewVersionFromShortPath ( imageVersion , versionsapi . VersionKindImage )
if err != nil {
resp . Diagnostics . AddAttributeError (
path . Root ( "version" ) ,
"Invalid Version" ,
fmt . Sprintf ( "When parsing the version (%s), an error occurred: %s" , imageVersion , err ) ,
)
return
}
imageSemver = apiCompatibleVer . Version ( )
}
2023-11-27 09:00:08 +01:00
// Retrieve Image Reference
2023-12-08 14:40:31 +01:00
imageRef , err := d . imageFetcher . FetchReference ( ctx , csp , attestationVariant ,
2023-12-14 15:47:55 +01:00
imageVersion , data . Region . ValueString ( ) , data . MarketplaceImage . ValueBool ( ) )
2023-11-27 09:00:08 +01:00
if err != nil {
resp . Diagnostics . AddError (
"Error fetching Image Reference" ,
fmt . Sprintf ( "When fetching the image reference, an error occurred: %s" , err ) ,
)
return
}
2023-12-11 15:55:44 +01:00
// Do adjustments for Azure casing
if csp == cloudprovider . Azure {
imageRef = caseInsensitiveCommunityGalleriesRegexp . ReplaceAllString ( imageRef , "/communityGalleries/" )
imageRef = caseInsensitiveImagesRegExp . ReplaceAllString ( imageRef , "/images/" )
imageRef = caseInsensitiveVersionsRegExp . ReplaceAllString ( imageRef , "/versions/" )
}
2023-11-27 09:00:08 +01:00
// Save data into Terraform state
2023-12-18 10:15:54 +01:00
diags := resp . State . SetAttribute ( ctx , path . Root ( "image" ) , imageAttribute {
2024-03-06 20:48:40 +01:00
Reference : imageRef ,
Version : imageSemver ,
ShortPath : apiCompatibleVer . ShortPath ( ) ,
MarketplaceImage : data . MarketplaceImage . ValueBoolPointer ( ) ,
2023-12-18 10:15:54 +01:00
} )
resp . Diagnostics . Append ( diags ... )
if resp . Diagnostics . HasError ( ) {
return
}
2023-11-27 09:00:08 +01:00
}