2023-05-12 17:52:30 +00:00
|
|
|
// 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 workspace
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"time"
|
|
|
|
|
2023-05-27 08:47:06 +00:00
|
|
|
"github.com/blang/semver"
|
|
|
|
|
2023-06-28 12:45:57 +00:00
|
|
|
"github.com/pulumi/pulumi/pkg/v3/util"
|
2023-05-12 17:52:30 +00:00
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/diag"
|
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/logging"
|
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
|
|
|
|
)
|
|
|
|
|
|
|
|
// InstallPluginError is returned by InstallPlugin if we couldn't install the plugin
|
|
|
|
type InstallPluginError struct {
|
2023-05-12 18:05:11 +00:00
|
|
|
// The specification of the plugin to install
|
|
|
|
Spec workspace.PluginSpec
|
2023-05-12 17:52:30 +00:00
|
|
|
// The underlying error that occurred during the download or install.
|
|
|
|
Err error
|
|
|
|
}
|
|
|
|
|
|
|
|
func (err *InstallPluginError) Error() string {
|
|
|
|
var server string
|
2023-05-12 18:05:11 +00:00
|
|
|
if err.Spec.PluginDownloadURL != "" {
|
2023-12-12 12:19:42 +00:00
|
|
|
server = " --server " + err.Spec.PluginDownloadURL
|
2023-05-12 17:52:30 +00:00
|
|
|
}
|
|
|
|
|
2023-05-12 18:05:11 +00:00
|
|
|
if err.Spec.Version != nil {
|
2023-05-12 17:52:30 +00:00
|
|
|
return fmt.Sprintf("Could not automatically download and install %[1]s plugin 'pulumi-%[1]s-%[2]s'"+
|
|
|
|
" at version v%[3]s"+
|
|
|
|
", install the plugin using `pulumi plugin install %[1]s %[2]s v%[3]s%[4]s`: %[5]v",
|
2023-05-12 18:05:11 +00:00
|
|
|
err.Spec.Kind, err.Spec.Name, err.Spec.Version, server, err.Err)
|
2023-05-12 17:52:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Sprintf("Could not automatically download and install %[1]s plugin 'pulumi-%[1]s-%[2]s'"+
|
|
|
|
", install the plugin using `pulumi plugin install %[1]s %[2]s%[3]s`: %[4]v",
|
2023-05-12 18:05:11 +00:00
|
|
|
err.Spec.Kind, err.Spec.Name, server, err.Err)
|
2023-05-12 17:52:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (err *InstallPluginError) Unwrap() error {
|
|
|
|
return err.Err
|
|
|
|
}
|
|
|
|
|
2023-05-27 08:47:06 +00:00
|
|
|
func InstallPlugin(pluginSpec workspace.PluginSpec, log func(sev diag.Severity, msg string)) (*semver.Version, error) {
|
2023-07-13 22:08:54 +00:00
|
|
|
util.SetKnownPluginDownloadURL(&pluginSpec)
|
2023-12-08 08:55:49 +00:00
|
|
|
util.SetKnownPluginVersion(&pluginSpec)
|
2023-05-12 17:52:30 +00:00
|
|
|
if pluginSpec.Version == nil {
|
|
|
|
var err error
|
|
|
|
pluginSpec.Version, err = pluginSpec.GetLatestVersion()
|
|
|
|
if err != nil {
|
2023-05-27 08:47:06 +00:00
|
|
|
return nil, fmt.Errorf("could not find latest version for provider %s: %w", pluginSpec.Name, err)
|
2023-05-12 17:52:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
wrapper := func(stream io.ReadCloser, size int64) io.ReadCloser {
|
2023-10-14 08:32:43 +00:00
|
|
|
// Log at info but to stderr so we don't pollute stdout for commands like `package get-schema`
|
2023-12-12 12:19:42 +00:00
|
|
|
log(diag.Infoerr, "Downloading provider: "+pluginSpec.Name)
|
2023-05-12 17:52:30 +00:00
|
|
|
return stream
|
|
|
|
}
|
|
|
|
|
|
|
|
retry := func(err error, attempt int, limit int, delay time.Duration) {
|
|
|
|
log(diag.Warning, fmt.Sprintf("error downloading provider: %s\n"+
|
|
|
|
"Will retry in %v [%d/%d]", err, delay, attempt, limit))
|
|
|
|
}
|
|
|
|
|
|
|
|
logging.V(1).Infof("Automatically downloading provider %s", pluginSpec.Name)
|
|
|
|
downloadedFile, err := workspace.DownloadToFile(pluginSpec, wrapper, retry)
|
|
|
|
if err != nil {
|
2023-05-27 08:47:06 +00:00
|
|
|
return nil, &InstallPluginError{
|
2023-05-12 18:05:11 +00:00
|
|
|
Spec: pluginSpec,
|
|
|
|
Err: fmt.Errorf("error downloading provider %s to file: %w", pluginSpec.Name, err),
|
2023-05-12 17:52:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
logging.V(1).Infof("Automatically installing provider %s", pluginSpec.Name)
|
|
|
|
err = pluginSpec.Install(downloadedFile, false)
|
|
|
|
if err != nil {
|
2023-05-27 08:47:06 +00:00
|
|
|
return nil, &InstallPluginError{
|
2023-05-12 18:05:11 +00:00
|
|
|
Spec: pluginSpec,
|
|
|
|
Err: fmt.Errorf("error installing provider %s: %w", pluginSpec.Name, err),
|
2023-05-12 17:52:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-27 08:47:06 +00:00
|
|
|
return pluginSpec.Version, nil
|
2023-05-12 17:52:30 +00:00
|
|
|
}
|