mirror of
https://github.com/opentofu/opentofu.git
synced 2025-02-25 18:45:20 -06:00
The previous implementation of views was copying and embedding the base View struct in each individual view. While this allowed for easy access to the interface of that struct (both in the view and externally), it more importantly completely broke the ability of the diagnostic printer to output source code snippets. This is because the `configSources` field on the base view is lazily set after the config loader is initialized. In the commands ported to use views, this happens after the base View struct is copied, so we are updating the wrong copy of the struct. This commit fixes this with a simple mechanical refactor: keep a pointer to the base View struct instead, and update all of the individual views to explicitly refer to that struct to access its fields and methods. This is not a particularly satisfying solution, but I can't find anything clearly better. It might be worth exploring the alternative approach in the view for the new test command, which explicitly pulls its dependencies out of the base view, rather than retaining a full reference. Maybe there's a third way which is better still.
74 lines
1.7 KiB
Go
74 lines
1.7 KiB
Go
package views
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/hashicorp/terraform/command/arguments"
|
|
"github.com/hashicorp/terraform/states"
|
|
"github.com/hashicorp/terraform/terraform"
|
|
"github.com/hashicorp/terraform/tfdiags"
|
|
)
|
|
|
|
// The Refresh view is used for the refresh command.
|
|
type Refresh interface {
|
|
Outputs(outputValues map[string]*states.OutputValue)
|
|
|
|
Operation() Operation
|
|
Hooks() []terraform.Hook
|
|
|
|
Diagnostics(diags tfdiags.Diagnostics)
|
|
HelpPrompt()
|
|
}
|
|
|
|
// NewRefresh returns an initialized Refresh implementation for the given ViewType.
|
|
func NewRefresh(vt arguments.ViewType, runningInAutomation bool, view *View) Refresh {
|
|
switch vt {
|
|
case arguments.ViewHuman:
|
|
return &RefreshHuman{
|
|
view: view,
|
|
inAutomation: runningInAutomation,
|
|
countHook: &countHook{},
|
|
}
|
|
default:
|
|
panic(fmt.Sprintf("unknown view type %v", vt))
|
|
}
|
|
}
|
|
|
|
// The RefreshHuman implementation renders human-readable text logs, suitable for
|
|
// a scrolling terminal.
|
|
type RefreshHuman struct {
|
|
view *View
|
|
|
|
inAutomation bool
|
|
|
|
countHook *countHook
|
|
}
|
|
|
|
var _ Refresh = (*RefreshHuman)(nil)
|
|
|
|
func (v *RefreshHuman) Outputs(outputValues map[string]*states.OutputValue) {
|
|
if len(outputValues) > 0 {
|
|
v.view.streams.Print(v.view.colorize.Color("[reset][bold][green]\nOutputs:\n\n"))
|
|
NewOutput(arguments.ViewHuman, v.view).Output("", outputValues)
|
|
}
|
|
}
|
|
|
|
func (v *RefreshHuman) Operation() Operation {
|
|
return NewOperation(arguments.ViewHuman, v.inAutomation, v.view)
|
|
}
|
|
|
|
func (v *RefreshHuman) Hooks() []terraform.Hook {
|
|
return []terraform.Hook{
|
|
v.countHook,
|
|
NewUiHook(v.view),
|
|
}
|
|
}
|
|
|
|
func (v *RefreshHuman) Diagnostics(diags tfdiags.Diagnostics) {
|
|
v.view.Diagnostics(diags)
|
|
}
|
|
|
|
func (v *RefreshHuman) HelpPrompt() {
|
|
v.view.HelpPrompt("refresh")
|
|
}
|