pulumi/cmd/pulumi-test-language/providers/config_grpc_provider.go

343 lines
9.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"
"strings"
"github.com/blang/semver"
pschema "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/util/contract"
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
pulumirpc "github.com/pulumi/pulumi/sdk/v3/proto/go"
)
// ConfigGrpcProvider helps testing gRPC-level payloads over CheckConfig, Configure, and DiffConfig methods.
//
// In particular this provider is used to verify that all languages have parity in how the payloads that get sent on the
// wire to the provider.
//
// The schema for configuration is intentionally interesting and has the following properties:
//
// "string1", "bool1", "int1", "num1": string, boolean, integer and number-typed properties.
// "listString1", "listBool1", "listInt1", "listNum1": primitives wrapped in a list
// "mapString1", "mapBool1", "mapInt1", "mapNum1": primitives wrapped in a map
// "objString1", "objBool1", "objInt1", "objNum1": primitives wrapped in an object type {"x": t}
//
// Each property is suffixed by 1,2,3 to allow more simultaneous tests.
//
// A special resource ConfigFetcher can be invoked to retrieve a JSON-encoded representation of what the provider
// received over the wire for configuration. It exposes this as the "config" output property.
type ConfigGrpcProvider struct {
plugin.UnimplementedProvider
lastCheckConfigRequest RPCRequest
lastConfigureRequest RPCRequest
// Do not apply legacy JSON encoding to provider configuration. This flag propagates to all the affected
// languages to configure their SDK generators accordingly.
DoNotJSONEncodeProviderConfiguration bool
}
var (
_ plugin.Provider = (*ConfigGrpcProvider)(nil)
_ ProviderWithCustomServer = (*ConfigGrpcProvider)(nil)
)
func (p *ConfigGrpcProvider) Pkg() tokens.Package {
if p.DoNotJSONEncodeProviderConfiguration {
return "config-grpc-no-jsonenc"
}
return "config-grpc"
}
func (*ConfigGrpcProvider) version() string {
return "1.0.0"
}
func (p *ConfigGrpcProvider) generateSchema(
types map[string]pschema.ComplexTypeSpec,
minN int,
maxN int,
) map[string]pschema.PropertySpec {
spec := map[string]pschema.PropertySpec{}
for n := minN; n <= maxN; n++ {
prefix := "secret"
markPropertiesAsSecret := true
p.populateSchema(types, n, spec, prefix, markPropertiesAsSecret)
prefix = ""
markPropertiesAsSecret = false
p.populateSchema(types, n, spec, prefix, markPropertiesAsSecret)
}
return spec
}
func (p *ConfigGrpcProvider) populateSchema(
types map[string]pschema.ComplexTypeSpec,
n int,
spec map[string]pschema.PropertySpec,
prefix string,
markPropertiesAsSecret bool,
) {
titleCase := func(s string) string {
return strings.ToUpper(s[0:1]) + s[1:]
}
withPrefix := func(prefix, s string) string {
if prefix == "" {
return s
}
return prefix + titleCase(s)
}
for name, t := range map[string]string{
withPrefix(prefix, "string"): "string",
withPrefix(prefix, "bool"): "boolean",
withPrefix(prefix, "int"): "integer",
withPrefix(prefix, "num"): "number",
} {
ts := pschema.TypeSpec{Type: t}
c := name
if n != 0 {
c = fmt.Sprintf("%s%d", c, n)
}
spec[c] = pschema.PropertySpec{
TypeSpec: ts,
Secret: markPropertiesAsSecret,
}
spec[withPrefix("list", c)] = pschema.PropertySpec{
TypeSpec: pschema.TypeSpec{
Type: "array",
Items: &ts,
},
Secret: markPropertiesAsSecret,
}
spec[withPrefix("map", c)] = pschema.PropertySpec{
TypeSpec: pschema.TypeSpec{
Type: "object",
AdditionalProperties: &ts,
},
Secret: markPropertiesAsSecret,
}
typeToken := fmt.Sprintf("%s:index:T%s", p.Pkg(), c)
typeRef := "#/types/" + typeToken
spec[withPrefix("obj", c)] = pschema.PropertySpec{
TypeSpec: pschema.TypeSpec{
Ref: typeRef,
},
Secret: markPropertiesAsSecret,
}
types[typeToken] = pschema.ComplexTypeSpec{
ObjectTypeSpec: pschema.ObjectTypeSpec{
Type: "object",
Properties: map[string]pschema.PropertySpec{
withPrefix(prefix, "x"): {
TypeSpec: ts,
Secret: markPropertiesAsSecret,
},
},
},
}
}
}
func (p *ConfigGrpcProvider) schema() pschema.PackageSpec {
types := map[string]pschema.ComplexTypeSpec{}
configSpec := pschema.ConfigSpec{
Variables: p.generateSchema(types, 1, 3),
}
languageConfig := map[string]any{
"nodejs": map[string]any{"respectSchemaVersion": true},
}
if p.DoNotJSONEncodeProviderConfiguration {
languageConfig = map[string]any{
"nodejs": map[string]any{
"respectSchemaVersion": true,
"doNotJSONEncodeProviderConfiguration": true,
},
"python": map[string]any{
"respectSchemaVersion": true,
"doNotJSONEncodeProviderConfiguration": true,
},
"dotnet": map[string]any{
"respectSchemaVersion": true,
"doNotJSONEncodeProviderConfiguration": true,
},
}
}
marshalledLanguageConfig := map[string]pschema.RawMessage{}
for k, v := range languageConfig {
raw, err := json.Marshal(v)
contract.AssertNoErrorf(err, "json.Marshal failed unexpectedly")
marshalledLanguageConfig[k] = raw
}
schema := pschema.PackageSpec{
Name: string(p.Pkg()),
Version: p.version(),
Config: configSpec,
Provider: pschema.ResourceSpec{
InputProperties: configSpec.Variables,
ObjectTypeSpec: pschema.ObjectTypeSpec{
Properties: configSpec.Variables,
},
},
Types: types,
Resources: map[string]pschema.ResourceSpec{
fmt.Sprintf("%s:index:ConfigFetcher", p.Pkg()): {
ObjectTypeSpec: pschema.ObjectTypeSpec{
Properties: map[string]pschema.PropertySpec{
"config": {
TypeSpec: pschema.TypeSpec{Type: "string"},
},
},
Required: []string{"config"},
},
},
},
Functions: map[string]pschema.FunctionSpec{},
Language: marshalledLanguageConfig,
}
toSecretSchema := p.generateSchema(types, 1, 3)
allProps := []string{}
for k := range toSecretSchema {
allProps = append(allProps, k)
}
schema.Functions[fmt.Sprintf("%s:index:toSecret", p.Pkg())] = pschema.FunctionSpec{
Inputs: &pschema.ObjectTypeSpec{
Properties: toSecretSchema,
},
Outputs: &pschema.ObjectTypeSpec{
Properties: toSecretSchema,
Required: allProps,
},
}
return schema
}
func (p *ConfigGrpcProvider) NewProviderServer() pulumirpc.ResourceProviderServer {
return &grpcCapturingProviderServer{
ResourceProviderServer: plugin.NewProviderServer(p),
onRequest: func(r RPCRequest) {
switch r.Method {
case ConfigureMethod:
p.lastConfigureRequest = r
case CheckConfigMethod:
p.lastCheckConfigRequest = r
case DiffConfigMethod:
return
}
},
}
}
func (p *ConfigGrpcProvider) GetSchema(
context.Context, plugin.GetSchemaRequest,
) (plugin.GetSchemaResponse, error) {
schema := p.schema()
schemaBytes, err := json.Marshal(schema)
if err != nil {
return plugin.GetSchemaResponse{}, err
}
return plugin.GetSchemaResponse{Schema: schemaBytes}, nil
}
func (p *ConfigGrpcProvider) GetPluginInfo(context.Context) (workspace.PluginInfo, error) {
ver := semver.MustParse(p.version())
return workspace.PluginInfo{
Version: &ver,
}, nil
}
func (p *ConfigGrpcProvider) Check(
_ context.Context, req plugin.CheckRequest,
) (plugin.CheckResponse, error) {
return plugin.CheckResponse{Properties: req.News}, nil
}
func (p *ConfigGrpcProvider) CheckConfig(
_ context.Context, req plugin.CheckConfigRequest,
) (plugin.CheckConfigResponse, error) {
return plugin.CheckConfigResponse{Properties: req.News}, nil
}
func (p *ConfigGrpcProvider) Configure(
context.Context, plugin.ConfigureRequest,
) (plugin.ConfigureResponse, error) {
return plugin.ConfigureResponse{}, nil
}
func (p *ConfigGrpcProvider) Create(
ctx context.Context,
req plugin.CreateRequest,
) (plugin.CreateResponse, error) {
if string(req.Type) == fmt.Sprintf("%s:index:ConfigFetcher", p.Pkg()) {
requestsJSON, err := json.Marshal([]RPCRequest{
p.lastCheckConfigRequest,
p.lastConfigureRequest,
})
contract.AssertNoErrorf(err, "json.Marshal failed")
id := ""
if !req.Preview {
id = "id0"
}
// Send out Config-related requests.
return plugin.CreateResponse{
ID: resource.ID(id),
Properties: resource.PropertyMap{
"config": resource.NewStringProperty(string(requestsJSON)),
},
Status: resource.StatusOK,
}, nil
}
return plugin.CreateResponse{}, errors.New("Unknown resource")
}
func (p *ConfigGrpcProvider) DiffConfig(
context.Context, plugin.DiffConfigRequest,
) (plugin.DiffConfigResponse, error) {
return plugin.DiffResult{}, nil
}
func (p *ConfigGrpcProvider) Invoke(
ctx context.Context, req plugin.InvokeRequest,
) (plugin.InvokeResponse, error) {
switch {
case string(req.Tok) == fmt.Sprintf("%s:index:toSecret", p.Pkg()):
secreted := req.Args.Copy()
for k, v := range secreted {
secreted[k] = resource.MakeSecret(v)
}
return plugin.InvokeResponse{Properties: secreted}, nil
default:
return plugin.InvokeResponse{}, errors.New("Unknown function")
}
}