mirror of https://github.com/pulumi/pulumi.git
380 lines
10 KiB
Go
380 lines
10 KiB
Go
// Copyright 2016-2022, 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 schema
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"sync"
|
|
|
|
"github.com/natefinch/atomic"
|
|
|
|
"github.com/blang/semver"
|
|
"github.com/segmentio/encoding/json"
|
|
|
|
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/plugin"
|
|
"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/workspace"
|
|
)
|
|
|
|
type Loader interface {
|
|
LoadPackage(pkg string, version *semver.Version) (*Package, error)
|
|
}
|
|
|
|
type ReferenceLoader interface {
|
|
Loader
|
|
|
|
LoadPackageReference(pkg string, version *semver.Version) (PackageReference, error)
|
|
}
|
|
|
|
type LoadSchemaParameterization struct {
|
|
Name tokens.Package
|
|
Version semver.Version
|
|
Value []byte
|
|
}
|
|
|
|
// LoadSchemaRequest is the domain model for codegenrpc.GetSchemaRequest.
|
|
type LoadSchemaRequest struct {
|
|
Package tokens.Package
|
|
Version *semver.Version
|
|
PluginDownloadURL string
|
|
Parameterization *LoadSchemaParameterization
|
|
}
|
|
|
|
// SchemaLoader that can load a package given the same package information that programs send to the engine for
|
|
// resource registrations. That is name, and version, but also pluginDownloadURL and parameterization.
|
|
type SchemaLoader interface { //nolint: revive
|
|
ReferenceLoader
|
|
|
|
LoadSchema(ctx context.Context, req *LoadSchemaRequest) (PackageReference, error)
|
|
}
|
|
|
|
type pluginLoader struct {
|
|
m sync.RWMutex
|
|
|
|
host plugin.Host
|
|
entries map[string]PackageReference
|
|
|
|
cacheOptions pluginLoaderCacheOptions
|
|
}
|
|
|
|
// Ensure this is a SchemaLoader
|
|
var _ SchemaLoader = &pluginLoader{}
|
|
|
|
// Caching options intended for benchmarking or debugging:
|
|
type pluginLoaderCacheOptions struct {
|
|
// useEntriesCache enables in-memory re-use of packages
|
|
disableEntryCache bool
|
|
// useFileCache enables skipping plugin loading when possible and caching JSON schemas to files
|
|
disableFileCache bool
|
|
// useMmap enables the use of memory mapped IO to avoid copying the JSON schema
|
|
disableMmap bool
|
|
}
|
|
|
|
func NewPluginLoader(host plugin.Host) SchemaLoader {
|
|
return &pluginLoader{
|
|
host: host,
|
|
entries: map[string]PackageReference{},
|
|
}
|
|
}
|
|
|
|
func newPluginLoaderWithOptions(host plugin.Host, cacheOptions pluginLoaderCacheOptions) SchemaLoader {
|
|
return &pluginLoader{
|
|
host: host,
|
|
entries: map[string]PackageReference{},
|
|
|
|
cacheOptions: cacheOptions,
|
|
}
|
|
}
|
|
|
|
func (l *pluginLoader) getPackage(key string) (PackageReference, bool) {
|
|
if l.cacheOptions.disableEntryCache {
|
|
return nil, false
|
|
}
|
|
p, ok := l.entries[key]
|
|
return p, ok
|
|
}
|
|
|
|
func (l *pluginLoader) setPackage(key string, p PackageReference) PackageReference {
|
|
if l.cacheOptions.disableEntryCache {
|
|
return p
|
|
}
|
|
|
|
if p, ok := l.entries[key]; ok {
|
|
return p
|
|
}
|
|
|
|
l.entries[key] = p
|
|
return p
|
|
}
|
|
|
|
func (l *pluginLoader) LoadPackage(pkg string, version *semver.Version) (*Package, error) {
|
|
ref, err := l.LoadPackageReference(pkg, version)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return ref.Definition()
|
|
}
|
|
|
|
var ErrGetSchemaNotImplemented = getSchemaNotImplemented{}
|
|
|
|
type getSchemaNotImplemented struct{}
|
|
|
|
func (f getSchemaNotImplemented) Error() string {
|
|
return "it looks like GetSchema is not implemented"
|
|
}
|
|
|
|
func schemaIsEmpty(schemaBytes []byte) bool {
|
|
// A non-empty schema is any that contains non-whitespace, non brace characters.
|
|
//
|
|
// Some providers implemented GetSchema initially by returning text matching the regular
|
|
// expression: "\s*\{\s*\}\s*". This handles those cases while not strictly checking that braces
|
|
// match or reading the whole document.
|
|
for _, v := range schemaBytes {
|
|
if v != ' ' && v != '\t' && v != '\r' && v != '\n' && v != '{' && v != '}' {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func (l *pluginLoader) LoadPackageReference(pkg string, version *semver.Version) (PackageReference, error) {
|
|
return l.LoadSchema(context.TODO(), &LoadSchemaRequest{
|
|
Package: tokens.Package(pkg),
|
|
Version: version,
|
|
})
|
|
}
|
|
|
|
func (l *pluginLoader) LoadSchema(ctx context.Context, req *LoadSchemaRequest) (PackageReference, error) {
|
|
if req.Package == "pulumi" {
|
|
return DefaultPulumiPackage.Reference(), nil
|
|
}
|
|
|
|
l.m.Lock()
|
|
defer l.m.Unlock()
|
|
|
|
key := packageIdentity(req.Package, req.Version)
|
|
if p, ok := l.getPackage(key); ok {
|
|
return p, nil
|
|
}
|
|
|
|
schemaBytes, version, err := l.loadSchemaBytes(ctx, req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if schemaIsEmpty(schemaBytes) {
|
|
return nil, getSchemaNotImplemented{}
|
|
}
|
|
|
|
var spec PartialPackageSpec
|
|
if _, err := json.Parse(schemaBytes, &spec, json.ZeroCopy); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Insert a version into the spec if the package does not provide one or if the
|
|
// existing version is less than the provided one
|
|
if version != nil {
|
|
setVersion := true
|
|
if spec.PackageInfoSpec.Version != "" {
|
|
vSemver, err := semver.Make(spec.PackageInfoSpec.Version)
|
|
if err == nil {
|
|
if vSemver.Compare(*version) == 1 {
|
|
setVersion = false
|
|
}
|
|
}
|
|
}
|
|
if setVersion {
|
|
spec.PackageInfoSpec.Version = version.String()
|
|
}
|
|
}
|
|
|
|
p, err := ImportPartialSpec(spec, nil, l)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return l.setPackage(key, p), nil
|
|
}
|
|
|
|
func LoadPackageReference(loader Loader, pkg string, version *semver.Version) (PackageReference, error) {
|
|
var ref PackageReference
|
|
var err error
|
|
if refLoader, ok := loader.(ReferenceLoader); ok {
|
|
ref, err = refLoader.LoadPackageReference(pkg, version)
|
|
} else {
|
|
p, pErr := loader.LoadPackage(pkg, version)
|
|
err = pErr
|
|
if err == nil {
|
|
ref = p.Reference()
|
|
}
|
|
}
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if pkg != ref.Name() || version != nil && ref.Version() != nil && !ref.Version().Equals(*version) {
|
|
if l, ok := loader.(*pluginLoader); ok {
|
|
return nil, fmt.Errorf("req: %s@%v: entries: %v (returned %s@%v)", pkg, version,
|
|
l.entries, ref.Name(), ref.Version())
|
|
}
|
|
return nil, fmt.Errorf("loader returned %s@%v: expected %s@%v", ref.Name(), ref.Version(), pkg, version)
|
|
}
|
|
|
|
return ref, nil
|
|
}
|
|
|
|
func (l *pluginLoader) loadSchemaBytes(
|
|
ctx context.Context, req *LoadSchemaRequest,
|
|
) ([]byte, *semver.Version, error) {
|
|
attachPort, err := plugin.GetProviderAttachPort(req.Package)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
// If PULUMI_DEBUG_PROVIDERS requested an attach port, skip caching and workspace
|
|
// interaction and load the schema directly from the given port.
|
|
if attachPort != nil {
|
|
schemaBytes, provider, err := l.loadPluginSchemaBytes(ctx, req)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("Error loading schema from plugin: %w", err)
|
|
}
|
|
|
|
if req.Version == nil {
|
|
info, err := provider.GetPluginInfo(ctx)
|
|
contract.IgnoreError(err) // nonfatal error
|
|
req.Version = info.Version
|
|
}
|
|
return schemaBytes, req.Version, nil
|
|
}
|
|
|
|
pluginInfo, err := l.host.ResolvePlugin(apitype.ResourcePlugin, req.Package.String(), req.Version)
|
|
if err != nil {
|
|
// Try and install the plugin if it was missing and try again, unless auto plugin installs are turned off.
|
|
var missingError *workspace.MissingError
|
|
if !errors.As(err, &missingError) || env.DisableAutomaticPluginAcquisition.Value() {
|
|
return nil, nil, err
|
|
}
|
|
|
|
spec := workspace.PluginSpec{
|
|
Kind: apitype.ResourcePlugin,
|
|
Name: req.Package.String(),
|
|
Version: req.Version,
|
|
}
|
|
|
|
log := func(sev diag.Severity, msg string) {
|
|
l.host.Log(sev, "", msg, 0)
|
|
}
|
|
|
|
_, err = pkgWorkspace.InstallPlugin(spec, log)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
pluginInfo, err = l.host.ResolvePlugin(apitype.ResourcePlugin, req.Package.String(), req.Version)
|
|
if err != nil {
|
|
return nil, req.Version, err
|
|
}
|
|
}
|
|
contract.Assertf(pluginInfo != nil, "loading pkg %q: pluginInfo was unexpectedly nil", req.Package)
|
|
|
|
if req.Version == nil {
|
|
req.Version = pluginInfo.Version
|
|
}
|
|
|
|
if pluginInfo.SchemaPath != "" && req.Version != nil {
|
|
schemaBytes, ok := l.loadCachedSchemaBytes(pluginInfo.SchemaPath, pluginInfo.SchemaTime)
|
|
if ok {
|
|
return schemaBytes, nil, nil
|
|
}
|
|
}
|
|
|
|
schemaBytes, provider, err := l.loadPluginSchemaBytes(ctx, req)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("Error loading schema from plugin: %w", err)
|
|
}
|
|
|
|
if pluginInfo.SchemaPath != "" {
|
|
err = atomic.WriteFile(pluginInfo.SchemaPath, bytes.NewReader(schemaBytes))
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("Error writing schema from plugin to cache: %w", err)
|
|
}
|
|
}
|
|
|
|
if req.Version == nil {
|
|
info, _ := provider.GetPluginInfo(ctx) // nonfatal error
|
|
req.Version = info.Version
|
|
}
|
|
|
|
return schemaBytes, req.Version, nil
|
|
}
|
|
|
|
func (l *pluginLoader) loadPluginSchemaBytes(
|
|
ctx context.Context, req *LoadSchemaRequest,
|
|
) ([]byte, plugin.Provider, error) {
|
|
provider, err := l.host.Provider(req.Package, req.Version)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
contract.Assertf(provider != nil, "unexpected nil provider for %s@%v", req.Package, req.Version)
|
|
|
|
var subpackageName tokens.Package
|
|
var subpackageVersion *semver.Version
|
|
if req.Parameterization != nil {
|
|
resp, err := provider.Parameterize(ctx, plugin.ParameterizeRequest{
|
|
Parameters: &plugin.ParameterizeValue{
|
|
Name: req.Parameterization.Name,
|
|
Version: req.Parameterization.Version,
|
|
Value: req.Parameterization.Value,
|
|
},
|
|
})
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("parameterizing provider: %w", err)
|
|
}
|
|
|
|
if resp.Name != req.Parameterization.Name {
|
|
return nil, nil, fmt.Errorf(
|
|
"unexpected parameterization response, expected name %s got %s",
|
|
req.Parameterization.Name, resp.Name)
|
|
}
|
|
if !resp.Version.EQ(req.Parameterization.Version) {
|
|
return nil, nil, fmt.Errorf(
|
|
"unexpected parameterization response, expected version %v got %v",
|
|
req.Parameterization.Version, resp.Version)
|
|
}
|
|
|
|
subpackageName = resp.Name
|
|
subpackageVersion = &resp.Version
|
|
}
|
|
|
|
schemaFormatVersion := 0
|
|
schema, err := provider.GetSchema(ctx, plugin.GetSchemaRequest{
|
|
Version: schemaFormatVersion,
|
|
SubpackageName: subpackageName,
|
|
SubpackageVersion: subpackageVersion,
|
|
})
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
return schema.Schema, provider, nil
|
|
}
|