pulumi/pkg/resource/deploy/source_query_test.go

874 lines
23 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 deploy
import (
"context"
"errors"
"fmt"
"io"
"sync"
"testing"
"github.com/blang/semver"
pbempty "github.com/golang/protobuf/ptypes/empty"
"github.com/hashicorp/hcl/v2"
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/providers"
"github.com/pulumi/pulumi/sdk/v3/go/common/diag"
"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/resource/plugin"
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/result"
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
pulumirpc "github.com/pulumi/pulumi/sdk/v3/proto/go"
"github.com/stretchr/testify/assert"
)
func TestQuerySource_Trivial_Wait(t *testing.T) {
t.Parallel()
// Trivial querySource returns immediately with `Wait()`, even with multiple invocations.
// Success case.
var called1 bool
resmon1 := mockResmon{
CancelF: func() error {
called1 = true
return nil
},
}
qs1, _ := newTestQuerySource(&resmon1, func(*querySource) error {
return nil
})
qs1.forkRun()
err := qs1.Wait()
assert.NoError(t, err)
assert.False(t, called1)
// Can be called twice.
err = qs1.Wait()
assert.NoError(t, err)
// Failure case.
var called2 bool
resmon2 := mockResmon{
CancelF: func() error {
called2 = true
return nil
},
}
qs2, _ := newTestQuerySource(&resmon2, func(*querySource) error {
return errors.New("failed")
})
qs2.forkRun()
err = qs2.Wait()
assert.False(t, result.IsBail(err))
assert.Error(t, err)
assert.False(t, called2)
// Can be called twice.
err = qs2.Wait()
assert.False(t, result.IsBail(err))
assert.Error(t, err)
assert.False(t, called2)
}
func TestQuerySource_Async_Wait(t *testing.T) {
t.Parallel()
// `Wait()` executes asynchronously.
// Success case.
//
// test blocks until querySource signals execution has started
// -> querySource blocks until test acknowledges querySource's signal
// -> test blocks on `Wait()` until querySource completes.
var called1 bool
resmon1 := mockResmon{
CancelF: func() error {
called1 = true
return nil
},
}
qs1Start, qs1StartAck := make(chan interface{}), make(chan interface{})
qs1, _ := newTestQuerySource(&resmon1, func(*querySource) error {
qs1Start <- struct{}{}
<-qs1StartAck
return nil
})
qs1.forkRun()
// Wait until querySource starts, then acknowledge starting.
<-qs1Start
go func() {
qs1StartAck <- struct{}{}
}()
// Wait for querySource to complete.
err := qs1.Wait()
assert.NoError(t, err)
assert.False(t, called1)
err = qs1.Wait()
assert.NoError(t, err)
assert.False(t, called1)
var called2 bool
resmon2 := mockResmon{
CancelF: func() error {
called2 = true
return nil
},
}
// Cancellation case.
//
// test blocks until querySource signals execution has started
// -> querySource blocks until test acknowledges querySource's signal
// -> test blocks on `Wait()` until querySource completes.
qs2Start, qs2StartAck := make(chan interface{}), make(chan interface{})
qs2, cancelQs2 := newTestQuerySource(&resmon2, func(*querySource) error {
qs2Start <- struct{}{}
// Block forever.
<-qs2StartAck
return nil
})
qs2.forkRun()
// Wait until querySource starts, then cancel.
<-qs2Start
go func() {
cancelQs2()
}()
// Wait for querySource to complete.
err = qs2.Wait()
assert.NoError(t, err)
assert.True(t, called2)
err = qs2.Wait()
assert.NoError(t, err)
assert.True(t, called2)
}
func TestQueryResourceMonitor_UnsupportedOperations(t *testing.T) {
t.Parallel()
rm := &queryResmon{}
_, err := rm.ReadResource(context.Background(), nil)
assert.Error(t, err)
assert.Equal(t, "Query mode does not support reading resources", err.Error())
_, err = rm.RegisterResource(context.Background(), nil)
assert.Error(t, err)
assert.Equal(t, "Query mode does not support creating, updating, or deleting resources", err.Error())
_, err = rm.RegisterResourceOutputs(context.Background(), nil)
assert.Error(t, err)
assert.Equal(t, "Query mode does not support registering resource operations", err.Error())
}
func TestQueryResourceMonitor(t *testing.T) {
t.Parallel()
t.Run("newQueryResourceMonitor", func(t *testing.T) {
t.Parallel()
t.Run("bad decrypter", func(t *testing.T) {
t.Parallel()
providerRegErrChan := make(chan error, 1)
expectedErr := fmt.Errorf("expected error")
resmon, err := newQueryResourceMonitor(
nil, nil, nil, nil, nil, providerRegErrChan, nil, &EvalRunInfo{
Proj: &workspace.Project{
Name: "expected-project",
},
Target: &Target{
Config: config.Map{
config.MustMakeKey("test", "secret"): config.NewSecureValue("secret-value"),
config.MustMakeKey("test", "regular"): config.NewValue("regular-value"),
},
Decrypter: &decrypterMock{
DecryptValueF: func(
ctx context.Context, ciphertext string,
) (string, error) {
return "", expectedErr
},
},
},
},
)
_ = resmon
assert.ErrorIs(t, err, expectedErr)
})
t.Run("ok", func(t *testing.T) {
t.Parallel()
providerRegErrChan := make(chan error, 1)
resmon, err := newQueryResourceMonitor(
nil, nil, nil, nil, nil, providerRegErrChan, nil, &EvalRunInfo{
Proj: &workspace.Project{
Name: "expected-project",
},
Target: &Target{
Name: tokens.MustParseStackName("expected-name"),
},
},
)
assert.NoError(t, err)
assert.Equal(t, "expected-project", resmon.callInfo.Project)
assert.Equal(t, "expected-name", resmon.callInfo.Stack)
})
})
t.Run("Cancel", func(t *testing.T) {
t.Parallel()
expectedErr := fmt.Errorf("expected-error")
done := make(chan error, 1)
done <- expectedErr
rm := &queryResmon{
cancel: make(chan bool),
done: done,
}
assert.ErrorIs(t, rm.Cancel(), expectedErr)
})
t.Run("Invoke", func(t *testing.T) {
t.Parallel()
t.Run("bad provider request", func(t *testing.T) {
t.Parallel()
t.Run("bad version", func(t *testing.T) {
t.Parallel()
rm := &queryResmon{}
_, err := rm.Invoke(context.Background(), &pulumirpc.ResourceInvokeRequest{
Tok: "pkgA:index:func",
Version: "bad-version",
})
assert.ErrorContains(t, err, "No Major.Minor.Patch elements found")
})
t.Run("default provider error", func(t *testing.T) {
t.Parallel()
providerRegChan := make(chan *registerResourceEvent, 1)
requests := make(chan defaultProviderRequest, 1)
rm := &queryResmon{
reg: &providers.Registry{},
defaultProviders: &defaultProviders{
requests: requests,
providerRegChan: providerRegChan,
},
}
wg := &sync.WaitGroup{}
wg.Add(1)
expectedErr := fmt.Errorf("expected error")
// Needed so defaultProviders.handleRequest() doesn't hang.
go func() {
req := <-rm.defaultProviders.requests
req.response <- defaultProviderResponse{
err: expectedErr,
}
wg.Done()
}()
_, err := rm.Invoke(context.Background(), &pulumirpc.ResourceInvokeRequest{
Tok: "pkgA:index:func",
Version: "1.0.0",
})
wg.Wait()
assert.ErrorIs(t, err, expectedErr)
})
})
})
}
//
// Test querySource constructor.
//
func newTestQuerySource(mon SourceResourceMonitor,
runLangPlugin func(*querySource) error,
) (*querySource, context.CancelFunc) {
cancel, cancelFunc := context.WithCancel(context.Background())
return &querySource{
mon: mon,
runLangPlugin: runLangPlugin,
langPluginFinChan: make(chan error),
cancel: cancel,
}, cancelFunc
}
//
// Mock resource monitor.
//
type mockResmon struct {
AddressF func() string
CancelF func() error
InvokeF func(ctx context.Context,
req *pulumirpc.ResourceInvokeRequest) (*pulumirpc.InvokeResponse, error)
CallF func(ctx context.Context,
req *pulumirpc.CallRequest) (*pulumirpc.CallResponse, error)
ReadResourceF func(ctx context.Context,
req *pulumirpc.ReadResourceRequest) (*pulumirpc.ReadResourceResponse, error)
RegisterResourceF func(ctx context.Context,
req *pulumirpc.RegisterResourceRequest) (*pulumirpc.RegisterResourceResponse, error)
RegisterResourceOutputsF func(ctx context.Context,
req *pulumirpc.RegisterResourceOutputsRequest) (*pbempty.Empty, error)
}
var _ SourceResourceMonitor = (*mockResmon)(nil)
func (rm *mockResmon) Address() string {
if rm.AddressF != nil {
return rm.AddressF()
}
panic("not implemented")
}
func (rm *mockResmon) Cancel() error {
if rm.CancelF != nil {
return rm.CancelF()
}
panic("not implemented")
}
func (rm *mockResmon) Invoke(ctx context.Context,
req *pulumirpc.ResourceInvokeRequest,
) (*pulumirpc.InvokeResponse, error) {
if rm.InvokeF != nil {
return rm.InvokeF(ctx, req)
}
panic("not implemented")
}
func (rm *mockResmon) Call(ctx context.Context,
req *pulumirpc.CallRequest,
) (*pulumirpc.CallResponse, error) {
if rm.CallF != nil {
return rm.CallF(ctx, req)
}
panic("not implemented")
}
func (rm *mockResmon) ReadResource(ctx context.Context,
req *pulumirpc.ReadResourceRequest,
) (*pulumirpc.ReadResourceResponse, error) {
if rm.ReadResourceF != nil {
return rm.ReadResourceF(ctx, req)
}
panic("not implemented")
}
func (rm *mockResmon) RegisterResource(ctx context.Context,
req *pulumirpc.RegisterResourceRequest,
) (*pulumirpc.RegisterResourceResponse, error) {
if rm.RegisterResourceF != nil {
return rm.RegisterResourceF(ctx, req)
}
panic("not implemented")
}
func (rm *mockResmon) RegisterResourceOutputs(ctx context.Context,
req *pulumirpc.RegisterResourceOutputsRequest,
) (*pbempty.Empty, error) {
if rm.RegisterResourceOutputsF != nil {
return rm.RegisterResourceOutputsF(ctx, req)
}
panic("not implemented")
}
func TestQuerySource(t *testing.T) {
t.Parallel()
t.Run("Wait", func(t *testing.T) {
t.Parallel()
var called bool
providerRegErrChan := make(chan error, 1)
expectedErr := fmt.Errorf("expected error")
providerRegErrChan <- expectedErr
src := &querySource{
providerRegErrChan: providerRegErrChan,
// Required to not nil ptr dereference.
cancel: context.Background(),
mon: &mockResmon{
CancelF: func() error {
called = true
return nil
},
},
}
err := src.Wait()
assert.ErrorIs(t, err, expectedErr)
assert.True(t, called)
})
}
func TestRunLangPlugin(t *testing.T) {
t.Parallel()
t.Run("failed to launch language host", func(t *testing.T) {
t.Parallel()
assert.ErrorContains(t, runLangPlugin(&querySource{
plugctx: &plugin.Context{
Host: &mockHost{
LanguageRuntimeF: func(root, pwd, runtime string, options map[string]interface{}) (plugin.LanguageRuntime, error) {
return nil, fmt.Errorf("expected error")
},
},
},
runinfo: &EvalRunInfo{
Proj: &workspace.Project{
Runtime: workspace.NewProjectRuntimeInfo("stuff", map[string]interface{}{}),
},
},
}), "failed to launch language host")
})
t.Run("bad decrypter", func(t *testing.T) {
t.Parallel()
expectedErr := fmt.Errorf("expected error")
err := runLangPlugin(&querySource{
plugctx: &plugin.Context{
Host: &mockHost{
LanguageRuntimeF: func(root, pwd, runtime string, options map[string]interface{}) (plugin.LanguageRuntime, error) {
return &mockLanguageRuntime{}, nil
},
},
},
runinfo: &EvalRunInfo{
Proj: &workspace.Project{
Runtime: workspace.NewProjectRuntimeInfo("stuff", map[string]interface{}{}),
},
Target: &Target{
Config: config.Map{
config.MustMakeKey("test", "secret"): config.NewSecureValue("secret-value"),
config.MustMakeKey("test", "regular"): config.NewValue("regular-value"),
},
Decrypter: &decrypterMock{
DecryptValueF: func(
ctx context.Context, ciphertext string,
) (string, error) {
return "", expectedErr
},
},
},
},
})
assert.ErrorIs(t, err, expectedErr)
})
t.Run("bail successfully", func(t *testing.T) {
t.Parallel()
err := runLangPlugin(&querySource{
plugctx: &plugin.Context{
Host: &mockHost{
LanguageRuntimeF: func(root, pwd, runtime string, options map[string]interface{}) (plugin.LanguageRuntime, error) {
return &mockLanguageRuntime{
RunF: func(info plugin.RunInfo) (string, bool, error) {
return "bail should override progerr", true /* bail */, nil
},
}, nil
},
},
},
// Prevent nilptr dereference.
mon: &mockResmon{
AddressF: func() string { return "" },
},
runinfo: &EvalRunInfo{
Proj: &workspace.Project{
Runtime: workspace.NewProjectRuntimeInfo("stuff", map[string]interface{}{}),
},
},
})
assert.ErrorContains(t, err, "run bailed")
})
t.Run("progerr", func(t *testing.T) {
t.Parallel()
err := runLangPlugin(&querySource{
plugctx: &plugin.Context{
Host: &mockHost{
LanguageRuntimeF: func(root, pwd, runtime string, options map[string]interface{}) (plugin.LanguageRuntime, error) {
return &mockLanguageRuntime{
RunF: func(info plugin.RunInfo) (string, bool, error) {
return "expected progerr", false /* bail */, nil
},
}, nil
},
},
},
// Prevent nilptr dereference.
mon: &mockResmon{
AddressF: func() string { return "" },
},
runinfo: &EvalRunInfo{
Proj: &workspace.Project{
Runtime: workspace.NewProjectRuntimeInfo("stuff", map[string]interface{}{}),
},
},
})
assert.ErrorContains(t, err, "expected progerr")
})
t.Run("langhost is run correctly", func(t *testing.T) {
t.Parallel()
var runCalled bool
err := runLangPlugin(&querySource{
plugctx: &plugin.Context{
Host: &mockHost{
LanguageRuntimeF: func(root, pwd, runtime string, options map[string]interface{}) (plugin.LanguageRuntime, error) {
return &mockLanguageRuntime{
RunF: func(info plugin.RunInfo) (string, bool, error) {
runCalled = true
assert.Equal(t, "expected-address", info.MonitorAddress)
assert.Equal(t, "expected-stack", info.Stack)
assert.Equal(t, "expected-project", info.Project)
assert.Equal(t, "expected-pwd", info.Pwd)
assert.Equal(t, "expected-program", info.Program)
assert.Equal(t, []string{"expected", "args"}, info.Args)
assert.Equal(t, "secret-value", info.Config[config.MustMakeKey("test", "secret")])
assert.Equal(t, "regular-value", info.Config[config.MustMakeKey("test", "regular")])
assert.True(t, info.QueryMode)
assert.True(t, info.DryRun)
// Disregard Parallel argument.
assert.Equal(t, "expected-organization", info.Organization)
return "", false, nil
},
}, nil
},
},
},
// Prevent nilptr dereference.
mon: &mockResmon{
AddressF: func() string { return "expected-address" },
},
runinfo: &EvalRunInfo{
Proj: &workspace.Project{
Name: "expected-project",
Runtime: workspace.NewProjectRuntimeInfo("stuff", map[string]interface{}{}),
},
Pwd: "expected-pwd",
Program: "expected-program",
Args: []string{"expected", "args"},
Target: &Target{
Config: config.Map{
config.MustMakeKey("test", "secret"): config.NewSecureValue("secret-value"),
config.MustMakeKey("test", "regular"): config.NewValue("regular-value"),
},
Name: tokens.MustParseStackName("expected-stack"),
Organization: "expected-organization",
Decrypter: &decrypterMock{
DecryptValueF: func(
ctx context.Context, ciphertext string,
) (string, error) {
return ciphertext, nil
},
},
},
},
})
assert.NoError(t, err)
assert.True(t, runCalled)
})
}
type mockHost struct {
ServerAddrF func() string
LogF func(sev diag.Severity, urn resource.URN, msg string, streamID int32)
LogStatusF func(sev diag.Severity, urn resource.URN, msg string, streamID int32)
AnalyzerF func(nm tokens.QName) (plugin.Analyzer, error)
PolicyAnalyzerF func(name tokens.QName, path string, opts *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error)
ListAnalyzersF func() []plugin.Analyzer
ProviderF func(pkg tokens.Package, version *semver.Version) (plugin.Provider, error)
CloseProviderF func(provider plugin.Provider) error
LanguageRuntimeF func(root, pwd, runtime string, options map[string]interface{}) (plugin.LanguageRuntime, error)
EnsurePluginsF func(plugins []workspace.PluginSpec, kinds plugin.Flags) error
ResolvePluginF func(kind workspace.PluginKind, name string, version *semver.Version) (*workspace.PluginInfo, error)
GetProjectPluginsF func() []workspace.ProjectPlugin
SignalCancellationF func() error
CloseF func() error
}
var _ plugin.Host = (*mockHost)(nil)
func (h *mockHost) ServerAddr() string {
if h.ServerAddrF != nil {
return h.ServerAddrF()
}
panic("unimplemented")
}
func (h *mockHost) Log(sev diag.Severity, urn resource.URN, msg string, streamID int32) {
if h.LogF != nil {
h.LogF(sev, urn, msg, streamID)
return
}
panic("unimplemented")
}
func (h *mockHost) LogStatus(sev diag.Severity, urn resource.URN, msg string, streamID int32) {
if h.LogStatusF != nil {
h.LogStatusF(sev, urn, msg, streamID)
return
}
panic("unimplemented")
}
func (h *mockHost) Analyzer(nm tokens.QName) (plugin.Analyzer, error) {
if h.AnalyzerF != nil {
return h.Analyzer(nm)
}
panic("unimplemented")
}
func (h *mockHost) PolicyAnalyzer(
name tokens.QName, path string, opts *plugin.PolicyAnalyzerOptions,
) (plugin.Analyzer, error) {
if h.PolicyAnalyzerF != nil {
return h.PolicyAnalyzerF(name, path, opts)
}
panic("unimplemented")
}
func (h *mockHost) ListAnalyzers() []plugin.Analyzer {
if h.ListAnalyzersF != nil {
return h.ListAnalyzersF()
}
panic("unimplemented")
}
func (h *mockHost) Provider(pkg tokens.Package, version *semver.Version) (plugin.Provider, error) {
if h.ProviderF != nil {
return h.ProviderF(pkg, version)
}
panic("unimplemented")
}
func (h *mockHost) CloseProvider(provider plugin.Provider) error {
if h.CloseProviderF != nil {
return h.CloseProviderF(provider)
}
panic("unimplemented")
}
func (h *mockHost) LanguageRuntime(
root, pwd, runtime string, options map[string]interface{},
) (plugin.LanguageRuntime, error) {
if h.LanguageRuntimeF != nil {
return h.LanguageRuntimeF(root, pwd, runtime, options)
}
panic("unimplemented")
}
func (h *mockHost) EnsurePlugins(plugins []workspace.PluginSpec, kinds plugin.Flags) error {
if h.EnsurePluginsF != nil {
return h.EnsurePluginsF(plugins, kinds)
}
panic("unimplemented")
}
func (h *mockHost) ResolvePlugin(
kind workspace.PluginKind, name string, version *semver.Version,
) (*workspace.PluginInfo, error) {
if h.ResolvePluginF != nil {
return h.ResolvePluginF(kind, name, version)
}
panic("unimplemented")
}
func (h *mockHost) GetProjectPlugins() []workspace.ProjectPlugin {
if h.GetProjectPluginsF != nil {
return h.GetProjectPluginsF()
}
panic("unimplemented")
}
func (h *mockHost) SignalCancellation() error {
if h.SignalCancellationF != nil {
return h.SignalCancellationF()
}
panic("unimplemented")
}
// Close reclaims any resources associated with the host.
func (h *mockHost) Close() error {
if h.CloseF != nil {
return h.CloseF()
}
panic("unimplemented")
}
type mockLanguageRuntime struct {
CloseF func() error
GetRequiredPluginsF func(info plugin.ProgInfo) ([]workspace.PluginSpec, error)
RunF func(info plugin.RunInfo) (string, bool, error)
GetPluginInfoF func() (workspace.PluginInfo, error)
InstallDependenciesF func(pwd, main string) error
AboutF func() (plugin.AboutInfo, error)
GetProgramDependenciesF func(
info plugin.ProgInfo, transitiveDependencies bool,
) ([]plugin.DependencyInfo, error)
RunPluginF func(
info plugin.RunPluginInfo,
) (io.Reader, io.Reader, context.CancelFunc, error)
GenerateProjectF func(
sourceDirectory, targetDirectory, project string,
strict bool, loaderTarget string, localDependencies map[string]string,
) (hcl.Diagnostics, error)
GeneratePackageF func(
directory string, schema string,
extraFiles map[string][]byte, loaderTarget string,
) (hcl.Diagnostics, error)
GenerateProgramF func(
program map[string]string,
loaderTarget string,
) (map[string][]byte, hcl.Diagnostics, error)
PackF func(
packageDirectory string,
version semver.Version,
destinationDirectory string,
) (string, error)
}
var _ plugin.LanguageRuntime = (*mockLanguageRuntime)(nil)
func (rt *mockLanguageRuntime) Close() error {
if rt.CloseF != nil {
return rt.CloseF()
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) GetRequiredPlugins(info plugin.ProgInfo) ([]workspace.PluginSpec, error) {
if rt.GetRequiredPluginsF != nil {
return rt.GetRequiredPluginsF(info)
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) Run(info plugin.RunInfo) (string, bool, error) {
if rt.RunF != nil {
return rt.RunF(info)
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) GetPluginInfo() (workspace.PluginInfo, error) {
if rt.GetPluginInfoF != nil {
return rt.GetPluginInfoF()
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) InstallDependencies(pwd, main string) error {
if rt.InstallDependenciesF != nil {
return rt.InstallDependenciesF(pwd, main)
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) About() (plugin.AboutInfo, error) {
if rt.AboutF != nil {
return rt.AboutF()
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) GetProgramDependencies(
info plugin.ProgInfo, transitiveDependencies bool,
) ([]plugin.DependencyInfo, error) {
if rt.GetProgramDependenciesF != nil {
return rt.GetProgramDependenciesF(info, transitiveDependencies)
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) RunPlugin(
info plugin.RunPluginInfo,
) (io.Reader, io.Reader, context.CancelFunc, error) {
if rt.RunPluginF != nil {
return rt.RunPluginF(info)
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) GenerateProject(
sourceDirectory, targetDirectory, project string,
strict bool, loaderTarget string, localDependencies map[string]string,
) (hcl.Diagnostics, error) {
if rt.GenerateProjectF != nil {
return rt.GenerateProjectF(
sourceDirectory, targetDirectory, project, strict, loaderTarget, localDependencies)
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) GeneratePackage(
directory string, schema string, extraFiles map[string][]byte, loaderTarget string,
) (hcl.Diagnostics, error) {
if rt.GeneratePackageF != nil {
return rt.GeneratePackageF(directory, schema, extraFiles, loaderTarget)
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) GenerateProgram(
program map[string]string, loaderTarget string,
) (map[string][]byte, hcl.Diagnostics, error) {
if rt.GenerateProgramF != nil {
return rt.GenerateProgramF(program, loaderTarget)
}
panic("unimplemented")
}
func (rt *mockLanguageRuntime) Pack(
packageDirectory string, version semver.Version, destinationDirectory string,
) (string, error) {
if rt.PackF != nil {
return rt.PackF(packageDirectory, version, destinationDirectory)
}
panic("unimplemented")
}