-
Notifications
You must be signed in to change notification settings - Fork 3.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Extract registry wrappers #11967
Extract registry wrappers #11967
Conversation
b519b12
to
dc62051
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just working through the code and leaving questions as I step through. We can discuss in detail but please feel free to iterate on my initial questions/suggestions as you see fit.
acctest/testing.go
Outdated
@@ -146,7 +146,7 @@ func Test(t TestT, c TestCase) { | |||
}, | |||
Template: tpl, | |||
}) | |||
err = core.Initialize() | |||
err = core.Initialize(packer.InitializeOptions{}) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why Packer an empty structure here? Maybe we can switch to functional args.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We can change this yes. We'd change the Handler
interface to make the arguments a variadic suite of functions, and we can adapt this for both PackerConfig
and Core
, I like the idea.
If you don't mind, I think it can be done in a later PR maybe? I don't want to bloat this one with extra changes, there's already a lot going on, but we can open an issue to track this, and we can iterate on the solution after this one is in a satisfactory state?
command/build.go
Outdated
ArtifactMetadataPublisher, diags := packerStarter.ConfiguredArtifactMetadataPublisher() | ||
if diags.HasErrors() { | ||
return writeDiags(c.Ui, nil, diags) | ||
hcpHandler, err := hcp.GetHCPHandler(packerStarter) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
GetHandler(PackerHandler) returns the appropriate HCP Handler for the Packer config type?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, exactly! If there's a HCP_PACKER_REGISTRY
variable set, or a hcp_packer_registry
block for HCL configs, we create the appropriate handler.
If none are set, we create a NoopHandler
that does nothing. This ensures we can call methods later in the process and have them behave appropriately.
command/build.go
Outdated
return writeDiags(c.Ui, nil, diags) | ||
hcpHandler, err := hcp.GetHCPHandler(packerStarter) | ||
if err != nil { | ||
c.Ui.Error(fmt.Sprintf("HCP setup failed: %s", err)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's change this to use hcl.Diagnostics as it can be used to provide addtional context and formatting.
command/build.go
Outdated
} | ||
err = hcpHandler.PopulateIteration(buildCtx) | ||
if err != nil { | ||
c.Ui.Error(fmt.Sprintf("HCP populating iteration failed: %s", err)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Need to change this error messaging.
@@ -137,6 +137,7 @@ func (p *Parser) Parse(filename string, varFiles []string, argVars map[string]st | |||
Basedir: basedir, | |||
Cwd: wd, | |||
CorePackerVersionString: p.CorePackerVersionString, | |||
HCPVars: map[string]cty.Value{}, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What are HCPVars?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
HCPVars is essentially how we set variables from the HCPHandler back into the configuration, for example now the iteration ID is propagated to the eval context, so it can be used in the configuration later on.
ectx.Variables[packerAccessor] = cty.ObjectVal(map[string]cty.Value{ | ||
"version": cty.StringVal(cfg.CorePackerVersionString), | ||
"iterationID": cty.StringVal(cfg.Bucket.Iteration.ID), | ||
"iterationID": iterID, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Right now this is accessible as packer.iterationID in HCL. Should we change this to be scope to hcp or something else?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We could change this eventually, but this would be a breaking change if we moved it directly.
I guess we should probably first define it in both places, and show a warning stating this is deprecated, and will be removed later on.
I guess having an hcp
accessor is relevant if we want to give access to more information that comes back from HCP later on, so we could probably do it.
I think it can be done in a later PR however
hcp/common.go
Outdated
ctx context.Context, | ||
handler HCPHandler, | ||
) error { | ||
err := handler.Bucket().Initialize(ctx) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We should probably ensure that handler is not nil.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We could ensure that, but since this is a private function, the only way this is called is inside of the package, and the callsite is from the handler themselves, so we should not reach such a case.
I think if that happens, it might as well be a crash that we should investigate and fix ASAP. We could replace the nil-ref panic by a more explicit one, but at that point I'm not sure it'd help too much, what do you think?
hcp/common.go
Outdated
|
||
err = handler.Bucket().PopulateIteration(ctx) | ||
if err != nil { | ||
return fmt.Errorf("HCP Packer Registry build initialization failed: %s", err) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: copy pasta on the error message. We want to think about the appropriate context to add for a user. I found the existing error messaging to be a bit terse.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Indeed, copy-pasta issue here.
I guess the HCP error could be returned directly, the context will be improved upon from the callsite anyway as it will end-up wrapped in an hcl.Diagnostic
.
hcp/common.go
Outdated
if err != nil { | ||
log.Printf("[ERROR] failed to update build %q status to FAILED: %s", buildName, err) | ||
} | ||
return artifacts, fmt.Errorf("build failed, not uploading artifacts") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What would cause a failure here?
If this is a communication hipcup should we try to dump the build artifact details for later use?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This error here is not a communication hiccup, it's that there was already an error with the build. I guess we could return nil,nil
here, since nothing is updated, and the artifacts are not uploaded, my bad.
hcp/common.go
Outdated
return artifacts, fmt.Errorf("build failed, not uploading artifacts") | ||
} | ||
|
||
for _, art := range artifacts { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: artifact is a bit clearer then art.
hcp/handler.go
Outdated
"context" | ||
|
||
sdkpacker "github.com/hashicorp/packer-plugin-sdk/packer" | ||
"github.com/hashicorp/packer/hcl2template" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This lint error appears to be an issue with golangci-lint v1.42.0 < 1.48.0, which is bit using Go 1.17 and not Go1.18.
I am able to reproduce locally. Updating to 1.48.0 does not report an issue. I suspect that it may be do to some of the Go dependencies.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
True, the PR I uploaded before that bumps the golint-ci should take care of this one here.
hcp/handler.go
Outdated
BuildDone(ctx context.Context, buildName string, artifacts []sdkpacker.Artifact, buildErr error) ([]sdkpacker.Artifact, error) | ||
} | ||
|
||
// TrySetupHCP attempts to setup the HCP-related structures if HCP is enabled |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Comment needs to be updated.
// TrySetupHCP attempts to setup the HCP-related structures if HCP is enabled | |
// GetHCPHandler ..... |
hcp/hcl_handler.go
Outdated
case *CoreWrapper: | ||
return setupRegistryForPackerCore(cfg) | ||
} | ||
type HCLHandler struct { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Guessing this is meant to be HCPHandler and not HCLHandler. Which is the same name as the HCPHandler interface so that wont work. Let think about this a bit more to see what to call it. We probably want to rename the HCPHandler interface as well.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In this case it's intended to be HCLHandler
since it handles HCL templates, but the name is confusing I agree, maybe we can see to replace it by something else.
I should probably add some comment on top to clarify why it's here though, the whole package is poorly documented for now, so I'll amend that for my next reroll.
ae1eabc
to
d2be583
Compare
d2be583
to
4272d9d
Compare
4272d9d
to
777e667
Compare
f33947b
to
0a80100
Compare
In order to determine whether or not a build is done or needs processing, we check 3 things: 1. that its status is not done 2. that its ID is not empty 3. that it has no images linked to it The was the image check was phrased indicated that we checked for the reverse, so we reverse the naming to make it less confusing.
To be able to track HCP builds during a Packer build, we need to propagate the name used for registering the build to HCP. This works out-of-the-box for JSON templates, as the build's name is always the builder's, so there's no confusion on that. For HCP templates however, there's a possibility for a template to define a name for the build, which is returned by a CoreBuild's Name() function, in addition to the source's Type/Name. The builds are registered to HCP with the source.String() function however, which does not contain the build's name. This prevents any build containing a name to work, as their name/source.String does not match. The name we registered the build with is stored in a CoreBuild as the Type attribute, but cannot be safely accessed, as builds are types within the command as packersdk.Build, which only exposes Name() for this purpose. In order to circumvent this problem, and as a way to present a prototype of solution that will likely have to be discussed before we merge it, this commit hotfixes the issue.
As part of the work to displace everything related to HCP from scattered places around the Packer code, we move it all to an hcp package. This in turn reduces the amount of code that the commands have to integrate, and leaves the HCP details to its own enclave.
2220846
to
da693f0
Compare
622d76e
to
ccb1f20
Compare
ccb1f20
to
d545b24
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The reroll looks good to me. I think there are a few naming changes that we discussed that need to be implemented, and a few comments that might still being referring to handler for the registry. But otherwise I good with it. When ready merge and let’s check if the artifact bug is resolved as well. Approving in advance.
command/build.go
Outdated
if err != nil { | ||
return writeDiags(c.Ui, nil, hcl.Diagnostics{ | ||
&hcl.Diagnostic{ | ||
Summary: "HCP: initial setup failed", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's update the messaging in the error to include call to action.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This was already updated on the latest reroll, now the diag comes from registry.New
and states that an unknown config type was provided, or in the related New.*Registry
functions, we get a more accurate diag(s) on the error we got.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this was a comment from a review I never submitted. Please disregard.
command/build.go
Outdated
if errors.As(err, &hcp.BuildDone{}) { | ||
ui.Say(fmt.Sprintf( | ||
"skipping HCP-enabled build %q: already done.", | ||
name)) | ||
return | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice! To not confuse BuildDone{} with BuildDone() I suggest calling this error what it is BuildDoneErr
.
if errors.As(err, &hcp.BuildDone{}) { | |
ui.Say(fmt.Sprintf( | |
"skipping HCP-enabled build %q: already done.", | |
name)) | |
return | |
} | |
} | |
if errors.As(err, &hcp.BuildDoneErr{}) { | |
ui.Say(fmt.Sprintf( | |
"skipping HCP-enabled build %q: already done.", | |
name)) | |
return | |
} | |
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have taken upon your suggestion and renamed it ErrBuildAlreadyDone
in the upcoming reroll
command/build.go
Outdated
@@ -88,38 +90,24 @@ func (c *BuildCommand) RunContext(buildCtx context.Context, cla *BuildArgs) int | |||
return ret | |||
} | |||
|
|||
diags = TrySetupHCP(packerStarter) | |||
hcpHandler, diags := hcp.GetOrchestrator(packerStarter) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What do we call this?
type nullRegistry struct{} | ||
|
||
// We don't need a constructor for such a simple type | ||
//func newNullHandler() Registry { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We can remove this.
internal/hcp/registry/registry.go
Outdated
//Configure(packer.Handler) | ||
PopulateIteration(context.Context) error | ||
BuildStart(context.Context, string) error | ||
BuildDone(ctx context.Context, buildName string, artifacts []sdkpacker.Artifact, buildErr error) ([]sdkpacker.Artifact, error) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let’s rename these methods as we discussed StartBuild
and CompleteBuild
d545b24
to
3ce7e0b
Compare
@@ -469,3 +495,115 @@ func (b *Bucket) HeartbeatBuild(ctx context.Context, build string) (func(), erro | |||
close(heartbeatChan) | |||
}, nil | |||
} | |||
|
|||
func (b *Bucket) StartBuild(ctx context.Context, buildName string) error { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
With these method names does the Bucket type also implement the Refistry interface?
if it does and that is not intentional we should probably give these different names. Maybe something like MarkBuildAsStarted etc …
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well, that's true, as it stands Bucket is an implementation of registry.
This has the side-effect of having to rename the current CompleteBuild
to completeBuild
, as it does the update of the status to DONE
and uploading the labels and other metadata. This can indeed be renamed in Bucket
so it's not an implementation of the Registry
interface.
I'll do another reroll later for that
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note: since the two are in the same package, we can make the Bucket
functions private, so it's not an implementation anymore. This removes the potential confusion when looking for the Registry implementations.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sounds good to me. 🚢
3ce7e0b
to
c074808
Compare
This commit reorganises the code for both the registry/API and the Orchestrator/Registry. The main difference with the previous version is how stuff is exposed. Now we only expose a Registry interface to the outside (previously named Orchestrator), which has several implementations: null is the default, and is returned if HCP is not enabled. The other implementations being HCL/JSON, both private to the hcp sub-package. The api (previously `registry') is the set of functionality that abstracts and calls the HCP API. It was meant to be merged with the `hcp' package, but because of a dependency loop with the datasources, both are separated for now.
The ancestry inferral code relied on HCP datasource outputs for deciding what would be the ancestry for an image built with HCP. This brings a dependency into the hcp package, which is not really necessary as we only need a few information from those entities, hence this commit removes the full dependency on the structures, in favour of more focused data structures where we only cherry pick what is necessary for this code.
c074808
to
207e099
Compare
I'm going to lock this pull request because it has been closed for 30 days ⏳. This helps our maintainers find and focus on the active issues. |
To be able to track HCP builds during a Packer build, we need to propagate the name used for registering the build to HCP.
This works out-of-the-box for JSON templates, as the build's name is always the builder's, so there's no confusion on that.
For HCP templates however, there's a possibility for a template to define a name for the build, which is returned by a CoreBuild's Name() function, in addition to the source's Type/Name. The builds are registered to HCP with the source.String() function however, which does not contain the build's name. This prevents any build containing a name to work, as their name/source.String does not match.
The name we registered the build with is stored in a CoreBuild as the Type attribute, but cannot be safely accessed, as builds are types within the command as packersdk.Build, which only exposes Name() for this purpose.
In order to circumvent this problem, and as a way to present a prototype of solution that will likely have to be discussed before we merge it, this commit hotfixes the issue.