pulumi/sdk/go/common/resource/plugin/converter_plugin.go

176 lines
5.3 KiB
Go

// Copyright 2016-2023, 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 plugin
import (
"context"
"fmt"
"os"
"github.com/blang/semver"
"github.com/grpc-ecosystem/grpc-opentracing/go/otgrpc"
"github.com/hashicorp/hcl/v2"
"google.golang.org/grpc"
"google.golang.org/grpc/credentials/insecure"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/logging"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/rpcutil"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/rpcutil/rpcerror"
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
pulumirpc "github.com/pulumi/pulumi/sdk/v3/proto/go"
)
// converter reflects a converter plugin, loaded dynamically from another process over gRPC.
type converter struct {
name string
plug *plugin // the actual plugin process wrapper.
clientRaw pulumirpc.ConverterClient // the raw provider client; usually unsafe to use directly.
}
func NewConverter(ctx *Context, name string, version *semver.Version) (Converter, error) {
prefix := fmt.Sprintf("%v (converter)", name)
// Load the plugin's path by using the standard workspace logic.
path, err := workspace.GetPluginPath(ctx.Diag, workspace.ConverterPlugin, name, version, ctx.Host.GetProjectPlugins())
if err != nil {
return nil, err
}
contract.Assertf(path != "", "unexpected empty path for plugin %s", name)
plug, err := newPlugin(ctx, ctx.Pwd, path, prefix,
workspace.ConverterPlugin, []string{}, os.Environ(), converterPluginDialOptions(ctx, name, ""))
if err != nil {
return nil, err
}
contract.Assertf(plug != nil, "unexpected nil converter plugin for %s", name)
c := &converter{
name: name,
plug: plug,
clientRaw: pulumirpc.NewConverterClient(plug.Conn),
}
return c, nil
}
func converterPluginDialOptions(ctx *Context, name string, path string) []grpc.DialOption {
dialOpts := append(
rpcutil.OpenTracingInterceptorDialOptions(otgrpc.SpanDecorator(decorateProviderSpans)),
grpc.WithTransportCredentials(insecure.NewCredentials()),
rpcutil.GrpcChannelOptions(),
)
if ctx.DialOptions != nil {
metadata := map[string]interface{}{
"mode": "client",
"kind": "converter",
}
if name != "" {
metadata["name"] = name
}
if path != "" {
metadata["path"] = path
}
dialOpts = append(dialOpts, ctx.DialOptions(metadata)...)
}
return dialOpts
}
// label returns a base label for tracing functions.
func (c *converter) label() string {
return fmt.Sprintf("Converter[%s, %p]", c.name, c)
}
func (c *converter) Close() error {
if c.plug == nil {
return nil
}
return c.plug.Close()
}
func (c *converter) ConvertState(ctx context.Context, req *ConvertStateRequest) (*ConvertStateResponse, error) {
label := c.label() + ".ConvertState"
logging.V(7).Infof("%s executing", label)
resp, err := c.clientRaw.ConvertState(ctx, &pulumirpc.ConvertStateRequest{
MapperTarget: req.MapperTarget,
Args: req.Args,
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(8).Infof("%s converter received rpc error `%s`: `%s`", label, rpcError.Code(), rpcError.Message())
return nil, err
}
resources := make([]ResourceImport, len(resp.Resources))
for i, resource := range resp.Resources {
resources[i] = ResourceImport{
Type: resource.Type,
Name: resource.Name,
ID: resource.Id,
Version: resource.Version,
PluginDownloadURL: resource.PluginDownloadURL,
LogicalName: resource.LogicalName,
IsRemote: resource.IsRemote,
IsComponent: resource.IsComponent,
}
}
// Translate the rpc diagnostics into hcl.Diagnostics.
var diags hcl.Diagnostics
for _, rpcDiag := range resp.Diagnostics {
diags = append(diags, RPCDiagnosticToHclDiagnostic(rpcDiag))
}
logging.V(7).Infof("%s success", label)
return &ConvertStateResponse{
Resources: resources,
Diagnostics: diags,
}, nil
}
func (c *converter) ConvertProgram(ctx context.Context, req *ConvertProgramRequest) (*ConvertProgramResponse, error) {
label := c.label() + ".ConvertProgram"
logging.V(7).Infof("%s executing", label)
resp, err := c.clientRaw.ConvertProgram(ctx, &pulumirpc.ConvertProgramRequest{
SourceDirectory: req.SourceDirectory,
TargetDirectory: req.TargetDirectory,
MapperTarget: req.MapperTarget,
LoaderTarget: req.LoaderTarget,
Args: req.Args,
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(8).Infof("%s converter received rpc error `%s`: `%s`", label, rpcError.Code(), rpcError.Message())
return nil, err
}
// Translate the rpc diagnostics into hcl.Diagnostics.
var diags hcl.Diagnostics
for _, rpcDiag := range resp.Diagnostics {
diags = append(diags, RPCDiagnosticToHclDiagnostic(rpcDiag))
}
logging.V(7).Infof("%s success", label)
return &ConvertProgramResponse{
Diagnostics: diags,
}, nil
}