mirror of
https://github.com/hashicorp/packer.git
synced 2026-03-28 05:14:13 -04:00
The lib name for the common components for writing packer_test suites was not clear, and did not follow the convention established in Packer core and plugins. Therefore this commit does two things: first the lib is renamed into common as to follow this convention, and clearly document which components are common to all tests. Also checkers are placed in a subpackage of common, common/check, so that it is clearer what is meant to be used as checks for a command's execution status after it's been run, as part of Assert.
92 lines
2.7 KiB
Go
92 lines
2.7 KiB
Go
// Copyright (c) HashiCorp, Inc.
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
//go:generate packer-sdc mapstructure-to-hcl2 -type Config,NestedFirst,NestedSecond
|
|
|
|
package dynamic
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/hashicorp/hcl/v2/hcldec"
|
|
"github.com/hashicorp/packer-plugin-sdk/common"
|
|
"github.com/hashicorp/packer-plugin-sdk/multistep"
|
|
"github.com/hashicorp/packer-plugin-sdk/multistep/commonsteps"
|
|
"github.com/hashicorp/packer-plugin-sdk/packer"
|
|
"github.com/hashicorp/packer-plugin-sdk/template/config"
|
|
)
|
|
|
|
const BuilderId = "dynamic.builder"
|
|
|
|
type NestedSecond struct {
|
|
Name string `mapstructure:"name" required:"true"`
|
|
}
|
|
|
|
type NestedFirst struct {
|
|
Name string `mapstructure:"name" required:"true"`
|
|
Nesteds []NestedSecond `mapstructure:"extra" required:"false"`
|
|
}
|
|
|
|
type Config struct {
|
|
common.PackerConfig `mapstructure:",squash"`
|
|
Nesteds []NestedFirst `mapstructure:"extra" required:"false"`
|
|
}
|
|
|
|
type Builder struct {
|
|
config Config
|
|
runner multistep.Runner
|
|
}
|
|
|
|
func (b *Builder) ConfigSpec() hcldec.ObjectSpec { return b.config.FlatMapstructure().HCL2Spec() }
|
|
|
|
func (b *Builder) Prepare(raws ...interface{}) (generatedVars []string, warnings []string, err error) {
|
|
err = config.Decode(&b.config, &config.DecodeOpts{
|
|
PluginType: "packer.builder.dynamic",
|
|
Interpolate: true,
|
|
}, raws...)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
// Return the placeholder for the generated data that will become available to provisioners and post-processors.
|
|
// If the builder doesn't generate any data, just return an empty slice of string: []string{}
|
|
buildGeneratedData := []string{"GeneratedMockData"}
|
|
return buildGeneratedData, nil, nil
|
|
}
|
|
|
|
func (b *Builder) Run(ctx context.Context, ui packer.Ui, hook packer.Hook) (packer.Artifact, error) {
|
|
steps := []multistep.Step{}
|
|
|
|
steps = append(steps,
|
|
&StepSayConfig{
|
|
b.config,
|
|
},
|
|
new(commonsteps.StepProvision),
|
|
)
|
|
|
|
// Setup the state bag and initial state for the steps
|
|
state := new(multistep.BasicStateBag)
|
|
state.Put("hook", hook)
|
|
state.Put("ui", ui)
|
|
|
|
// Set the value of the generated data that will become available to provisioners.
|
|
// To share the data with post-processors, use the StateData in the artifact.
|
|
state.Put("generated_data", map[string]interface{}{
|
|
"GeneratedMockData": "mock-build-data",
|
|
})
|
|
|
|
// Run!
|
|
b.runner = commonsteps.NewRunner(steps, b.config.PackerConfig, ui)
|
|
b.runner.Run(ctx, state)
|
|
|
|
// If there was an error, return that
|
|
if err, ok := state.GetOk("error"); ok {
|
|
return nil, err.(error)
|
|
}
|
|
|
|
artifact := &Artifact{
|
|
// Add the builder generated data to the artifact StateData so that post-processors
|
|
// can access them.
|
|
StateData: map[string]interface{}{"generated_data": state.Get("generated_data")},
|
|
}
|
|
return artifact, nil
|
|
}
|