mirror of https://github.com/pulumi/pulumi.git
717 lines
25 KiB
Go
717 lines
25 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 (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"sync"
|
|
|
|
"github.com/blang/semver"
|
|
"github.com/hashicorp/go-multierror"
|
|
|
|
pkgWorkspace "github.com/pulumi/pulumi/pkg/v3/workspace"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/apitype"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/diag"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/env"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/config"
|
|
"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/util/logging"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
|
|
)
|
|
|
|
// A Host hosts provider plugins and makes them easily accessible by package name.
|
|
type Host interface {
|
|
// ServerAddr returns the address at which the host's RPC interface may be found.
|
|
ServerAddr() string
|
|
|
|
// Log logs a message, including errors and warnings. Messages can have a resource URN
|
|
// associated with them. If no urn is provided, the message is global.
|
|
Log(sev diag.Severity, urn resource.URN, msg string, streamID int32)
|
|
|
|
// LogStatus logs a status message message, including errors and warnings. Status messages show
|
|
// up in the `Info` column of the progress display, but not in the final output. Messages can
|
|
// have a resource URN associated with them. If no urn is provided, the message is global.
|
|
LogStatus(sev diag.Severity, urn resource.URN, msg string, streamID int32)
|
|
|
|
// Analyzer fetches the analyzer with a given name, possibly lazily allocating the plugins for
|
|
// it. If an analyzer could not be found, or an error occurred while creating it, a non-nil
|
|
// error is returned.
|
|
Analyzer(nm tokens.QName) (Analyzer, error)
|
|
|
|
// PolicyAnalyzer boots the nodejs analyzer plugin located at a given path. This is useful
|
|
// because policy analyzers generally do not need to be "discovered" -- the engine is given a
|
|
// set of policies that are required to be run during an update, so they tend to be in a
|
|
// well-known place.
|
|
PolicyAnalyzer(name tokens.QName, path string, opts *PolicyAnalyzerOptions) (Analyzer, error)
|
|
|
|
// ListAnalyzers returns a list of all analyzer plugins known to the plugin host.
|
|
ListAnalyzers() []Analyzer
|
|
|
|
// Provider loads a new copy of the provider for a given package. If a provider for this package could not be
|
|
// found, or an error occurs while creating it, a non-nil error is returned.
|
|
Provider(descriptor workspace.PackageDescriptor) (Provider, error)
|
|
// CloseProvider closes the given provider plugin and deregisters it from this host.
|
|
CloseProvider(provider Provider) error
|
|
// LanguageRuntime fetches the language runtime plugin for a given language, lazily allocating if necessary. If
|
|
// an implementation of this language runtime wasn't found, on an error occurs, a non-nil error is returned.
|
|
LanguageRuntime(runtime string, info ProgramInfo) (LanguageRuntime, error)
|
|
|
|
// EnsurePlugins ensures all plugins in the given array are loaded and ready to use. If any plugins are missing,
|
|
// and/or there are errors loading one or more plugins, a non-nil error is returned.
|
|
EnsurePlugins(plugins []workspace.PluginSpec, kinds Flags) error
|
|
|
|
// ResolvePlugin resolves a plugin kind, name, and optional semver to a candidate plugin to load.
|
|
ResolvePlugin(kind apitype.PluginKind, name string, version *semver.Version) (*workspace.PluginInfo, error)
|
|
|
|
GetProjectPlugins() []workspace.ProjectPlugin
|
|
|
|
// SignalCancellation asks all resource providers to gracefully shut down and abort any ongoing
|
|
// operations. Operation aborted in this way will return an error (e.g., `Update` and `Create`
|
|
// will either a creation error or an initialization error. SignalCancellation is advisory and
|
|
// non-blocking; it is up to the host to decide how long to wait after SignalCancellation is
|
|
// called before (e.g.) hard-closing any gRPC connection.
|
|
SignalCancellation() error
|
|
|
|
// StartDebugging asks the host to start a debugging session with the given configuration.
|
|
StartDebugging(DebuggingInfo) error
|
|
|
|
// Close reclaims any resources associated with the host.
|
|
Close() error
|
|
}
|
|
|
|
// NewDefaultHost implements the standard plugin logic, using the standard installation root to find them.
|
|
func NewDefaultHost(ctx *Context, runtimeOptions map[string]interface{},
|
|
disableProviderPreview bool, plugins *workspace.Plugins, config map[config.Key]string,
|
|
debugging DebugEventEmitter,
|
|
) (Host, error) {
|
|
// Create plugin info from providers
|
|
projectPlugins := make([]workspace.ProjectPlugin, 0)
|
|
if plugins != nil {
|
|
for _, providerOpts := range plugins.Providers {
|
|
info, err := parsePluginOpts(ctx.Root, providerOpts, apitype.ResourcePlugin)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
projectPlugins = append(projectPlugins, info)
|
|
}
|
|
for _, languageOpts := range plugins.Languages {
|
|
info, err := parsePluginOpts(ctx.Root, languageOpts, apitype.LanguagePlugin)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
projectPlugins = append(projectPlugins, info)
|
|
}
|
|
for _, analyzerOpts := range plugins.Analyzers {
|
|
info, err := parsePluginOpts(ctx.Root, analyzerOpts, apitype.AnalyzerPlugin)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
projectPlugins = append(projectPlugins, info)
|
|
}
|
|
}
|
|
|
|
host := &defaultHost{
|
|
ctx: ctx,
|
|
runtimeOptions: runtimeOptions,
|
|
analyzerPlugins: make(map[tokens.QName]*analyzerPlugin),
|
|
languagePlugins: make(map[string]*languagePlugin),
|
|
resourcePlugins: make(map[Provider]*resourcePlugin),
|
|
reportedResourcePlugins: make(map[string]struct{}),
|
|
languageLoadRequests: make(chan pluginLoadRequest),
|
|
loadRequests: make(chan pluginLoadRequest),
|
|
disableProviderPreview: disableProviderPreview,
|
|
config: config,
|
|
closer: new(sync.Once),
|
|
projectPlugins: projectPlugins,
|
|
debugging: debugging,
|
|
}
|
|
|
|
// Fire up a gRPC server to listen for requests. This acts as a RPC interface that plugins can use
|
|
// to "phone home" in case there are things the host must do on behalf of the plugins (like log, etc).
|
|
svr, err := newHostServer(host, ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
host.server = svr
|
|
|
|
// Start a goroutine we'll use to satisfy load requests serially and avoid race conditions.
|
|
go func() {
|
|
for req := range host.loadRequests {
|
|
req.result <- req.load()
|
|
}
|
|
}()
|
|
|
|
// Start another goroutine we'll use to satisfy load language plugin requests, this is so other plugins
|
|
// can be started up by a language plugin.
|
|
go func() {
|
|
for req := range host.languageLoadRequests {
|
|
req.result <- req.load()
|
|
}
|
|
}()
|
|
|
|
return host, nil
|
|
}
|
|
|
|
func parsePluginOpts(
|
|
root string, providerOpts workspace.PluginOptions, k apitype.PluginKind,
|
|
) (workspace.ProjectPlugin, error) {
|
|
handleErr := func(msg string, a ...interface{}) (workspace.ProjectPlugin, error) {
|
|
return workspace.ProjectPlugin{},
|
|
fmt.Errorf("parsing plugin options for '%s': %w", providerOpts.Name, fmt.Errorf(msg, a...))
|
|
}
|
|
if providerOpts.Name == "" {
|
|
return handleErr("name must not be empty")
|
|
}
|
|
var v *semver.Version
|
|
if providerOpts.Version != "" {
|
|
ver, err := semver.Parse(providerOpts.Version)
|
|
if err != nil {
|
|
return workspace.ProjectPlugin{}, err
|
|
}
|
|
v = &ver
|
|
}
|
|
|
|
stat, err := os.Stat(providerOpts.Path)
|
|
if os.IsNotExist(err) {
|
|
return handleErr("no folder at path '%s'", providerOpts.Path)
|
|
} else if err != nil {
|
|
return handleErr("checking provider folder: %w", err)
|
|
} else if !stat.IsDir() {
|
|
return handleErr("provider folder '%s' is not a directory", providerOpts.Path)
|
|
}
|
|
|
|
// The path is relative to the project root. Make it absolute here so we don't need to track that everywhere its used.
|
|
path := providerOpts.Path
|
|
if !filepath.IsAbs(path) {
|
|
path, err = filepath.Abs(filepath.Join(root, path))
|
|
if err != nil {
|
|
return handleErr("getting absolute path for plugin path %s: %w", providerOpts.Path, err)
|
|
}
|
|
}
|
|
|
|
pluginInfo := workspace.ProjectPlugin{
|
|
Name: providerOpts.Name,
|
|
Path: path,
|
|
Kind: k,
|
|
Version: v,
|
|
}
|
|
return pluginInfo, nil
|
|
}
|
|
|
|
// PolicyAnalyzerOptions includes a bag of options to pass along to a policy analyzer.
|
|
type PolicyAnalyzerOptions struct {
|
|
Organization string
|
|
Project string
|
|
Stack string
|
|
Config map[config.Key]string
|
|
DryRun bool
|
|
}
|
|
|
|
type pluginLoadRequest struct {
|
|
load func() error
|
|
result chan<- error
|
|
}
|
|
|
|
type defaultHost struct {
|
|
ctx *Context // the shared context for this host.
|
|
|
|
// the runtime options for the project, passed to resource providers to support dynamic providers.
|
|
runtimeOptions map[string]interface{}
|
|
analyzerPlugins map[tokens.QName]*analyzerPlugin // a cache of analyzer plugins and their processes.
|
|
languagePlugins map[string]*languagePlugin // a cache of language plugins and their processes.
|
|
resourcePlugins map[Provider]*resourcePlugin // the set of loaded resource plugins.
|
|
reportedResourcePlugins map[string]struct{} // the set of unique resource plugins we'll report.
|
|
languageLoadRequests chan pluginLoadRequest // a channel used to satisfy language load requests.
|
|
loadRequests chan pluginLoadRequest // a channel used to satisfy plugin load requests.
|
|
server *hostServer // the server's RPC machinery.
|
|
disableProviderPreview bool // true if provider plugins should disable provider preview
|
|
config map[config.Key]string // the configuration map for the stack, if any.
|
|
debugging DebugEventEmitter
|
|
|
|
// Used to synchronize shutdown with in-progress plugin loads.
|
|
pluginLock sync.RWMutex
|
|
|
|
closer *sync.Once
|
|
projectPlugins []workspace.ProjectPlugin
|
|
}
|
|
|
|
var _ Host = (*defaultHost)(nil)
|
|
|
|
type analyzerPlugin struct {
|
|
Plugin Analyzer
|
|
Info workspace.PluginInfo
|
|
}
|
|
|
|
type languagePlugin struct {
|
|
Plugin LanguageRuntime
|
|
Info workspace.PluginInfo
|
|
}
|
|
|
|
type resourcePlugin struct {
|
|
Plugin Provider
|
|
Info workspace.PluginInfo
|
|
}
|
|
|
|
func (host *defaultHost) ServerAddr() string {
|
|
return host.server.Address()
|
|
}
|
|
|
|
func (host *defaultHost) Log(sev diag.Severity, urn resource.URN, msg string, streamID int32) {
|
|
host.ctx.Diag.Logf(sev, diag.StreamMessage(urn, msg, streamID))
|
|
}
|
|
|
|
func (host *defaultHost) LogStatus(sev diag.Severity, urn resource.URN, msg string, streamID int32) {
|
|
host.ctx.StatusDiag.Logf(sev, diag.StreamMessage(urn, msg, streamID))
|
|
}
|
|
|
|
func (host *defaultHost) StartDebugging(info DebuggingInfo) error {
|
|
contract.Assertf(host.debugging != nil, "expected host.debugging to be non-nil")
|
|
return host.debugging.StartDebugging(info)
|
|
}
|
|
|
|
// loadPlugin sends an appropriate load request to the plugin loader and returns the loaded plugin (if any) and error.
|
|
func (host *defaultHost) loadPlugin(
|
|
loadRequestChannel chan pluginLoadRequest, load func() (interface{}, error),
|
|
) (interface{}, error) {
|
|
var plugin interface{}
|
|
|
|
locked := host.pluginLock.TryRLock()
|
|
if !locked {
|
|
// If we couldn't get a read lock that must be because we're shutting down, so just return an error.
|
|
return nil, errors.New("plugin host is shutting down")
|
|
}
|
|
defer host.pluginLock.RUnlock()
|
|
|
|
result := make(chan error)
|
|
loadRequestChannel <- pluginLoadRequest{
|
|
load: func() error {
|
|
p, err := load()
|
|
plugin = p
|
|
return err
|
|
},
|
|
result: result,
|
|
}
|
|
return plugin, <-result
|
|
}
|
|
|
|
func (host *defaultHost) Analyzer(name tokens.QName) (Analyzer, error) {
|
|
plugin, err := host.loadPlugin(host.loadRequests, func() (interface{}, error) {
|
|
// First see if we already loaded this plugin.
|
|
if plug, has := host.analyzerPlugins[name]; has {
|
|
contract.Assertf(plug != nil, "analyzer plugin %v was loaded but is nil", name)
|
|
return plug.Plugin, nil
|
|
}
|
|
|
|
// If not, try to load and bind to a plugin.
|
|
plug, err := NewAnalyzer(host, host.ctx, name)
|
|
if err == nil && plug != nil {
|
|
info, infoerr := plug.GetPluginInfo()
|
|
if infoerr != nil {
|
|
return nil, infoerr
|
|
}
|
|
|
|
// Memoize the result.
|
|
host.analyzerPlugins[name] = &analyzerPlugin{Plugin: plug, Info: info}
|
|
}
|
|
|
|
return plug, err
|
|
})
|
|
if plugin == nil || err != nil {
|
|
return nil, err
|
|
}
|
|
return plugin.(Analyzer), nil
|
|
}
|
|
|
|
func (host *defaultHost) PolicyAnalyzer(name tokens.QName, path string, opts *PolicyAnalyzerOptions) (Analyzer, error) {
|
|
plugin, err := host.loadPlugin(host.loadRequests, func() (interface{}, error) {
|
|
// First see if we already loaded this plugin.
|
|
if plug, has := host.analyzerPlugins[name]; has {
|
|
contract.Assertf(plug != nil, "analyzer plugin %v was loaded but is nil", name)
|
|
return plug.Plugin, nil
|
|
}
|
|
|
|
// If not, try to load and bind to a plugin.
|
|
plug, err := NewPolicyAnalyzer(host, host.ctx, name, path, opts)
|
|
if err == nil && plug != nil {
|
|
info, infoerr := plug.GetPluginInfo()
|
|
if infoerr != nil {
|
|
return nil, infoerr
|
|
}
|
|
|
|
// Memoize the result.
|
|
host.analyzerPlugins[name] = &analyzerPlugin{Plugin: plug, Info: info}
|
|
}
|
|
|
|
return plug, err
|
|
})
|
|
if plugin == nil || err != nil {
|
|
return nil, err
|
|
}
|
|
return plugin.(Analyzer), nil
|
|
}
|
|
|
|
func (host *defaultHost) ListAnalyzers() []Analyzer {
|
|
analyzers := []Analyzer{}
|
|
for _, analyzer := range host.analyzerPlugins {
|
|
analyzers = append(analyzers, analyzer.Plugin)
|
|
}
|
|
return analyzers
|
|
}
|
|
|
|
func (host *defaultHost) Provider(descriptor workspace.PackageDescriptor) (Provider, error) {
|
|
load := func() (interface{}, error) {
|
|
pkg := descriptor.Name
|
|
version := descriptor.Version
|
|
|
|
// There are cases where a provider may want to make use of configuration defined at a stack level, as well
|
|
// as configuration that is explicitly passed to the provider during its lifetime (e.g. with a Configure call).
|
|
// To support these cases, we set the PULUMI_CONFIG environment variable to a JSON object containing host
|
|
// configuration that is namespaced by the provider's package, stripping the namespace from the keys in the
|
|
// configuration object.
|
|
//
|
|
// So, given an example host configuration of:
|
|
//
|
|
// config:
|
|
// aws:region: eu-west-1
|
|
// foo:bar: baz
|
|
//
|
|
// * A provider with package "aws" would receive:
|
|
//
|
|
// PULUMI_CONFIG='{"region":"eu-west-1"}'
|
|
//
|
|
// * A provider with package "gcp" would receive:
|
|
//
|
|
// PULUMI_CONFIG='{}'
|
|
result := make(map[string]string)
|
|
for k, v := range host.config {
|
|
if k.Namespace() != pkg {
|
|
continue
|
|
}
|
|
result[k.Name()] = v
|
|
}
|
|
jsonConfig, err := json.Marshal(result)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Could not marshal config to JSON: %w", err)
|
|
}
|
|
|
|
plug, err := NewProvider(
|
|
host, host.ctx, tokens.Package(pkg), version,
|
|
host.runtimeOptions, host.disableProviderPreview, string(jsonConfig))
|
|
if err == nil && plug != nil {
|
|
info, infoerr := plug.GetPluginInfo(host.ctx.Request())
|
|
if infoerr != nil {
|
|
return nil, infoerr
|
|
}
|
|
|
|
// Warn if the plugin version was not what we expected.
|
|
if version != nil && !env.Dev.Value() {
|
|
if info.Version == nil || !info.Version.GTE(*version) {
|
|
var v string
|
|
if info.Version != nil {
|
|
v = info.Version.String()
|
|
}
|
|
host.ctx.Diag.Warningf(
|
|
diag.Message("", /*urn*/
|
|
"resource plugin %s is expected to have version >=%s, but has %s; "+
|
|
"the wrong version may be on your path, or this may be a bug in the plugin"),
|
|
info.Name, version.String(), v)
|
|
}
|
|
}
|
|
|
|
// Record the result and add the plugin's info to our list of loaded plugins if it's the first copy of its
|
|
// kind.
|
|
key := info.Name
|
|
if info.Version != nil {
|
|
key += info.Version.String()
|
|
}
|
|
_, alreadyReported := host.reportedResourcePlugins[key]
|
|
if !alreadyReported {
|
|
host.reportedResourcePlugins[key] = struct{}{}
|
|
}
|
|
host.resourcePlugins[plug] = &resourcePlugin{Plugin: plug, Info: info}
|
|
}
|
|
|
|
return plug, err
|
|
}
|
|
|
|
plugin, err := host.loadPlugin(host.loadRequests, load)
|
|
if err == nil && plugin != nil {
|
|
return plugin.(Provider), nil
|
|
}
|
|
|
|
// We might fail to load a plugin because it's missing, e.g. due to it not being picked up by GetRequiredPlugins.
|
|
// Before we give up and report an error, we'll first try to install it and attempt another load.
|
|
|
|
var me *workspace.MissingError
|
|
if !errors.As(err, &me) {
|
|
// We didn't fail because the plugin was missing -- return the error as-is.
|
|
return nil, err
|
|
}
|
|
|
|
// If automatic plugin installation is disabled, we can't attempt an installation, so return the missing error as-is.
|
|
if env.DisableAutomaticPluginAcquisition.Value() {
|
|
return nil, err
|
|
}
|
|
|
|
log := func(sev diag.Severity, msg string) {
|
|
host.Log(sev, "", msg, 0)
|
|
}
|
|
|
|
_, err = pkgWorkspace.InstallPlugin(descriptor.PluginSpec, log)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
plugin, err = host.loadPlugin(host.loadRequests, load)
|
|
if err == nil && plugin != nil {
|
|
return plugin.(Provider), nil
|
|
}
|
|
|
|
return nil, err
|
|
}
|
|
|
|
func (host *defaultHost) LanguageRuntime(runtime string, info ProgramInfo,
|
|
) (LanguageRuntime, error) {
|
|
// Language runtimes use their own loading channel not the main one
|
|
plugin, err := host.loadPlugin(host.languageLoadRequests, func() (interface{}, error) {
|
|
// Key our cached runtime plugins by the runtime name and the options
|
|
jsonOptions, err := json.Marshal(info.Options())
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not marshal runtime options to JSON: %w", err)
|
|
}
|
|
|
|
key := runtime + ":" + info.RootDirectory() + ":" + info.ProgramDirectory() + ":" + string(jsonOptions)
|
|
|
|
// First see if we already loaded this plugin.
|
|
if plug, has := host.languagePlugins[key]; has {
|
|
contract.Assertf(plug != nil, "language plugin %v was loaded but is nil", key)
|
|
return plug.Plugin, nil
|
|
}
|
|
|
|
// If not, allocate a new one.
|
|
plug, err := NewLanguageRuntime(host, host.ctx, runtime, host.ctx.Pwd, info)
|
|
if err == nil && plug != nil {
|
|
info, infoerr := plug.GetPluginInfo()
|
|
if infoerr != nil {
|
|
return nil, infoerr
|
|
}
|
|
|
|
// Memoize the result.
|
|
host.languagePlugins[key] = &languagePlugin{Plugin: plug, Info: info}
|
|
}
|
|
|
|
return plug, err
|
|
})
|
|
if plugin == nil || err != nil {
|
|
return nil, err
|
|
}
|
|
return plugin.(LanguageRuntime), nil
|
|
}
|
|
|
|
// EnsurePlugins ensures all plugins in the given array are loaded and ready to use. If any plugins are missing,
|
|
// and/or there are errors loading one or more plugins, a non-nil error is returned.
|
|
func (host *defaultHost) EnsurePlugins(plugins []workspace.PluginSpec, kinds Flags) error {
|
|
// Use a multieerror to track failures so we can return one big list of all failures at the end.
|
|
var result error
|
|
for _, plugin := range plugins {
|
|
switch plugin.Kind {
|
|
case apitype.AnalyzerPlugin:
|
|
if kinds&AnalyzerPlugins != 0 {
|
|
if _, err := host.Analyzer(tokens.QName(plugin.Name)); err != nil {
|
|
result = multierror.Append(result,
|
|
fmt.Errorf("failed to load analyzer plugin %s: %w", plugin.Name, err))
|
|
}
|
|
}
|
|
case apitype.LanguagePlugin:
|
|
if kinds&LanguagePlugins != 0 {
|
|
// Pass nil options here, we just need to check the language plugin is loadable. We can't use
|
|
// host.runtimePlugins because there might be other language plugins reported here (e.g
|
|
// shimless multi-language providers). Pass the host root for the plugin directory, it
|
|
// shouldn't matter because we're starting with no options but it's a directory we've already
|
|
// got hold of.
|
|
info := NewProgramInfo(host.ctx.Root, host.ctx.Pwd, ".", nil)
|
|
if _, err := host.LanguageRuntime(plugin.Name, info); err != nil {
|
|
result = multierror.Append(result,
|
|
fmt.Errorf("failed to load language plugin %s: %w", plugin.Name, err))
|
|
}
|
|
}
|
|
case apitype.ResourcePlugin:
|
|
if kinds&ResourcePlugins != 0 {
|
|
if _, err := host.Provider(workspace.PackageDescriptor{PluginSpec: plugin}); err != nil {
|
|
result = multierror.Append(result,
|
|
fmt.Errorf("failed to load resource plugin %s: %w", plugin.Name, err))
|
|
}
|
|
}
|
|
case apitype.ConverterPlugin, apitype.ToolPlugin:
|
|
contract.Failf("unexpected plugin kind: %s", plugin.Kind)
|
|
}
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func (host *defaultHost) ResolvePlugin(
|
|
kind apitype.PluginKind, name string, version *semver.Version,
|
|
) (*workspace.PluginInfo, error) {
|
|
return workspace.GetPluginInfo(host.ctx.Diag, kind, name, version, host.GetProjectPlugins())
|
|
}
|
|
|
|
func (host *defaultHost) GetProjectPlugins() []workspace.ProjectPlugin {
|
|
return host.projectPlugins
|
|
}
|
|
|
|
func (host *defaultHost) SignalCancellation() error {
|
|
// NOTE: we're abusing loadPlugin in order to ensure proper synchronization.
|
|
_, err := host.loadPlugin(host.loadRequests, func() (interface{}, error) {
|
|
var result error
|
|
for _, plug := range host.resourcePlugins {
|
|
if err := plug.Plugin.SignalCancellation(host.ctx.Request()); err != nil {
|
|
result = multierror.Append(result, fmt.Errorf(
|
|
"Error signaling cancellation to resource provider '%s': %w", plug.Info.Name, err))
|
|
}
|
|
}
|
|
return nil, result
|
|
})
|
|
return err
|
|
}
|
|
|
|
func (host *defaultHost) CloseProvider(provider Provider) error {
|
|
// NOTE: we're abusing loadPlugin in order to ensure proper synchronization.
|
|
_, err := host.loadPlugin(host.loadRequests, func() (interface{}, error) {
|
|
if err := provider.Close(); err != nil {
|
|
return nil, err
|
|
}
|
|
delete(host.resourcePlugins, provider)
|
|
return nil, nil
|
|
})
|
|
return err
|
|
}
|
|
|
|
func (host *defaultHost) Close() (err error) {
|
|
host.closer.Do(func() {
|
|
// Wait for all plugins to finish loading, we do this by taking a Write lock on the pluginLock. This
|
|
// won't take until all read locks are released (indicating that no plugins are currently loading) and
|
|
// it will then block further read locks from being taken (preventing any new plugins from loading).
|
|
host.pluginLock.Lock()
|
|
// N.B We purposefully do not unlock this.
|
|
|
|
// Close all plugins.
|
|
for _, plug := range host.analyzerPlugins {
|
|
if err := plug.Plugin.Close(); err != nil {
|
|
logging.V(5).Infof("Error closing '%s' analyzer plugin during shutdown; ignoring: %v", plug.Info.Name, err)
|
|
}
|
|
}
|
|
for _, plug := range host.resourcePlugins {
|
|
if err := plug.Plugin.Close(); err != nil {
|
|
logging.V(5).Infof("Error closing '%s' resource plugin during shutdown; ignoring: %v", plug.Info.Name, err)
|
|
}
|
|
}
|
|
for _, plug := range host.languagePlugins {
|
|
if err := plug.Plugin.Close(); err != nil {
|
|
logging.V(5).Infof("Error closing '%s' language plugin during shutdown; ignoring: %v", plug.Info.Name, err)
|
|
}
|
|
}
|
|
|
|
// Empty out all maps.
|
|
host.analyzerPlugins = make(map[tokens.QName]*analyzerPlugin)
|
|
host.languagePlugins = make(map[string]*languagePlugin)
|
|
host.resourcePlugins = make(map[Provider]*resourcePlugin)
|
|
|
|
// Shut down the plugin loader.
|
|
close(host.languageLoadRequests)
|
|
close(host.loadRequests)
|
|
|
|
// Finally, shut down the host's gRPC server.
|
|
err = host.server.Cancel()
|
|
})
|
|
return err
|
|
}
|
|
|
|
// Flags can be used to filter out plugins during loading that aren't necessary.
|
|
type Flags int
|
|
|
|
const (
|
|
// AnalyzerPlugins is used to only load analyzers.
|
|
AnalyzerPlugins Flags = 1 << iota
|
|
// LanguagePlugins is used to only load language plugins.
|
|
LanguagePlugins
|
|
// ResourcePlugins is used to only load resource provider plugins.
|
|
ResourcePlugins
|
|
)
|
|
|
|
// AllPlugins uses flags to ensure that all plugin kinds are loaded.
|
|
var AllPlugins = AnalyzerPlugins | LanguagePlugins | ResourcePlugins
|
|
|
|
// GetRequiredPlugins lists a full set of plugins that will be required by the given program.
|
|
func GetRequiredPlugins(
|
|
host Host,
|
|
runtime string,
|
|
project string,
|
|
info ProgramInfo,
|
|
kinds Flags) (
|
|
[]workspace.PluginSpec, error,
|
|
) {
|
|
var plugins []workspace.PluginSpec
|
|
|
|
if kinds&LanguagePlugins != 0 {
|
|
// First make sure the language plugin is present. We need this to load the required resource plugins.
|
|
// TODO: we need to think about how best to version this. For now, it always picks the latest.
|
|
lang, err := host.LanguageRuntime(runtime, info)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to load language plugin %s: %w", runtime, err)
|
|
}
|
|
// Query the language runtime plugin for its version.
|
|
langInfo, err := lang.GetPluginInfo()
|
|
if err != nil {
|
|
// Don't error if this fails, just warn and return the version as unknown.
|
|
host.Log(diag.Warning, "", fmt.Sprintf("failed to get plugin info for language plugin %s: %v", runtime, err), 0)
|
|
plugins = append(plugins, workspace.PluginSpec{
|
|
Name: runtime,
|
|
Kind: apitype.LanguagePlugin,
|
|
})
|
|
} else {
|
|
plugins = append(plugins, workspace.PluginSpec{
|
|
Name: langInfo.Name,
|
|
Kind: langInfo.Kind,
|
|
Version: langInfo.Version,
|
|
})
|
|
}
|
|
|
|
if kinds&ResourcePlugins != 0 {
|
|
// Use the language plugin to compute this project's set of plugin dependencies.
|
|
// TODO: we want to support loading precisely what the project needs, rather than doing a static scan of resolved
|
|
// packages. Doing this requires that we change our RPC interface and figure out how to configure plugins
|
|
// later than we do (right now, we do it up front, but at that point we don't know the version).
|
|
deps, err := lang.GetRequiredPlugins(info)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to discover plugin requirements: %w", err)
|
|
}
|
|
plugins = append(plugins, deps...)
|
|
}
|
|
} else {
|
|
// If we can't load the language plugin, we can't discover the resource plugins.
|
|
contract.Assertf(kinds&ResourcePlugins != 0,
|
|
"cannot load resource plugins without also loading the language plugin")
|
|
}
|
|
|
|
return plugins, nil
|
|
}
|