Skip to content

Commit

Permalink
hcp: generate fingerprints on each new build
Browse files Browse the repository at this point in the history
Fingerprints are how we link a packer build to an iteration on HCP.
These are computed automatically from the Git SHA in the current state,
and are unique to the bucket/iteration.

The main problem with this approach is that while sound in theory, it
quickly falls apart when users want to run the same build configuration
twice, but expect a new image to be created.

With the current model, this fails, as the iteration with the current
SHA already exists.

While this is solvable through environment variables, or by committing a
change to the repository, we think this is not clear enough, and causes
an extra step to what should otherwise be a simple process.

Therefore, to lower the barrier of entry into HCP, we change this
behaviour with this commit.

Now, fingerprints are randomly generated ULIDs instead of a git SHA, and
a new one is always generated, unless one is already specified in the
environment.

This makes continuation of an existing iteration a conscious choice
rather than something automatic, and virtually eliminates conflicts such
as the ones described above.
  • Loading branch information
lbajolet-hashicorp committed Dec 22, 2022
1 parent 3ea17fc commit cbe4b99
Show file tree
Hide file tree
Showing 12 changed files with 164 additions and 94 deletions.
9 changes: 9 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,15 @@
displayed if both were available, and now both will be displayed. This does
not impact the behaviour of options such as only or except, they will continue
to work as they did before.
* hcp: iteration fingerprints used to be computed from the Git SHA of the repository
where the template is located when running packer build. This changes with this
release, and now fingerprints are automatically generated as a ULID. This implies
that continuing an existing iteration will require users to define the
fingerprint in the environment manually in order to adopt this behaviour,
otherwise, by default, a new iteration will be created. This does not impact
workflows where the fingerprint was defined through the
HCP_PACKER_ITERATION_FINGERPRINT environment variable, and these builds will work
exactly as they did before.

## 1.8.5 (December 12, 2022)

Expand Down
2 changes: 2 additions & 0 deletions command/build.go
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,8 @@ func (c *BuildCommand) RunContext(buildCtx context.Context, cla *BuildArgs) int
return ret
}

defer hcpRegistry.IterationStatusSummary(c.Ui)

err := hcpRegistry.PopulateIteration(buildCtx)
if err != nil {
return writeDiags(c.Ui, nil, hcl.Diagnostics{
Expand Down
13 changes: 13 additions & 0 deletions internal/hcp/registry/hcl.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package registry
import (
"context"
"fmt"
"log"

"github.com/hashicorp/hcl/v2"
"github.com/hashicorp/hcp-sdk-go/clients/cloud-packer-service/stable/2021-04-30/models"
Expand Down Expand Up @@ -42,6 +43,13 @@ func (h *HCLMetadataRegistry) PopulateIteration(ctx context.Context) error {

h.configuration.HCPVars["iterationID"] = cty.StringVal(iterationID)

sha, err := getGitSHA(h.configuration.Basedir)
if err != nil {
log.Printf("failed to get GIT SHA from environment, won't set as build labels")
} else {
h.bucket.Iteration.AddSHAToBuildLabels(sha)
}

return nil
}

Expand Down Expand Up @@ -70,6 +78,11 @@ func (h *HCLMetadataRegistry) CompleteBuild(
return h.bucket.completeBuild(ctx, name, artifacts, buildErr)
}

// IterationStatusSummary prints a status report in the UI if the iteration is not yet done
func (h *HCLMetadataRegistry) IterationStatusSummary(ui sdkpacker.Ui) {
h.bucket.Iteration.iterationStatusSummary(ui)
}

func NewHCLMetadataRegistry(config *hcl2template.PackerConfig) (*HCLMetadataRegistry, hcl.Diagnostics) {
var diags hcl.Diagnostics
if len(config.Builds) > 1 {
Expand Down
45 changes: 42 additions & 3 deletions internal/hcp/registry/hcp.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package registry
import (
"fmt"

git "github.com/go-git/go-git/v5"
"github.com/hashicorp/hcl/v2"
"github.com/hashicorp/packer/hcl2template"
"github.com/hashicorp/packer/internal/hcp/env"
Expand Down Expand Up @@ -87,9 +88,15 @@ func createConfiguredBucket(templateDir string, opts ...bucketConfigurationOpts)
})
}

err := bucket.Iteration.Initialize(IterationOptions{
TemplateBaseDir: templateDir,
})
err := bucket.Iteration.Initialize()
if err != nil {
diags = append(diags, &hcl.Diagnostic{
Severity: hcl.DiagError,
Summary: "Failed to initialize iteration",
Detail: fmt.Sprintf("The iteration failed to be initialized for bucket %q: %s",
bucket.Slug, err),
})
}

if err != nil {
diags = append(diags, &hcl.Diagnostic{
Expand All @@ -109,3 +116,35 @@ func withPackerEnvConfiguration(bucket *Bucket) hcl.Diagnostics {

return nil
}

// getGitSHA returns the HEAD commit for some template dir defined in baseDir.
// If the base directory is not under version control an error is returned.
func getGitSHA(baseDir string) (string, error) {
r, err := git.PlainOpenWithOptions(baseDir, &git.PlainOpenOptions{
DetectDotGit: true,
})

if err != nil {
return "", fmt.Errorf("Packer could not read the fingerprint from git.")
}

// The config can be used to retrieve user identity. for example,
// c.User.Email. Leaving in but commented because I'm not sure we care
// about this identity right now. - Megan
//
// c, err := r.ConfigScoped(config.GlobalScope)
// if err != nil {
// return "", fmt.Errorf("Error setting git scope", err)
// }
ref, err := r.Head()
if err != nil {
// If we get there, we're in a Git dir, but HEAD cannot be read.
//
// This may happen when there's no commit in the git dir.
return "", fmt.Errorf("Packer could not read a git SHA in directory %q: %s", baseDir, err)
}

// log.Printf("Author: %v, Commit: %v\n", c.User.Email, ref.Hash())

return ref.Hash().String(), nil
}
13 changes: 13 additions & 0 deletions internal/hcp/registry/json.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package registry
import (
"context"
"fmt"
"log"
"path/filepath"

"github.com/hashicorp/hcl/v2"
Expand Down Expand Up @@ -63,6 +64,13 @@ func (h *JSONMetadataRegistry) PopulateIteration(ctx context.Context) error {
return err
}

sha, err := getGitSHA(h.configuration.Template.Path)
if err != nil {
log.Printf("failed to get GIT SHA from environment, won't set as build labels")
} else {
h.bucket.Iteration.AddSHAToBuildLabels(sha)
}

return nil
}

Expand All @@ -80,3 +88,8 @@ func (h *JSONMetadataRegistry) CompleteBuild(
) ([]sdkpacker.Artifact, error) {
return h.bucket.completeBuild(ctx, build.Name(), artifacts, buildErr)
}

// IterationStatusSummary prints a status report in the UI if the iteration is not yet done
func (h *JSONMetadataRegistry) IterationStatusSummary(ui sdkpacker.Ui) {
h.bucket.Iteration.iterationStatusSummary(ui)
}
3 changes: 3 additions & 0 deletions internal/hcp/registry/null_registry.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,3 +25,6 @@ func (r nullRegistry) CompleteBuild(
) ([]sdkpacker.Artifact, error) {
return artifacts, nil
}

func (r nullRegistry) IterationStatusSummary(_ sdkpacker.Ui) {
}
2 changes: 1 addition & 1 deletion internal/hcp/registry/registry.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,10 +12,10 @@ import (

// Registry is an entity capable to orchestrate a Packer build and upload metadata to HCP
type Registry interface {
//Configure(packer.Handler)
PopulateIteration(context.Context) error
StartBuild(context.Context, sdkpacker.Build) error
CompleteBuild(ctx context.Context, build sdkpacker.Build, artifacts []sdkpacker.Artifact, buildErr error) ([]sdkpacker.Artifact, error)
IterationStatusSummary(writer sdkpacker.Ui)
}

// New instanciates the appropriate registry for the Packer configuration template type.
Expand Down
21 changes: 7 additions & 14 deletions internal/hcp/registry/types.bucket_service_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ import (
)

func TestInitialize_NewBucketNewIteration(t *testing.T) {
t.Setenv("HCP_PACKER_BUILD_FINGERPRINT", "testnumber")
mockService := api.NewMockPackerClientService()

b := &Bucket{
Expand All @@ -20,7 +19,7 @@ func TestInitialize_NewBucketNewIteration(t *testing.T) {
}

b.Iteration = NewIteration()
err := b.Iteration.Initialize(IterationOptions{})
err := b.Iteration.Initialize()
if err != nil {
t.Errorf("unexpected failure: %v", err)
}
Expand Down Expand Up @@ -62,7 +61,6 @@ func TestInitialize_NewBucketNewIteration(t *testing.T) {
}

func TestInitialize_UnsetTemplateTypeError(t *testing.T) {
t.Setenv("HCP_PACKER_BUILD_FINGERPRINT", "testunsettemplate")
mockService := api.NewMockPackerClientService()
mockService.BucketAlreadyExist = true

Expand All @@ -74,7 +72,7 @@ func TestInitialize_UnsetTemplateTypeError(t *testing.T) {
}

b.Iteration = NewIteration()
err := b.Iteration.Initialize(IterationOptions{})
err := b.Iteration.Initialize()
if err != nil {
t.Errorf("unexpected failure: %v", err)
}
Expand All @@ -88,7 +86,6 @@ func TestInitialize_UnsetTemplateTypeError(t *testing.T) {
}

func TestInitialize_ExistingBucketNewIteration(t *testing.T) {
t.Setenv("HCP_PACKER_BUILD_FINGERPRINT", "testnumber")
mockService := api.NewMockPackerClientService()
mockService.BucketAlreadyExist = true

Expand All @@ -100,7 +97,7 @@ func TestInitialize_ExistingBucketNewIteration(t *testing.T) {
}

b.Iteration = NewIteration()
err := b.Iteration.Initialize(IterationOptions{})
err := b.Iteration.Initialize()
if err != nil {
t.Errorf("unexpected failure: %v", err)
}
Expand Down Expand Up @@ -142,7 +139,6 @@ func TestInitialize_ExistingBucketNewIteration(t *testing.T) {
}

func TestInitialize_ExistingBucketExistingIteration(t *testing.T) {
t.Setenv("HCP_PACKER_BUILD_FINGERPRINT", "testnumber")
mockService := api.NewMockPackerClientService()
mockService.BucketAlreadyExist = true
mockService.IterationAlreadyExist = true
Expand All @@ -155,7 +151,7 @@ func TestInitialize_ExistingBucketExistingIteration(t *testing.T) {
}

b.Iteration = NewIteration()
err := b.Iteration.Initialize(IterationOptions{})
err := b.Iteration.Initialize()
if err != nil {
t.Errorf("unexpected failure: %v", err)
}
Expand Down Expand Up @@ -212,7 +208,6 @@ func TestInitialize_ExistingBucketExistingIteration(t *testing.T) {
}

func TestInitialize_ExistingBucketCompleteIteration(t *testing.T) {
t.Setenv("HCP_PACKER_BUILD_FINGERPRINT", "testnumber")
mockService := api.NewMockPackerClientService()
mockService.BucketAlreadyExist = true
mockService.IterationAlreadyExist = true
Expand All @@ -227,7 +222,7 @@ func TestInitialize_ExistingBucketCompleteIteration(t *testing.T) {
}

b.Iteration = NewIteration()
err := b.Iteration.Initialize(IterationOptions{})
err := b.Iteration.Initialize()
if err != nil {
t.Errorf("unexpected failure: %v", err)
}
Expand Down Expand Up @@ -258,7 +253,6 @@ func TestInitialize_ExistingBucketCompleteIteration(t *testing.T) {
}

func TestUpdateBuildStatus(t *testing.T) {
t.Setenv("HCP_PACKER_BUILD_FINGERPRINT", "testnumber")
mockService := api.NewMockPackerClientService()
mockService.BucketAlreadyExist = true
mockService.IterationAlreadyExist = true
Expand All @@ -271,7 +265,7 @@ func TestUpdateBuildStatus(t *testing.T) {
}

b.Iteration = NewIteration()
err := b.Iteration.Initialize(IterationOptions{})
err := b.Iteration.Initialize()
if err != nil {
t.Errorf("unexpected failure: %v", err)
}
Expand Down Expand Up @@ -312,7 +306,6 @@ func TestUpdateBuildStatus(t *testing.T) {
}

func TestUpdateBuildStatus_DONENoImages(t *testing.T) {
t.Setenv("HCP_PACKER_BUILD_FINGERPRINT", "testnumber")
mockService := api.NewMockPackerClientService()
mockService.BucketAlreadyExist = true
mockService.IterationAlreadyExist = true
Expand All @@ -325,7 +318,7 @@ func TestUpdateBuildStatus_DONENoImages(t *testing.T) {
}

b.Iteration = NewIteration()
err := b.Iteration.Initialize(IterationOptions{})
err := b.Iteration.Initialize()
if err != nil {
t.Errorf("unexpected failure: %v", err)
}
Expand Down
6 changes: 2 additions & 4 deletions internal/hcp/registry/types.bucket_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,11 +10,9 @@ import (
)

func createInitialTestBucket(t testing.TB) *Bucket {
t.Setenv("HCP_PACKER_BUILD_FINGERPRINT", "no-fingerprint-here")

t.Helper()
bucket := NewBucketWithIteration()
err := bucket.Iteration.Initialize(IterationOptions{})
err := bucket.Iteration.Initialize()
if err != nil {
t.Errorf("failed to initialize Bucket: %s", err)
return nil
Expand Down Expand Up @@ -289,7 +287,7 @@ func TestBucket_PopulateIteration(t *testing.T) {
mockService.BuildAlreadyDone = tt.buildCompleted

bucket := NewBucketWithIteration()
err := bucket.Iteration.Initialize(IterationOptions{})
err := bucket.Iteration.Initialize()
if err != nil {
t.Fatalf("failed when calling NewBucketWithIteration: %s", err)
}
Expand Down
Loading

0 comments on commit cbe4b99

Please sign in to comment.