mirror of https://github.com/pulumi/pulumi.git
276 lines
7.6 KiB
Go
276 lines
7.6 KiB
Go
// Copyright 2016-2024, Pulumi Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package providers
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/blang/semver"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/schema"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/plugin"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
|
|
)
|
|
|
|
type ParameterizedProvider struct {
|
|
plugin.UnimplementedProvider
|
|
parameterValue []byte
|
|
parameterVersion string
|
|
parameterPackage string
|
|
config string
|
|
}
|
|
|
|
var _ plugin.Provider = (*ParameterizedProvider)(nil)
|
|
|
|
func (p *ParameterizedProvider) Close() error {
|
|
return nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) Configure(
|
|
context.Context, plugin.ConfigureRequest,
|
|
) (plugin.ConfigureResponse, error) {
|
|
return plugin.ConfigureResponse{}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) Pkg() tokens.Package {
|
|
return "parameterized"
|
|
}
|
|
|
|
func (p *ParameterizedProvider) GetSchema(
|
|
_ context.Context, req plugin.GetSchemaRequest,
|
|
) (plugin.GetSchemaResponse, error) {
|
|
if p.parameterPackage == "" {
|
|
return plugin.GetSchemaResponse{}, errors.New("expected a parameter package name for a parameterized provider")
|
|
}
|
|
|
|
if p.parameterVersion == "" {
|
|
return plugin.GetSchemaResponse{}, errors.New("expected a parameter version for a parameterized provider")
|
|
}
|
|
|
|
subpackage := p.parameterPackage
|
|
version := p.parameterVersion
|
|
|
|
// the name of the resource is the parameterized value
|
|
parameterizedResource := string(p.parameterValue)
|
|
if parameterizedResource == "" {
|
|
return plugin.GetSchemaResponse{}, errors.New("expected parameter value to be non-empty")
|
|
}
|
|
|
|
token := fmt.Sprintf("%s:index:%s", subpackage, parameterizedResource)
|
|
|
|
pkg := schema.PackageSpec{
|
|
Name: subpackage,
|
|
Version: version,
|
|
Provider: schema.ResourceSpec{
|
|
InputProperties: map[string]schema.PropertySpec{
|
|
"text": {
|
|
TypeSpec: schema.TypeSpec{
|
|
Type: "string",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
Resources: map[string]schema.ResourceSpec{
|
|
token: {
|
|
ObjectTypeSpec: schema.ObjectTypeSpec{
|
|
Type: "object",
|
|
Properties: map[string]schema.PropertySpec{
|
|
"parameterValue": {
|
|
TypeSpec: schema.TypeSpec{
|
|
Type: "string",
|
|
},
|
|
},
|
|
},
|
|
Required: []string{"parameterValue"},
|
|
},
|
|
},
|
|
},
|
|
Parameterization: &schema.ParameterizationSpec{
|
|
BaseProvider: schema.BaseProviderSpec{
|
|
Name: "parameterized",
|
|
Version: "1.2.3",
|
|
},
|
|
Parameter: p.parameterValue,
|
|
},
|
|
}
|
|
|
|
jsonBytes, err := json.Marshal(pkg)
|
|
return plugin.GetSchemaResponse{Schema: jsonBytes}, err
|
|
}
|
|
|
|
func (p *ParameterizedProvider) Parameterize(
|
|
_ context.Context, req plugin.ParameterizeRequest,
|
|
) (plugin.ParameterizeResponse, error) {
|
|
param, ok := req.Parameters.(*plugin.ParameterizeValue)
|
|
if !ok {
|
|
err := fmt.Errorf("expected a ParameterizeValue when parameterizing, instead got %T", req.Parameters)
|
|
return plugin.ParameterizeResponse{}, err
|
|
}
|
|
|
|
if param.Value == nil {
|
|
return plugin.ParameterizeResponse{}, errors.New("expected a non-nil value when parameterizing")
|
|
}
|
|
|
|
if param.Name == "" {
|
|
return plugin.ParameterizeResponse{}, errors.New("expected a non-empty name when parameterizing")
|
|
}
|
|
|
|
p.parameterPackage = param.Name
|
|
p.parameterVersion = param.Version.String()
|
|
p.parameterValue = param.Value
|
|
|
|
return plugin.ParameterizeResponse{
|
|
Name: param.Name,
|
|
Version: param.Version,
|
|
}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) CheckConfig(
|
|
_ context.Context, req plugin.CheckConfigRequest,
|
|
) (plugin.CheckConfigResponse, error) {
|
|
// Expect the version
|
|
version, ok := req.News["version"]
|
|
if !ok {
|
|
return plugin.CheckConfigResponse{
|
|
Failures: makeCheckFailure("version", "missing version"),
|
|
}, nil
|
|
}
|
|
if !version.IsString() {
|
|
return plugin.CheckConfigResponse{
|
|
Failures: makeCheckFailure("version", "version is not a string"),
|
|
}, nil
|
|
}
|
|
|
|
// assert that the version is the parameterized version
|
|
if version.StringValue() != p.parameterVersion {
|
|
return plugin.CheckConfigResponse{
|
|
Failures: makeCheckFailure("version", "version in CheckConfig is not the parameterized version"),
|
|
}, nil
|
|
}
|
|
|
|
// Optionally expect the text config
|
|
text, ok := req.News["text"]
|
|
if ok {
|
|
if !text.IsString() {
|
|
return plugin.CheckConfigResponse{
|
|
Failures: makeCheckFailure("text", "text is not a string"),
|
|
}, nil
|
|
}
|
|
p.config = text.StringValue()
|
|
}
|
|
|
|
if len(req.News) > 2 {
|
|
return plugin.CheckConfigResponse{
|
|
Failures: makeCheckFailure("", fmt.Sprintf("too many properties: %v", req.News)),
|
|
}, nil
|
|
}
|
|
|
|
return plugin.CheckConfigResponse{Properties: req.News}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) Check(
|
|
_ context.Context, req plugin.CheckRequest,
|
|
) (plugin.CheckResponse, error) {
|
|
// URN should be of the form "{sub-package}:index:{parameterized-resource}"
|
|
expectedToken := fmt.Sprintf("%s:index:%s", p.parameterPackage, string(p.parameterValue))
|
|
if string(req.URN.Type()) != expectedToken {
|
|
return plugin.CheckResponse{
|
|
Failures: makeCheckFailure("",
|
|
fmt.Sprintf("invalid URN type: %s. Expected %s", req.URN.Type(), expectedToken)),
|
|
}, nil
|
|
}
|
|
|
|
return plugin.CheckResponse{Properties: req.News}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) Create(
|
|
_ context.Context, req plugin.CreateRequest,
|
|
) (plugin.CreateResponse, error) {
|
|
// URN should be of the form "{sub-package}:index:{parameterized-resource}"
|
|
expectedToken := fmt.Sprintf("%s:index:%s", p.parameterPackage, string(p.parameterValue))
|
|
if string(req.URN.Type()) != expectedToken {
|
|
return plugin.CreateResponse{
|
|
Status: resource.StatusUnknown,
|
|
}, fmt.Errorf("invalid URN type: %s. Expected %s", req.URN.Type(), expectedToken)
|
|
}
|
|
|
|
id := "id"
|
|
if req.Preview {
|
|
id = ""
|
|
}
|
|
|
|
var configText string
|
|
if p.config != "" {
|
|
configText = " " + p.config
|
|
}
|
|
|
|
// parameterized resource outputs will include the parameter value that the provider was parameterized with
|
|
outputs := resource.NewPropertyMapFromMap(map[string]interface{}{
|
|
"parameterValue": string(p.parameterValue) + configText,
|
|
})
|
|
|
|
return plugin.CreateResponse{
|
|
ID: resource.ID(id),
|
|
Properties: outputs,
|
|
Status: resource.StatusOK,
|
|
}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) GetPluginInfo(context.Context) (workspace.PluginInfo, error) {
|
|
ver := semver.MustParse("1.2.3")
|
|
return workspace.PluginInfo{
|
|
Version: &ver,
|
|
}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) SignalCancellation(context.Context) error {
|
|
return nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) GetMapping(
|
|
context.Context, plugin.GetMappingRequest,
|
|
) (plugin.GetMappingResponse, error) {
|
|
return plugin.GetMappingResponse{}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) GetMappings(
|
|
context.Context, plugin.GetMappingsRequest,
|
|
) (plugin.GetMappingsResponse, error) {
|
|
return plugin.GetMappingsResponse{}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) DiffConfig(
|
|
context.Context, plugin.DiffConfigRequest,
|
|
) (plugin.DiffConfigResponse, error) {
|
|
return plugin.DiffResult{}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) Diff(
|
|
context.Context, plugin.DiffRequest,
|
|
) (plugin.DiffResponse, error) {
|
|
return plugin.DiffResult{}, nil
|
|
}
|
|
|
|
func (p *ParameterizedProvider) Delete(
|
|
context.Context, plugin.DeleteRequest,
|
|
) (plugin.DeleteResponse, error) {
|
|
return plugin.DeleteResponse{}, nil
|
|
}
|