Skip to content

Commit

Permalink
add addresses to diagnostics
Browse files Browse the repository at this point in the history
Add an address argument to tfdiags.InConfigBody, and store the address
string the diagnostics details. Since nearly every place where we want
to annotate the diagnostics with the config context we also have some
sort of address, we can use the same call to insert them both into the
diagnostic.

Perhaps we should rename InConfigBody and ElaborateFromConfigBody to
reflect the additional address parameter, but for now we can verify this
is a pattern that suits us.
  • Loading branch information
jbardin committed Apr 6, 2021
1 parent 973ed6e commit 503c413
Show file tree
Hide file tree
Showing 4 changed files with 38 additions and 8 deletions.
27 changes: 20 additions & 7 deletions tfdiags/contextual.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,25 +19,28 @@ import (

// contextualFromConfig is an interface type implemented by diagnostic types
// that can elaborate themselves when given information about the configuration
// body they are embedded in.
// body they are embedded in, as well as the runtime address associated with
// that configuration.
//
// Usually this entails extracting source location information in order to
// populate the "Subject" range.
type contextualFromConfigBody interface {
ElaborateFromConfigBody(hcl.Body) Diagnostic
ElaborateFromConfigBody(hcl.Body, string) Diagnostic
}

// InConfigBody returns a copy of the receiver with any config-contextual
// diagnostics elaborated in the context of the given body.
func (diags Diagnostics) InConfigBody(body hcl.Body) Diagnostics {
// diagnostics elaborated in the context of the given body. An optional address
// argument may be added to indicate which instance of the configuration the
// error related to.
func (diags Diagnostics) InConfigBody(body hcl.Body, addr string) Diagnostics {
if len(diags) == 0 {
return nil
}

ret := make(Diagnostics, len(diags))
for i, srcDiag := range diags {
if cd, isCD := srcDiag.(contextualFromConfigBody); isCD {
ret[i] = cd.ElaborateFromConfigBody(body)
ret[i] = cd.ElaborateFromConfigBody(body, addr)
} else {
ret[i] = srcDiag
}
Expand Down Expand Up @@ -112,7 +115,12 @@ type attributeDiagnostic struct {
// source location information is still available, for more accuracy. This
// is not always possible due to system architecture, so this serves as a
// "best effort" fallback behavior for such situations.
func (d *attributeDiagnostic) ElaborateFromConfigBody(body hcl.Body) Diagnostic {
func (d *attributeDiagnostic) ElaborateFromConfigBody(body hcl.Body, addr string) Diagnostic {
// don't change an existing address
if d.address == "" {
d.address = addr
}

if len(d.attrPath) < 1 {
// Should never happen, but we'll allow it rather than crashing.
return d
Expand Down Expand Up @@ -353,7 +361,12 @@ type wholeBodyDiagnostic struct {
subject *SourceRange // populated only after ElaborateFromConfigBody
}

func (d *wholeBodyDiagnostic) ElaborateFromConfigBody(body hcl.Body) Diagnostic {
func (d *wholeBodyDiagnostic) ElaborateFromConfigBody(body hcl.Body, addr string) Diagnostic {
// don't change an existing address
if d.address == "" {
d.address = addr
}

if d.subject != nil {
// Don't modify an already-elaborated diagnostic.
return d
Expand Down
16 changes: 15 additions & 1 deletion tfdiags/contextual_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -185,6 +185,7 @@ simple_attr = "val"
diagnosticBase: diagnosticBase{
summary: "preexisting",
detail: "detail",
address: "original",
},
subject: &SourceRange{
Filename: "somewhere_else.tf",
Expand Down Expand Up @@ -535,9 +536,22 @@ simple_attr = "val"
for i, tc := range testCases {
t.Run(fmt.Sprintf("%d:%s", i, tc.Diag.Description()), func(t *testing.T) {
var diags Diagnostics

origAddr := tc.Diag.Description().Address
diags = diags.Append(tc.Diag)
gotDiags := diags.InConfigBody(f.Body)

gotDiags := diags.InConfigBody(f.Body, "test.addr")
gotRange := gotDiags[0].Source().Subject
gotAddr := gotDiags[0].Description().Address

switch {
case origAddr != "":
if gotAddr != origAddr {
t.Errorf("original diagnostic address modified from %s to %s", origAddr, gotAddr)
}
case gotAddr != "test.addr":
t.Error("missing detail address")
}

for _, problem := range deep.Equal(gotRange, tc.ExpectedRange) {
t.Error(problem)
Expand Down
1 change: 1 addition & 0 deletions tfdiags/diagnostic.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ const (
)

type Description struct {
Address string
Summary string
Detail string
}
Expand Down
2 changes: 2 additions & 0 deletions tfdiags/diagnostic_base.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ type diagnosticBase struct {
severity Severity
summary string
detail string
address string
}

func (d diagnosticBase) Severity() Severity {
Expand All @@ -19,6 +20,7 @@ func (d diagnosticBase) Description() Description {
return Description{
Summary: d.summary,
Detail: d.detail,
Address: d.address,
}
}

Expand Down

0 comments on commit 503c413

Please sign in to comment.