Files
opentf/internal/builtin/providers/tf/provider.go
2024-12-25 13:21:59 +04:00

223 lines
7.6 KiB
Go

// Copyright (c) The OpenTofu Authors
// SPDX-License-Identifier: MPL-2.0
// Copyright (c) 2023 HashiCorp, Inc.
// SPDX-License-Identifier: MPL-2.0
package tf
import (
"fmt"
"log"
"strings"
"github.com/opentofu/opentofu/internal/addrs"
"github.com/opentofu/opentofu/internal/encryption"
"github.com/opentofu/opentofu/internal/providers"
"github.com/zclconf/go-cty/cty"
)
// Provider is an implementation of providers.Interface
type Provider struct {
funcs map[string]providerFunc
}
// NewProvider returns a new tofu provider
func NewProvider() providers.Interface {
return &Provider{
funcs: getProviderFuncs(),
}
}
func (p *Provider) getFunctionSpecs() map[string]providers.FunctionSpec {
funcSpecs := make(map[string]providers.FunctionSpec)
for name, fn := range p.funcs {
funcSpecs[name] = fn.GetFunctionSpec()
}
return funcSpecs
}
// GetSchema returns the complete schema for the provider.
func (p *Provider) GetProviderSchema() providers.GetProviderSchemaResponse {
return providers.GetProviderSchemaResponse{
DataSources: map[string]providers.Schema{
"terraform_remote_state": dataSourceRemoteStateGetSchema(),
},
ResourceTypes: map[string]providers.Schema{
"terraform_data": dataStoreResourceSchema(),
},
Functions: p.getFunctionSpecs(),
}
}
// ValidateProviderConfig is used to validate the configuration values.
func (p *Provider) ValidateProviderConfig(req providers.ValidateProviderConfigRequest) providers.ValidateProviderConfigResponse {
// At this moment there is nothing to configure for the tofu provider,
// so we will happily return without taking any action
var res providers.ValidateProviderConfigResponse
res.PreparedConfig = req.Config
return res
}
// ValidateDataResourceConfig is used to validate the data source configuration values.
func (p *Provider) ValidateDataResourceConfig(req providers.ValidateDataResourceConfigRequest) providers.ValidateDataResourceConfigResponse {
// FIXME: move the backend configuration validate call that's currently
// inside the read method into here so that we can catch provider configuration
// errors in tofu validate as well as during tofu plan.
var res providers.ValidateDataResourceConfigResponse
// This should not happen
if req.TypeName != "terraform_remote_state" {
res.Diagnostics.Append(fmt.Errorf("Error: unsupported data source %s", req.TypeName))
return res
}
diags := dataSourceRemoteStateValidate(req.Config)
res.Diagnostics = diags
return res
}
// Configure configures and initializes the provider.
func (p *Provider) ConfigureProvider(providers.ConfigureProviderRequest) providers.ConfigureProviderResponse {
// At this moment there is nothing to configure for the terraform provider,
// so we will happily return without taking any action
var res providers.ConfigureProviderResponse
return res
}
// ReadDataSource returns the data source's current state.
func (p *Provider) ReadDataSource(req providers.ReadDataSourceRequest) providers.ReadDataSourceResponse {
panic("Should not be called directly, special case for terraform_remote_state")
}
func (p *Provider) ReadDataSourceEncrypted(req providers.ReadDataSourceRequest, path addrs.AbsResourceInstance, enc encryption.Encryption) providers.ReadDataSourceResponse {
// call function
var res providers.ReadDataSourceResponse
// This should not happen
if req.TypeName != "terraform_remote_state" {
res.Diagnostics.Append(fmt.Errorf("Error: unsupported data source %s", req.TypeName))
return res
}
// These string manipulations are kind of funky
key := path.String()
// data.terraform_remote_state.foo[4] -> foo[4]
// module.submod[1].data.terraform_remote_state.bar -> module.submod[1].bar
key = strings.Replace(key, "data.terraform_remote_state.", "", 1)
// module.submod[1].bar -> submod[1].bar
key = strings.TrimPrefix(key, "module.")
log.Printf("[DEBUG] accessing remote state at %s", key)
newState, diags := dataSourceRemoteStateRead(req.Config, enc.RemoteState(key))
if diags.HasErrors() {
diags = diags.Append(fmt.Errorf("%s: Unable to read remote state", path.String()))
}
res.State = newState
res.Diagnostics = diags
return res
}
// Stop is called when the provider should halt any in-flight actions.
func (p *Provider) Stop() error {
log.Println("[DEBUG] terraform provider cannot Stop")
return nil
}
// All the Resource-specific functions are below.
// The terraform provider supplies a single data source, `terraform_remote_state`
// and no resources.
// UpgradeResourceState is called when the state loader encounters an
// instance state whose schema version is less than the one reported by the
// currently-used version of the corresponding provider, and the upgraded
// result is used for any further processing.
func (p *Provider) UpgradeResourceState(req providers.UpgradeResourceStateRequest) providers.UpgradeResourceStateResponse {
return upgradeDataStoreResourceState(req)
}
// ReadResource refreshes a resource and returns its current state.
func (p *Provider) ReadResource(req providers.ReadResourceRequest) providers.ReadResourceResponse {
return readDataStoreResourceState(req)
}
// PlanResourceChange takes the current state and proposed state of a
// resource, and returns the planned final state.
func (p *Provider) PlanResourceChange(req providers.PlanResourceChangeRequest) providers.PlanResourceChangeResponse {
return planDataStoreResourceChange(req)
}
// ApplyResourceChange takes the planned state for a resource, which may
// yet contain unknown computed values, and applies the changes returning
// the final state.
func (p *Provider) ApplyResourceChange(req providers.ApplyResourceChangeRequest) providers.ApplyResourceChangeResponse {
return applyDataStoreResourceChange(req)
}
// ImportResourceState requests that the given resource be imported.
func (p *Provider) ImportResourceState(req providers.ImportResourceStateRequest) providers.ImportResourceStateResponse {
if req.TypeName == "terraform_data" {
return importDataStore(req)
}
panic("unimplemented - terraform_remote_state has no resources")
}
// ValidateResourceConfig is used to to validate the resource configuration values.
func (p *Provider) ValidateResourceConfig(req providers.ValidateResourceConfigRequest) providers.ValidateResourceConfigResponse {
return validateDataStoreResourceConfig(req)
}
func (p *Provider) GetFunctions() providers.GetFunctionsResponse {
return providers.GetFunctionsResponse{
Functions: p.getFunctionSpecs(),
}
}
func (p *Provider) CallFunction(r providers.CallFunctionRequest) providers.CallFunctionResponse {
fn, ok := p.funcs[r.Name]
if !ok {
return providers.CallFunctionResponse{
Error: fmt.Errorf("provider function %q not found", r.Name),
}
}
v, err := fn.Call(r.Arguments)
return providers.CallFunctionResponse{
Result: v,
Error: err,
}
}
// Close is a noop for this provider, since it's run in-process.
func (p *Provider) Close() error {
return nil
}
// providerFunc is an interface representing a built-in provider function
type providerFunc interface {
// Name returns the name of the function which is used to call it
Name() string
// GetFunctionSpec returns the provider function specification
GetFunctionSpec() providers.FunctionSpec
// Call is used to invoke the function
Call(args []cty.Value) (cty.Value, error)
}
// getProviderFuncs returns a map of functions that are registered in the provider
func getProviderFuncs() map[string]providerFunc {
decodeTFVars := &decodeTFVarsFunc{}
encodeTFVars := &encodeTFVarsFunc{}
encodeExpr := &encodeExprFunc{}
return map[string]providerFunc{
decodeTFVars.Name(): decodeTFVars,
encodeTFVars.Name(): encodeTFVars,
encodeExpr.Name(): encodeExpr,
}
}