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

601 lines
19 KiB
Go

// Copyright 2016-2018, 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"
"io"
"os"
"path/filepath"
"strings"
"github.com/blang/semver"
"github.com/hashicorp/hcl/v2"
"github.com/pkg/errors"
"google.golang.org/grpc"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/credentials/insecure"
"google.golang.org/protobuf/types/known/emptypb"
"google.golang.org/protobuf/types/known/structpb"
"github.com/pulumi/pulumi/sdk/v3/go/common/diag/colors"
"github.com/pulumi/pulumi/sdk/v3/go/common/slice"
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/cmdutil"
"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"
)
// langhost reflects a language host plugin, loaded dynamically for a single language/runtime pair.
type langhost struct {
ctx *Context
runtime string
plug *plugin
client pulumirpc.LanguageRuntimeClient
root string
options map[string]interface{}
}
// NewLanguageRuntime binds to a language's runtime plugin and then creates a gRPC connection to it. If the
// plugin could not be found, or an error occurs while creating the child process, an error is returned.
func NewLanguageRuntime(host Host, ctx *Context, root, pwd, runtime string,
options map[string]interface{},
) (LanguageRuntime, error) {
path, err := workspace.GetPluginPath(ctx.Diag,
workspace.LanguagePlugin, strings.ReplaceAll(runtime, tokens.QNameDelimiter, "_"), nil, host.GetProjectPlugins())
if err != nil {
return nil, err
}
contract.Assertf(path != "", "unexpected empty path for language plugin %s", runtime)
args, err := buildArgsForNewPlugin(host, root, options)
if err != nil {
return nil, err
}
plug, err := newPlugin(ctx, pwd, path, runtime,
workspace.LanguagePlugin, args, nil /*env*/, langRuntimePluginDialOptions(ctx, runtime))
if err != nil {
return nil, err
}
contract.Assertf(plug != nil, "unexpected nil language plugin for %s", runtime)
return &langhost{
ctx: ctx,
runtime: runtime,
root: root,
options: options,
plug: plug,
client: pulumirpc.NewLanguageRuntimeClient(plug.Conn),
}, nil
}
func langRuntimePluginDialOptions(ctx *Context, runtime string) []grpc.DialOption {
dialOpts := append(
rpcutil.OpenTracingInterceptorDialOptions(),
grpc.WithTransportCredentials(insecure.NewCredentials()),
rpcutil.GrpcChannelOptions(),
)
if ctx.DialOptions != nil {
metadata := map[string]interface{}{
"mode": "client",
"kind": "language",
}
if runtime != "" {
metadata["runtime"] = runtime
}
dialOpts = append(dialOpts, ctx.DialOptions(metadata)...)
}
return dialOpts
}
func buildArgsForNewPlugin(host Host, root string, options map[string]interface{}) ([]string, error) {
root, err := filepath.Abs(root)
if err != nil {
return nil, err
}
args := slice.Prealloc[string](len(options))
for k, v := range options {
args = append(args, fmt.Sprintf("-%s=%v", k, v))
}
args = append(args, "-root="+filepath.Clean(root))
// NOTE: positional argument for the server addresss must come last
args = append(args, host.ServerAddr())
return args, nil
}
func NewLanguageRuntimeClient(ctx *Context, runtime string, client pulumirpc.LanguageRuntimeClient) LanguageRuntime {
return &langhost{
ctx: ctx,
runtime: runtime,
client: client,
}
}
// GetRequiredPlugins computes the complete set of anticipated plugins required by a program.
func (h *langhost) GetRequiredPlugins(info ProgInfo) ([]workspace.PluginSpec, error) {
logging.V(7).Infof("langhost[%v].GetRequiredPlugins(pwd=%s,program=%s) executing",
h.runtime, info.Pwd, info.Program)
opts, err := structpb.NewStruct(h.options)
if err != nil {
return nil, fmt.Errorf("failed to marshal options: %w", err)
}
resp, err := h.client.GetRequiredPlugins(h.ctx.Request(), &pulumirpc.GetRequiredPluginsRequest{
Project: "deprecated",
Pwd: info.Pwd,
Program: info.Program,
Info: &pulumirpc.ProgramInfo{
RootDirectory: h.root,
ProgramDirectory: info.Pwd,
EntryPoint: info.Program,
Options: opts,
},
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].GetRequiredPlugins(pwd=%s,program=%s) failed: err=%v",
h.runtime, info.Pwd, info.Program, rpcError)
// It's possible this is just an older language host, prior to the emergence of the GetRequiredPlugins
// method. In such cases, we will silently error (with the above log left behind).
if rpcError.Code() == codes.Unimplemented {
return nil, nil
}
return nil, rpcError
}
results := slice.Prealloc[workspace.PluginSpec](len(resp.GetPlugins()))
for _, info := range resp.GetPlugins() {
var version *semver.Version
if v := info.GetVersion(); v != "" {
sv, err := semver.ParseTolerant(v)
if err != nil {
return nil, errors.Wrapf(err, "illegal semver returned by language host: %s@%s", info.GetName(), v)
}
version = &sv
}
if !workspace.IsPluginKind(info.Kind) {
return nil, errors.Errorf("unrecognized plugin kind: %s", info.Kind)
}
results = append(results, workspace.PluginSpec{
Name: info.Name,
Kind: workspace.PluginKind(info.Kind),
Version: version,
PluginDownloadURL: info.Server,
Checksums: info.Checksums,
})
}
logging.V(7).Infof("langhost[%v].GetRequiredPlugins(pwd=%s,program=%s) success: #versions=%d",
h.runtime, info.Pwd, info.Program, len(results))
return results, nil
}
// Run executes a program in the language runtime for planning or deployment purposes. If
// info.DryRun is true, the code must not assume that side-effects or final values resulting from
// resource deployments are actually available. If it is false, on the other hand, a real
// deployment is occurring and it may safely depend on these.
func (h *langhost) Run(info RunInfo) (string, bool, error) {
logging.V(7).Infof("langhost[%v].Run(pwd=%v,program=%v,#args=%v,proj=%s,stack=%v,#config=%v,dryrun=%v) executing",
h.runtime, info.Pwd, info.Program, len(info.Args), info.Project, info.Stack, len(info.Config), info.DryRun)
config := make(map[string]string, len(info.Config))
for k, v := range info.Config {
config[k.String()] = v
}
configSecretKeys := make([]string, len(info.ConfigSecretKeys))
for i, k := range info.ConfigSecretKeys {
configSecretKeys[i] = k.String()
}
configPropertyMap, err := MarshalProperties(info.ConfigPropertyMap,
MarshalOptions{RejectUnknowns: true, KeepSecrets: true, SkipInternalKeys: true})
if err != nil {
return "", false, err
}
opts, err := structpb.NewStruct(h.options)
if err != nil {
return "", false, fmt.Errorf("failed to marshal options: %w", err)
}
resp, err := h.client.Run(h.ctx.Request(), &pulumirpc.RunRequest{
MonitorAddress: info.MonitorAddress,
Pwd: info.Pwd,
Program: info.Program,
Args: info.Args,
Project: info.Project,
Stack: info.Stack,
Config: config,
ConfigSecretKeys: configSecretKeys,
ConfigPropertyMap: configPropertyMap,
DryRun: info.DryRun,
QueryMode: info.QueryMode,
Parallel: int32(info.Parallel),
Organization: info.Organization,
Info: &pulumirpc.ProgramInfo{
RootDirectory: h.root,
ProgramDirectory: info.Pwd,
EntryPoint: info.Program,
Options: opts,
},
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].Run(pwd=%v,program=%v,...,dryrun=%v) failed: err=%v",
h.runtime, info.Pwd, info.Program, info.DryRun, rpcError)
return "", false, rpcError
}
progerr := resp.GetError()
bail := resp.GetBail()
logging.V(7).Infof("langhost[%v].Run(pwd=%v,program=%v,...,dryrun=%v) success: progerr=%v, bail=%v",
h.runtime, info.Pwd, info.Program, info.DryRun, progerr, bail)
return progerr, bail, nil
}
// GetPluginInfo returns this plugin's information.
func (h *langhost) GetPluginInfo() (workspace.PluginInfo, error) {
logging.V(7).Infof("langhost[%v].GetPluginInfo() executing", h.runtime)
plugInfo := workspace.PluginInfo{
Name: h.runtime,
Kind: workspace.LanguagePlugin,
}
plugInfo.Path = h.plug.Bin
resp, err := h.client.GetPluginInfo(h.ctx.Request(), &emptypb.Empty{})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].GetPluginInfo() failed: err=%v", h.runtime, rpcError)
return workspace.PluginInfo{}, rpcError
}
vers := resp.Version
if vers != "" {
sv, err := semver.ParseTolerant(vers)
if err != nil {
return workspace.PluginInfo{}, err
}
plugInfo.Version = &sv
}
return plugInfo, nil
}
// Close tears down the underlying plugin RPC connection and process.
func (h *langhost) Close() error {
if h.plug != nil {
return h.plug.Close()
}
return nil
}
func (h *langhost) InstallDependencies(pwd, main string) error {
logging.V(7).Infof("langhost[%v].InstallDependencies(pwd=%s, main=%s) executing",
h.runtime, pwd, main)
opts, err := structpb.NewStruct(h.options)
if err != nil {
return fmt.Errorf("failed to marshal options: %w", err)
}
resp, err := h.client.InstallDependencies(h.ctx.Request(), &pulumirpc.InstallDependenciesRequest{
Directory: pwd,
IsTerminal: cmdutil.GetGlobalColorization() != colors.Never,
Info: &pulumirpc.ProgramInfo{
RootDirectory: h.root,
ProgramDirectory: pwd,
EntryPoint: main,
Options: opts,
},
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].InstallDependencies(pwd=%s, main=%s) failed: err=%v",
h.runtime, pwd, main, rpcError)
// It's possible this is just an older language host, prior to the emergence of the InstallDependencies
// method. In such cases, we will silently error (with the above log left behind).
if rpcError.Code() == codes.Unimplemented {
return nil
}
return rpcError
}
for {
output, err := resp.Recv()
if err != nil {
if err == io.EOF {
break
}
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].InstallDependencies(pwd=%s, main=%s) failed: err=%v",
h.runtime, pwd, main, rpcError)
return rpcError
}
if len(output.Stdout) != 0 {
os.Stdout.Write(output.Stdout)
}
if len(output.Stderr) != 0 {
os.Stderr.Write(output.Stderr)
}
}
logging.V(7).Infof("langhost[%v].InstallDependencies(pwd=%s, main=%s) success",
h.runtime, pwd, main)
return nil
}
func (h *langhost) About() (AboutInfo, error) {
logging.V(7).Infof("langhost[%v].About() executing", h.runtime)
resp, err := h.client.About(h.ctx.Request(), &emptypb.Empty{})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].About() failed: err=%v", h.runtime, rpcError)
return AboutInfo{}, rpcError
}
result := AboutInfo{
Executable: resp.Executable,
Version: resp.Version,
Metadata: resp.Metadata,
}
logging.V(7).Infof("langhost[%v].About() success", h.runtime)
return result, nil
}
func (h *langhost) GetProgramDependencies(info ProgInfo, transitiveDependencies bool) ([]DependencyInfo, error) {
prefix := fmt.Sprintf("langhost[%v].GetProgramDependencies(pwd=%s,program=%s,transitiveDependencies=%t)",
h.runtime, info.Pwd, info.Program, transitiveDependencies)
opts, err := structpb.NewStruct(h.options)
if err != nil {
return nil, fmt.Errorf("failed to marshal options: %w", err)
}
logging.V(7).Infof("%s executing", prefix)
resp, err := h.client.GetProgramDependencies(h.ctx.Request(), &pulumirpc.GetProgramDependenciesRequest{
Project: "deprecated",
Pwd: info.Pwd,
Program: info.Program,
TransitiveDependencies: transitiveDependencies,
Info: &pulumirpc.ProgramInfo{
RootDirectory: h.root,
ProgramDirectory: info.Pwd,
EntryPoint: info.Program,
Options: opts,
},
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("%s failed: err=%v", prefix, rpcError)
return nil, rpcError
}
results := slice.Prealloc[DependencyInfo](len(resp.GetDependencies()))
for _, dep := range resp.GetDependencies() {
var version semver.Version
if v := dep.Version; v != "" {
version, err = semver.ParseTolerant(v)
if err != nil {
return nil, errors.Wrapf(err, "illegal semver returned by language host: %s@%s", dep.Name, v)
}
}
results = append(results, DependencyInfo{
Name: dep.Name,
Version: version,
})
}
logging.V(7).Infof("%s success: #versions=%d", prefix, len(results))
return results, nil
}
func (h *langhost) RunPlugin(info RunPluginInfo) (io.Reader, io.Reader, context.CancelFunc, error) {
logging.V(7).Infof("langhost[%v].RunPlugin(pwd=%s,program=%s) executing",
h.runtime, info.Pwd, info.Program)
opts, err := structpb.NewStruct(h.options)
if err != nil {
return nil, nil, nil, fmt.Errorf("failed to marshal options: %w", err)
}
ctx, kill := context.WithCancel(h.ctx.Request())
resp, err := h.client.RunPlugin(ctx, &pulumirpc.RunPluginRequest{
Pwd: info.Pwd,
Program: info.Program,
Args: info.Args,
Env: info.Env,
Info: &pulumirpc.ProgramInfo{
RootDirectory: h.root,
ProgramDirectory: info.Program,
EntryPoint: ".",
Options: opts,
},
})
if err != nil {
// If there was an error starting the plugin kill the context for this request to ensure any lingering
// connection terminates.
kill()
return nil, nil, nil, err
}
outr, outw := io.Pipe()
errr, errw := io.Pipe()
go func() {
for {
logging.V(10).Infoln("Waiting for plugin message")
msg, err := resp.Recv()
if err != nil {
contract.IgnoreError(outw.CloseWithError(err))
contract.IgnoreError(errw.CloseWithError(err))
break
}
logging.V(10).Infoln("Got plugin response: ", msg)
if value, ok := msg.Output.(*pulumirpc.RunPluginResponse_Stdout); ok {
n, err := outw.Write(value.Stdout)
contract.AssertNoErrorf(err, "failed to write to stdout pipe: %v", err)
contract.Assertf(n == len(value.Stdout), "wrote fewer bytes (%d) than expected (%d)", n, len(value.Stdout))
} else if value, ok := msg.Output.(*pulumirpc.RunPluginResponse_Stderr); ok {
n, err := errw.Write(value.Stderr)
contract.AssertNoErrorf(err, "failed to write to stderr pipe: %v", err)
contract.Assertf(n == len(value.Stderr), "wrote fewer bytes (%d) than expected (%d)", n, len(value.Stderr))
} else if _, ok := msg.Output.(*pulumirpc.RunPluginResponse_Exitcode); ok {
// If stdout and stderr are empty we've flushed and are returning the exit code
outw.Close()
errw.Close()
break
}
}
}()
return outr, errr, kill, nil
}
func (h *langhost) GenerateProject(
sourceDirectory, targetDirectory, project string, strict bool,
loaderTarget string, localDependencies map[string]string,
) (hcl.Diagnostics, error) {
logging.V(7).Infof("langhost[%v].GenerateProject() executing", h.runtime)
resp, err := h.client.GenerateProject(h.ctx.Request(), &pulumirpc.GenerateProjectRequest{
SourceDirectory: sourceDirectory,
TargetDirectory: targetDirectory,
Project: project,
Strict: strict,
LoaderTarget: loaderTarget,
LocalDependencies: localDependencies,
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].GenerateProject() failed: err=%v", h.runtime, rpcError)
return nil, rpcError
}
// 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("langhost[%v].GenerateProject() success", h.runtime)
return diags, nil
}
func (h *langhost) GeneratePackage(
directory string, schema string, extraFiles map[string][]byte, loaderTarget string,
) (hcl.Diagnostics, error) {
logging.V(7).Infof("langhost[%v].GeneratePackage() executing", h.runtime)
resp, err := h.client.GeneratePackage(h.ctx.Request(), &pulumirpc.GeneratePackageRequest{
Directory: directory,
Schema: schema,
ExtraFiles: extraFiles,
LoaderTarget: loaderTarget,
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].GeneratePackage() failed: err=%v", h.runtime, rpcError)
return nil, rpcError
}
logging.V(7).Infof("langhost[%v].GeneratePackage() success", h.runtime)
// Translate the rpc diagnostics into hcl.Diagnostics.
var diags hcl.Diagnostics
for _, rpcDiag := range resp.Diagnostics {
diags = append(diags, RPCDiagnosticToHclDiagnostic(rpcDiag))
}
return diags, nil
}
func (h *langhost) GenerateProgram(program map[string]string, loaderTarget string,
) (map[string][]byte, hcl.Diagnostics, error) {
logging.V(7).Infof("langhost[%v].GenerateProgram() executing", h.runtime)
resp, err := h.client.GenerateProgram(h.ctx.Request(), &pulumirpc.GenerateProgramRequest{
Source: program,
LoaderTarget: loaderTarget,
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("langhost[%v].GenerateProgram() failed: err=%v", h.runtime, rpcError)
return nil, nil, rpcError
}
// 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("langhost[%v].GenerateProgram() success", h.runtime)
return resp.Source, diags, nil
}
func (h *langhost) Pack(
packageDirectory string, version semver.Version, destinationDirectory string,
) (string, error) {
label := fmt.Sprintf("langhost[%v].Pack(%s, %s, %s)", h.runtime, packageDirectory, version, destinationDirectory)
logging.V(7).Infof("%s executing", label)
// Always send absolute paths to the plugin, as it may be running in a different working directory.
packageDirectory, err := filepath.Abs(packageDirectory)
if err != nil {
return "", err
}
destinationDirectory, err = filepath.Abs(destinationDirectory)
if err != nil {
return "", err
}
req, err := h.client.Pack(h.ctx.Request(), &pulumirpc.PackRequest{
PackageDirectory: packageDirectory,
Version: version.String(),
DestinationDirectory: destinationDirectory,
})
if err != nil {
rpcError := rpcerror.Convert(err)
logging.V(7).Infof("%s failed: err=%v", label, rpcError)
return "", rpcError
}
logging.V(7).Infof("%s success: artifactPath=%s", label, req.ArtifactPath)
return req.ArtifactPath, nil
}