pulumi/sdk/go/common/workspace/plugins_test.go

1311 lines
39 KiB
Go

// Copyright 2016-2019, 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 (
"bytes"
"encoding/hex"
"errors"
"fmt"
"io"
"net/http"
"net/http/httptest"
"os"
"path/filepath"
"regexp"
"testing"
"time"
"github.com/blang/semver"
"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/testing/diagtest"
"github.com/pulumi/pulumi/sdk/v3/go/common/testing/iotest"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
func TestLegacyPluginSelection_Prerelease(t *testing.T) {
t.Parallel()
v1 := semver.MustParse("0.1.0")
v2 := semver.MustParse("0.2.0")
v3 := semver.MustParse("0.3.0-alpha")
candidatePlugins := []PluginInfo{
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v2,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "notmyplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "myplugin",
Kind: apitype.AnalyzerPlugin,
Version: &v3,
},
}
result := LegacySelectCompatiblePlugin(candidatePlugins, apitype.ResourcePlugin, "myplugin", nil)
assert.NotNil(t, result)
assert.Equal(t, "myplugin", result.Name)
assert.Equal(t, "0.2.0", result.Version.String())
}
func TestLegacyPluginSelection_PrereleaseRequested(t *testing.T) {
t.Parallel()
v1 := semver.MustParse("0.1.0")
v2 := semver.MustParse("0.2.0-alpha")
v3 := semver.MustParse("0.3.0-alpha")
candidatePlugins := []PluginInfo{
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v2,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "notmyplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "myplugin",
Kind: apitype.AnalyzerPlugin,
Version: &v3,
},
}
v := semver.MustParse("0.2.0")
result := LegacySelectCompatiblePlugin(candidatePlugins, apitype.ResourcePlugin, "myplugin", &v)
assert.NotNil(t, result)
assert.Equal(t, "myplugin", result.Name)
assert.Equal(t, "0.3.0-alpha", result.Version.String())
}
func TestPluginSelection_ExactMatch(t *testing.T) {
t.Parallel()
v1 := semver.MustParse("0.1.0")
v2 := semver.MustParse("0.2.0")
v3 := semver.MustParse("0.3.0")
candidatePlugins := []PluginInfo{
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v2,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "notmyplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "myplugin",
Kind: apitype.AnalyzerPlugin,
Version: &v3,
},
}
requested := semver.MustParseRange("0.2.0")
result := SelectCompatiblePlugin(candidatePlugins, apitype.ResourcePlugin, "myplugin", requested)
assert.NotNil(t, result)
assert.Equal(t, "myplugin", result.Name)
assert.Equal(t, "0.2.0", result.Version.String())
}
func TestPluginSelection_ExactMatchNotFound(t *testing.T) {
t.Parallel()
v1 := semver.MustParse("0.1.0")
v2 := semver.MustParse("0.2.1")
v3 := semver.MustParse("0.3.0")
candidatePlugins := []PluginInfo{
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v2,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "notmyplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "myplugin",
Kind: apitype.AnalyzerPlugin,
Version: &v3,
},
}
requested := semver.MustParseRange("0.2.0")
result := SelectCompatiblePlugin(candidatePlugins, apitype.ResourcePlugin, "myplugin", requested)
assert.Nil(t, result)
}
func TestPluginSelection_PatchVersionSlide(t *testing.T) {
t.Parallel()
v1 := semver.MustParse("0.1.0")
v2 := semver.MustParse("0.2.0")
v21 := semver.MustParse("0.2.1")
v3 := semver.MustParse("0.3.0")
candidatePlugins := []PluginInfo{
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v2,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v21,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "notmyplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "myplugin",
Kind: apitype.AnalyzerPlugin,
Version: &v3,
},
}
requested := semver.MustParseRange(">=0.2.0 <0.3.0")
result := SelectCompatiblePlugin(candidatePlugins, apitype.ResourcePlugin, "myplugin", requested)
assert.NotNil(t, result)
assert.Equal(t, "myplugin", result.Name)
assert.Equal(t, "0.2.1", result.Version.String())
}
func TestPluginSelection_EmptyVersionNoAlternatives(t *testing.T) {
t.Parallel()
v1 := semver.MustParse("0.1.0")
v2 := semver.MustParse("0.2.1")
v3 := semver.MustParse("0.3.0")
candidatePlugins := []PluginInfo{
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v2,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: nil,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "notmyplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "myplugin",
Kind: apitype.AnalyzerPlugin,
Version: &v3,
},
}
requested := semver.MustParseRange("0.2.0")
result := SelectCompatiblePlugin(candidatePlugins, apitype.ResourcePlugin, "myplugin", requested)
assert.NotNil(t, result)
assert.Equal(t, "myplugin", result.Name)
assert.Nil(t, result.Version)
}
func TestPluginSelection_EmptyVersionWithAlternatives(t *testing.T) {
t.Parallel()
v1 := semver.MustParse("0.1.0")
v2 := semver.MustParse("0.2.0")
v3 := semver.MustParse("0.3.0")
candidatePlugins := []PluginInfo{
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v2,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: nil,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: nil,
},
{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "notmyplugin",
Kind: apitype.ResourcePlugin,
Version: &v3,
},
{
Name: "myplugin",
Kind: apitype.AnalyzerPlugin,
Version: &v3,
},
}
requested := semver.MustParseRange("0.2.0")
result := SelectCompatiblePlugin(candidatePlugins, apitype.ResourcePlugin, "myplugin", requested)
assert.NotNil(t, result)
assert.Equal(t, "myplugin", result.Name)
assert.Equal(t, "0.2.0", result.Version.String())
}
func newMockReadCloser(data []byte) (io.ReadCloser, int64, error) {
return io.NopCloser(bytes.NewReader(data)), int64(len(data)), nil
}
func newMockReadCloserString(data string) (io.ReadCloser, int64, error) {
return newMockReadCloser([]byte(data))
}
//nolint:paralleltest // mutates environment variables
func TestPluginDownload(t *testing.T) {
expectedBytes := []byte{1, 2, 3}
token := "RaNd0m70K3n_"
t.Run("Pulumi GitHub Releases", func(t *testing.T) {
t.Setenv("GITHUB_TOKEN", "")
version := semver.MustParse("4.30.0")
spec := PluginSpec{
PluginDownloadURL: "",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
if req.URL.String() == "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/tags/v4.30.0" {
assert.Equal(t, "", req.Header.Get("Authorization"))
assert.Equal(t, "application/json", req.Header.Get("Accept"))
// Minimal JSON from the releases API to get the test to pass
return newMockReadCloserString(`{
"assets": [
{
"url": "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/654321",
"name": "pulumi-mockdl_4.30.0_checksums.txt"
},
{
"url": "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/123456",
"name": "pulumi-resource-mockdl-v4.30.0-darwin-amd64.tar.gz"
}
]
}
`)
}
assert.Equal(t, "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/123456", req.URL.String())
assert.Equal(t, "application/octet-stream", req.Header.Get("Accept"))
return newMockReadCloser(expectedBytes)
}
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
t.Run("get.pulumi.com", func(t *testing.T) {
version := semver.MustParse("4.32.0")
spec := PluginSpec{
PluginDownloadURL: "",
Name: "otherdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
// Test that the asset isn't on github
if req.URL.String() == "https://api.github.com/repos/pulumi/pulumi-otherdl/releases/tags/v4.32.0" {
return nil, -1, errors.New("404 not found")
}
assert.Equal(t,
"https://get.pulumi.com/releases/plugins/pulumi-resource-otherdl-v4.32.0-darwin-amd64.tar.gz",
req.URL.String())
return newMockReadCloser(expectedBytes)
}
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
t.Run("Custom http URL", func(t *testing.T) {
version := semver.MustParse("4.32.0")
spec := PluginSpec{
PluginDownloadURL: "http://customurl.jfrog.io/artifactory/pulumi-packages/package-name/v${VERSION}/${OS}/${ARCH}",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
assert.Equal(t,
"http://customurl.jfrog.io/artifactory/pulumi-packages/"+
"package-name/v4.32.0/darwin/amd64/pulumi-resource-mockdl-v4.32.0-darwin-amd64.tar.gz",
req.URL.String())
return newMockReadCloser(expectedBytes)
}
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
t.Run("Custom https URL", func(t *testing.T) {
version := semver.MustParse("4.32.0")
spec := PluginSpec{
PluginDownloadURL: "https://customurl.jfrog.io/artifactory/pulumi-packages/" +
"package-name/${NAME}/v${VERSION}/${OS}/${ARCH}/",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
assert.Equal(t,
"https://customurl.jfrog.io/artifactory/pulumi-packages/"+
"package-name/mockdl/v4.32.0/darwin/amd64/pulumi-resource-mockdl-v4.32.0-darwin-amd64.tar.gz",
req.URL.String())
return newMockReadCloser(expectedBytes)
}
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
t.Run("Private Pulumi GitHub Releases", func(t *testing.T) {
t.Setenv("GITHUB_TOKEN", token)
version := semver.MustParse("4.32.0")
spec := PluginSpec{
PluginDownloadURL: "",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
if req.URL.String() == "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/tags/v4.32.0" {
assert.Equal(t, "token "+token, req.Header.Get("Authorization"))
assert.Equal(t, "application/json", req.Header.Get("Accept"))
// Minimal JSON from the releases API to get the test to pass
return newMockReadCloserString(`{
"assets": [
{
"url": "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/654321",
"name": "pulumi-mockdl_4.32.0_checksums.txt"
},
{
"url": "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/123456",
"name": "pulumi-resource-mockdl-v4.32.0-darwin-amd64.tar.gz"
}
]
}
`)
}
assert.Equal(t, "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/123456", req.URL.String())
assert.Equal(t, "token "+token, req.Header.Get("Authorization"))
assert.Equal(t, "application/octet-stream", req.Header.Get("Accept"))
return newMockReadCloser(expectedBytes)
}
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
t.Run("Internal GitHub Releases", func(t *testing.T) {
t.Setenv("GITHUB_TOKEN", token)
version := semver.MustParse("4.32.0")
spec := PluginSpec{
PluginDownloadURL: "github://api.git.org/ourorg/mock",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
// Test that the asset isn't on github
if req.URL.String() == "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/tags/v4.32.0" {
return nil, -1, errors.New("404 not found")
}
if req.URL.String() == "https://api.git.org/repos/ourorg/mock/releases/tags/v4.32.0" {
assert.Equal(t, "token "+token, req.Header.Get("Authorization"))
assert.Equal(t, "application/json", req.Header.Get("Accept"))
// Minimal JSON from the releases API to get the test to pass
return newMockReadCloserString(`{
"assets": [
{
"url": "https://api.git.org/repos/ourorg/mock/releases/assets/654321",
"name": "pulumi-mockdl_4.32.0_checksums.txt"
},
{
"url": "https://api.git.org/repos/ourorg/mock/releases/assets/123456",
"name": "pulumi-resource-mockdl-v4.32.0-darwin-amd64.tar.gz"
}
]
}
`)
}
assert.Equal(t, "https://api.git.org/repos/ourorg/mock/releases/assets/123456", req.URL.String())
assert.Equal(t, "token "+token, req.Header.Get("Authorization"))
assert.Equal(t, "application/octet-stream", req.Header.Get("Accept"))
return newMockReadCloser(expectedBytes)
}
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
t.Run("Pulumi GitHub Releases With Checksum", func(t *testing.T) {
t.Setenv("GITHUB_TOKEN", "")
version := semver.MustParse("4.30.0")
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
if req.URL.String() == "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/tags/v4.30.0" {
assert.Equal(t, "", req.Header.Get("Authorization"))
assert.Equal(t, "application/json", req.Header.Get("Accept"))
// Minimal JSON from the releases API to get the test to pass
return newMockReadCloserString(`{
"assets": [
{
"url": "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/654321",
"name": "pulumi-mockdl_4.30.0_checksums.txt"
},
{
"url": "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/123456",
"name": "pulumi-resource-mockdl-v4.30.0-darwin-amd64.tar.gz"
}
]
}
`)
}
assert.Equal(t, "https://api.github.com/repos/pulumi/pulumi-mockdl/releases/assets/123456", req.URL.String())
assert.Equal(t, "application/octet-stream", req.Header.Get("Accept"))
return newMockReadCloser(expectedBytes)
}
chksum := "039058c6f2c0cb492c533b0a4d14ef77cc0f78abccced5287d84a1a2011cfb81" //nolint:gosec
t.Run("Invalid Checksum", func(t *testing.T) {
spec := PluginSpec{
PluginDownloadURL: "",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
Checksums: map[string][]byte{
"darwin-amd64": {0},
},
}
source, err := spec.GetSource()
require.NoError(t, err)
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
assert.Error(t, err, "invalid checksum, expected 00, actual "+chksum)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
t.Run("Valid Checksum", func(t *testing.T) {
checksum, err := hex.DecodeString(chksum)
assert.NoError(t, err)
spec := PluginSpec{
PluginDownloadURL: "",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
Checksums: map[string][]byte{
"darwin-amd64": checksum,
},
}
source, err := spec.GetSource()
require.NoError(t, err)
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
t.Run("Missing Checksum", func(t *testing.T) {
// In this test the specification has checksums, but is missing the checksum for the current platform.
// There are two sensible ways to handle this:
// 1. Behave as if no checksums were specified at all, and simply fall back to not checking anything.
// 2. Error that the checksum for the current platform is missing.
// We choose to do the former, for now as that's more lenient.
spec := PluginSpec{
PluginDownloadURL: "",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
Checksums: map[string][]byte{
"windows-amd64": {0},
},
}
source, err := spec.GetSource()
require.NoError(t, err)
r, l, err := source.Download(*spec.Version, "darwin", "amd64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
})
t.Run("GitLab Releases", func(t *testing.T) {
t.Setenv("GITLAB_TOKEN", token)
version := semver.MustParse("1.23.4")
spec := PluginSpec{
PluginDownloadURL: "gitlab://gitlab.com/278964",
Name: "mock-gitlab",
Version: &version,
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
assert.Equal(t,
"https://gitlab.com/api/v4/projects/278964/releases/v1.23.4/downloads/"+
"pulumi-resource-mock-gitlab-v1.23.4-windows-arm64.tar.gz", req.URL.String())
assert.Equal(t, "Bearer "+token, req.Header.Get("Authorization"))
assert.Equal(t, "application/octet-stream", req.Header.Get("Accept"))
return newMockReadCloser(expectedBytes)
}
r, l, err := source.Download(*spec.Version, "windows", "arm64", getHTTPResponse)
require.NoError(t, err)
readBytes, err := io.ReadAll(r)
require.NoError(t, err)
assert.Equal(t, int(l), len(readBytes))
assert.Equal(t, expectedBytes, readBytes)
})
}
//nolint:paralleltest // mutates environment variables
func TestPluginGetLatestVersion(t *testing.T) {
token := "RaNd0m70K3n_"
t.Run("Pulumi GitHub Releases", func(t *testing.T) {
t.Setenv("GITHUB_TOKEN", "")
spec := PluginSpec{
PluginDownloadURL: "",
Name: "mock-latest",
Kind: apitype.PluginKind("resource"),
}
expectedVersion := semver.MustParse("4.37.5")
source, err := spec.GetSource()
assert.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
assert.Equal(t,
"https://api.github.com/repos/pulumi/pulumi-mock-latest/releases/latest",
req.URL.String())
// Minimal JSON from the releases API to get the test to pass
return newMockReadCloserString(`{
"tag_name": "v4.37.5"
}`)
}
version, err := source.GetLatestVersion(getHTTPResponse)
require.NoError(t, err)
assert.Equal(t, expectedVersion, *version)
})
t.Run("Custom http URL", func(t *testing.T) {
spec := PluginSpec{
PluginDownloadURL: "http://customurl.jfrog.io/artifactory/pulumi-packages/package-name",
Name: "mock-latest",
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
version, err := source.GetLatestVersion(getHTTPResponse)
assert.Nil(t, version)
assert.EqualError(t, err, "GetLatestVersion is not supported for plugins from http sources")
})
t.Run("Custom https URL", func(t *testing.T) {
spec := PluginSpec{
PluginDownloadURL: "https://customurl.jfrog.io/artifactory/pulumi-packages/package-name",
Name: "mock-latest",
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
require.NoError(t, err)
version, err := source.GetLatestVersion(getHTTPResponse)
assert.Nil(t, version)
assert.EqualError(t, err, "GetLatestVersion is not supported for plugins from http sources")
})
t.Run("Private Pulumi GitHub Releases", func(t *testing.T) {
t.Setenv("GITHUB_TOKEN", token)
spec := PluginSpec{
PluginDownloadURL: "",
Name: "mock-private",
Kind: apitype.PluginKind("resource"),
}
expectedVersion := semver.MustParse("4.37.5")
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
if req.URL.String() == "https://api.github.com/repos/pulumi/pulumi-mock-private/releases/latest" {
assert.Equal(t, "token "+token, req.Header.Get("Authorization"))
assert.Equal(t, "application/json", req.Header.Get("Accept"))
// Minimal JSON from the releases API to get the test to pass
return newMockReadCloserString(`{
"tag_name": "v4.37.5"
}`)
}
panic("Unexpected call to getHTTPResponse")
}
version, err := source.GetLatestVersion(getHTTPResponse)
require.NoError(t, err)
assert.Equal(t, expectedVersion, *version)
})
t.Run("Internal GitHub Releases", func(t *testing.T) {
t.Setenv("GITHUB_TOKEN", token)
spec := PluginSpec{
PluginDownloadURL: "github://api.git.org/ourorg/mock",
Name: "mock-private",
Kind: apitype.PluginKind("resource"),
}
expectedVersion := semver.MustParse("4.37.5")
source, err := spec.GetSource()
assert.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
if req.URL.String() == "https://api.git.org/repos/ourorg/mock/releases/latest" {
assert.Equal(t, "token "+token, req.Header.Get("Authorization"))
assert.Equal(t, "application/json", req.Header.Get("Accept"))
// Minimal JSON from the releases API to get the test to pass
return newMockReadCloserString(`{
"tag_name": "v4.37.5"
}`)
}
panic("Unexpected call to getHTTPResponse")
}
version, err := source.GetLatestVersion(getHTTPResponse)
require.NoError(t, err)
assert.Equal(t, expectedVersion, *version)
})
t.Run("GitLab Releases", func(t *testing.T) {
t.Setenv("GITLAB_TOKEN", token)
spec := PluginSpec{
PluginDownloadURL: "gitlab://gitlab.com/278964",
Name: "mock-gitlab",
Kind: apitype.PluginKind("resource"),
}
expectedVersion := semver.MustParse("1.23.0")
source, err := spec.GetSource()
require.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
if req.URL.String() == "https://gitlab.com/api/v4/projects/278964/releases/permalink/latest" {
assert.Equal(t, "Bearer "+token, req.Header.Get("Authorization"))
assert.Equal(t, "application/json", req.Header.Get("Accept"))
// Minimal JSON from the releases API to get the test to pass
return newMockReadCloserString(`{
"tag_name": "v1.23"
}`)
}
panic("Unexpected call to getHTTPResponse")
}
version, err := source.GetLatestVersion(getHTTPResponse)
require.NoError(t, err)
assert.Equal(t, expectedVersion, *version)
})
t.Run("Hit GitHub ratelimit", func(t *testing.T) {
t.Setenv("GITHUB_TOKEN", "")
spec := PluginSpec{
PluginDownloadURL: "",
Name: "mock-latest",
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
assert.NoError(t, err)
getHTTPResponse := func(req *http.Request) (io.ReadCloser, int64, error) {
return nil, 0, newDownloadError(403, req.URL, http.Header{"X-Ratelimit-Remaining": []string{"0"}})
}
_, err = source.GetLatestVersion(getHTTPResponse)
assert.ErrorContains(t, err, "rate limit exceeded")
assert.ErrorContains(t, err, "https://api.github.com/repos/pulumi/pulumi-mock-latest/releases/latest")
})
}
func TestParsePluginDownloadURLOverride(t *testing.T) {
t.Parallel()
type match struct {
name string
url string
ok bool
}
tests := []struct {
input string
expected pluginDownloadOverrideArray
matches []match
expectedError string
}{
{
input: "",
expected: pluginDownloadOverrideArray{},
},
{
input: "^foo.*=https://foo",
expected: pluginDownloadOverrideArray{
{
reg: regexp.MustCompile("^foo.*"),
url: "https://foo",
},
},
matches: []match{
{
name: "foo",
url: "https://foo",
ok: true,
},
{
name: "foo-bar",
url: "https://foo",
ok: true,
},
{
name: "fo",
url: "",
ok: false,
},
{
name: "",
url: "",
ok: false,
},
{
name: "nope",
url: "",
ok: false,
},
},
},
{
input: "^foo.*=https://foo,^bar.*=https://bar",
expected: pluginDownloadOverrideArray{
{
reg: regexp.MustCompile("^foo.*"),
url: "https://foo",
},
{
reg: regexp.MustCompile("^bar.*"),
url: "https://bar",
},
},
matches: []match{
{
name: "foo",
url: "https://foo",
ok: true,
},
{
name: "foo-bar",
url: "https://foo",
ok: true,
},
{
name: "fo",
url: "",
ok: false,
},
{
name: "",
url: "",
ok: false,
},
{
name: "bar",
url: "https://bar",
ok: true,
},
{
name: "barbaz",
url: "https://bar",
ok: true,
},
{
name: "ba",
url: "",
ok: false,
},
{
name: "nope",
url: "",
ok: false,
},
},
},
{
input: "=", // missing regex and url
expectedError: "expected format to be \"regexp1=URL1,regexp2=URL2\"; got \"=\"",
},
{
input: "^foo.*=", // missing url
expectedError: "expected format to be \"regexp1=URL1,regexp2=URL2\"; got \"^foo.*=\"",
},
{
input: "=https://foo", // missing regex
expectedError: "expected format to be \"regexp1=URL1,regexp2=URL2\"; got \"=https://foo\"",
},
{
input: "^foo.*=https://foo,", // trailing comma
expectedError: "expected format to be \"regexp1=URL1,regexp2=URL2\"; got \"^foo.*=https://foo,\"",
},
{
input: "[=https://foo", // invalid regex
expectedError: "error parsing regexp: missing closing ]: `[`",
},
}
for _, tt := range tests {
tt := tt
t.Run(tt.input, func(t *testing.T) {
t.Parallel()
actual, err := parsePluginDownloadURLOverrides(tt.input)
if tt.expectedError != "" {
assert.EqualError(t, err, tt.expectedError)
} else {
assert.NoError(t, err)
}
assert.Equal(t, tt.expected, actual)
if len(tt.matches) > 0 {
for _, match := range tt.matches {
actualURL, actualOK := actual.get(match.name)
assert.Equal(t, match.url, actualURL)
assert.Equal(t, match.ok, actualOK)
}
}
})
}
}
func TestDownloadToFile_retries(t *testing.T) {
t.Parallel()
// Verifies that DownloadToFile retries on transient errors
// when trying to download plugins,
// and that it calls the wrapper and retry functions as expected.
//
// Regression test for https://github.com/pulumi/pulumi/issues/12456.
var numRequests int
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
assert.Equal(t, http.MethodGet, r.Method, "expected GET request")
assert.Regexp(t, `/pulumi-language-myplugin-v1.0.0-\S+\.tar\.gz`, r.URL.Path,
"unexpected URL path")
// Fails all requests with a 500 error.
// This will cause every download attempt to fail
// and be retried.
w.WriteHeader(http.StatusInternalServerError)
numRequests++
}))
t.Cleanup(server.Close)
defer func() {
assert.Equal(t, 5, numRequests,
"server received more requests than expected")
}()
// Create a fake plugin.
version := semver.MustParse("1.0.0")
spec := PluginSpec{
Name: "myplugin",
Kind: apitype.LanguagePlugin,
Version: &version,
PluginDownloadURL: server.URL,
PluginDir: t.TempDir(),
}
// numRetries is tracked separately from numRequests.
// numRequests is the number of requests received by the server,
// while numRetries is the number of times the retry function is called.
// These should match--the function is called on all failures.
var numRetries int
currentTime := time.Now()
_, err := (&pluginDownloader{
OnRetry: func(err error, attempt, limit int, delay time.Duration) {
assert.Equal(t, 5, limit, "unexpected retry limit")
numRetries++
assert.Equal(t, numRetries, attempt, "unexpected attempt number")
},
After: func(d time.Duration) <-chan time.Time {
currentTime = currentTime.Add(d)
ch := make(chan time.Time, 1)
ch <- currentTime
return ch
},
}).DownloadToFile(spec)
assert.ErrorContains(t, err, "failed to download plugin: myplugin-1.0.0")
assert.Equal(t, numRequests, numRetries)
}
//nolint:paralleltest // changes directory for process
func TestUnmarshalProjectWithProviderList(t *testing.T) {
t.Parallel()
tempdir := t.TempDir()
pyaml := filepath.Join(tempdir, "Pulumi.yaml")
// write to pyaml
err := os.WriteFile(pyaml, []byte(`name: test-yaml
runtime: yaml
description: "Test Pulumi YAML"
plugins:
providers:
- name: aws
version: 1.0.0
path: ../bin/aws`), 0o600)
assert.NoError(t, err)
proj, err := LoadProject(pyaml)
assert.NoError(t, err)
assert.NotNil(t, proj.Plugins)
assert.Equal(t, 1, len(proj.Plugins.Providers))
assert.Equal(t, "aws", proj.Plugins.Providers[0].Name)
assert.Equal(t, "1.0.0", proj.Plugins.Providers[0].Version)
assert.Equal(t, "../bin/aws", proj.Plugins.Providers[0].Path)
}
func TestPluginBadSource(t *testing.T) {
t.Parallel()
version := semver.MustParse("4.30.0")
spec := PluginSpec{
PluginDownloadURL: "strange-scheme://what.is.this?oh-no",
Name: "mockdl",
Version: &version,
Kind: apitype.PluginKind("resource"),
}
source, err := spec.GetSource()
assert.ErrorContains(t, err, "unknown plugin source scheme: strange-scheme")
assert.Nil(t, source)
}
func TestMissingErrorText(t *testing.T) {
t.Parallel()
v1 := semver.MustParse("0.1.0")
tests := []struct {
Name string
Plugin PluginInfo
IncludeAmbient bool
ExpectedError string
}{
{
Name: "ResourceWithVersion",
Plugin: PluginInfo{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
IncludeAmbient: true,
ExpectedError: "no resource plugin 'pulumi-resource-myplugin' found in the workspace at version v0.1.0 " +
"or on your $PATH",
},
{
Name: "ResourceWithVersion_ExcludeAmbient",
Plugin: PluginInfo{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: &v1,
},
IncludeAmbient: false,
ExpectedError: "no resource plugin 'pulumi-resource-myplugin' found in the workspace at version v0.1.0",
},
{
Name: "ResourceWithoutVersion",
Plugin: PluginInfo{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: nil,
},
IncludeAmbient: true,
ExpectedError: "no resource plugin 'pulumi-resource-myplugin' found in the workspace or on your $PATH",
},
{
Name: "ResourceWithoutVersion_ExcludeAmbient",
Plugin: PluginInfo{
Name: "myplugin",
Kind: apitype.ResourcePlugin,
Version: nil,
},
IncludeAmbient: false,
ExpectedError: "no resource plugin 'pulumi-resource-myplugin' found in the workspace",
},
{
Name: "LanguageWithoutVersion",
Plugin: PluginInfo{
Name: "dotnet",
Kind: apitype.LanguagePlugin,
Version: nil,
},
IncludeAmbient: true,
ExpectedError: "no language plugin 'pulumi-language-dotnet' found in the workspace or on your $PATH",
},
}
for _, tt := range tests {
tt := tt
t.Run(tt.Name, func(t *testing.T) {
t.Parallel()
err := NewMissingError(tt.Plugin.Kind, tt.Plugin.Name, tt.Plugin.Version, tt.IncludeAmbient)
assert.EqualError(t, err, tt.ExpectedError)
})
}
}
//nolint:paralleltest // modifies environment variables
func TestBundledPluginSearch(t *testing.T) {
// Get the path of this executable
exe, err := os.Executable()
require.NoError(t, err)
// Create a fake side-by-side plugin next to this executable, it must match one of our bundled names
bundledPath := filepath.Join(filepath.Dir(exe), "pulumi-language-nodejs")
err = os.WriteFile(bundledPath, []byte{}, 0o700) //nolint: gosec // we intended to write an executable file here
require.NoError(t, err)
bundledPath, _ = filepath.EvalSymlinks(bundledPath)
t.Cleanup(func() {
err := os.Remove(bundledPath)
require.NoError(t, err)
})
// Create another copy of the fake plugin in $PATH
pathDir := t.TempDir()
t.Setenv("PATH", pathDir)
ambientPath := filepath.Join(pathDir, "pulumi-language-nodejs")
err = os.WriteFile(ambientPath, []byte{}, 0o700) //nolint: gosec
require.NoError(t, err)
d := diagtest.LogSink(t)
// Lookup the plugin with ambient search turned on
t.Setenv("PULUMI_IGNORE_AMBIENT_PLUGINS", "false")
path, err := GetPluginPath(d, apitype.LanguagePlugin, "nodejs", nil, nil)
require.NoError(t, err)
assert.Equal(t, ambientPath, path)
// Lookup the plugin with ambient search turned off
t.Setenv("PULUMI_IGNORE_AMBIENT_PLUGINS", "true")
path, err = GetPluginPath(d, apitype.LanguagePlugin, "nodejs", nil, nil)
require.NoError(t, err)
assert.Equal(t, bundledPath, path)
}
//nolint:paralleltest // modifies environment variables
func TestAmbientPluginsWarn(t *testing.T) {
// Create a fake plugin in the path
pathDir := t.TempDir()
t.Setenv("PATH", pathDir)
ambientPath := filepath.Join(pathDir, "pulumi-resource-mock")
err := os.WriteFile(ambientPath, []byte{}, 0o700) //nolint: gosec
require.NoError(t, err)
var stderr bytes.Buffer
d := diag.DefaultSink(
iotest.LogWriter(t), // stdout
&stderr,
diag.FormatOptions{Color: "never"},
)
// Lookup the plugin with ambient search turned on
t.Setenv("PULUMI_IGNORE_AMBIENT_PLUGINS", "false")
path, err := GetPluginPath(d, apitype.ResourcePlugin, "mock", nil, nil)
require.NoError(t, err)
assert.Equal(t, ambientPath, path)
// Check we get a warning about loading this plugin
expectedMessage := fmt.Sprintf("warning: using pulumi-resource-mock from $PATH at %s\n", ambientPath)
assert.Equal(t, expectedMessage, stderr.String())
}
//nolint:paralleltest // modifies environment variables
func TestBundledPluginsDoNotWarn(t *testing.T) {
// Get the path of this executable
exe, err := os.Executable()
require.NoError(t, err)
// Create a fake side-by-side plugin next to this executable, it must match one of our bundled names
bundledPath := filepath.Join(filepath.Dir(exe), "pulumi-language-nodejs")
err = os.WriteFile(bundledPath, []byte{}, 0o700) //nolint: gosec // we intended to write an executable file here
require.NoError(t, err)
t.Cleanup(func() {
err := os.Remove(bundledPath)
require.NoError(t, err)
})
// Add the executable directory to PATH
t.Setenv("PATH", filepath.Dir(exe))
var stderr bytes.Buffer
d := diag.DefaultSink(
iotest.LogWriter(t), // stdout
&stderr,
diag.FormatOptions{Color: "never"},
)
// Lookup the plugin with ambient search turned on
t.Setenv("PULUMI_IGNORE_AMBIENT_PLUGINS", "false")
path, err := GetPluginPath(d, apitype.LanguagePlugin, "nodejs", nil, nil)
require.NoError(t, err)
assert.Equal(t, bundledPath, path)
// Check we don't get a warning about loading this plugin, because it's the bundled one _even_ though it's also on PATH
assert.Empty(t, stderr.String())
}
// Regression test for https://github.com/pulumi/pulumi/issues/13656
//
//nolint:paralleltest // modifies environment variables
func TestSymlinkPathPluginsDoNotWarn(t *testing.T) {
// Get the path of this executable
exe, err := os.Executable()
require.NoError(t, err)
// Create a fake side-by-side plugin next to this executable, it must match one of our bundled names
bundledPath := filepath.Join(filepath.Dir(exe), "pulumi-language-nodejs")
err = os.WriteFile(bundledPath, []byte{}, 0o700) //nolint: gosec
require.NoError(t, err)
t.Cleanup(func() {
err := os.Remove(bundledPath)
require.NoError(t, err)
})
// Create a fake plugin in the path that is a symlink to the bundled plugin
pathDir := t.TempDir()
t.Setenv("PATH", pathDir)
ambientPath := filepath.Join(pathDir, "pulumi-language-nodejs")
err = os.Symlink(bundledPath, ambientPath)
require.NoError(t, err)
var stderr bytes.Buffer
d := diag.DefaultSink(
iotest.LogWriter(t), // stdout
&stderr,
diag.FormatOptions{Color: "never"},
)
// Lookup the plugin with ambient search turned on
t.Setenv("PULUMI_IGNORE_AMBIENT_PLUGINS", "false")
path, err := GetPluginPath(d, apitype.LanguagePlugin, "nodejs", nil, nil)
require.NoError(t, err)
// We expect the ambient path to be returned, but not to warn because it resolves to the same file as the
// bundled path.
assert.Equal(t, ambientPath, path)
assert.Empty(t, stderr.String())
}