opentofu/repl/session.go

100 lines
3.0 KiB
Go
Raw Normal View History

2016-11-14 00:04:21 -06:00
package repl
import (
"errors"
"fmt"
"strings"
"github.com/zclconf/go-cty/cty"
"github.com/hashicorp/hcl2/hcl"
"github.com/hashicorp/hcl2/hcl/hclsyntax"
2019-08-06 18:58:58 -05:00
"github.com/hashicorp/terraform/configs/hcl2shim"
"github.com/hashicorp/terraform/lang"
"github.com/hashicorp/terraform/tfdiags"
2016-11-14 00:04:21 -06:00
)
// ErrSessionExit is a special error result that should be checked for
// from Handle to signal a graceful exit.
var ErrSessionExit = errors.New("session exit")
// Session represents the state for a single REPL session.
type Session struct {
// Scope is the evaluation scope where expressions will be evaluated.
Scope *lang.Scope
2016-11-14 00:04:21 -06:00
}
// Handle handles a single line of input from the REPL.
//
// This is a stateful operation if a command is given (such as setting
// a variable). This function should not be called in parallel.
//
// The return value is the output and the error to show.
func (s *Session) Handle(line string) (string, bool, tfdiags.Diagnostics) {
2016-11-14 00:04:21 -06:00
switch {
case strings.TrimSpace(line) == "":
return "", false, nil
2016-11-14 00:04:21 -06:00
case strings.TrimSpace(line) == "exit":
return "", true, nil
2016-11-14 00:04:21 -06:00
case strings.TrimSpace(line) == "help":
ret, diags := s.handleHelp()
return ret, false, diags
2016-11-14 00:04:21 -06:00
default:
ret, diags := s.handleEval(line)
return ret, false, diags
2016-11-14 00:04:21 -06:00
}
}
func (s *Session) handleEval(line string) (string, tfdiags.Diagnostics) {
var diags tfdiags.Diagnostics
2016-11-14 00:04:21 -06:00
// Parse the given line as an expression
expr, parseDiags := hclsyntax.ParseExpression([]byte(line), "<console-input>", hcl.Pos{Line: 1, Column: 1})
diags = diags.Append(parseDiags)
if parseDiags.HasErrors() {
return "", diags
2016-11-14 00:04:21 -06:00
}
val, valDiags := s.Scope.EvalExpr(expr, cty.DynamicPseudoType)
diags = diags.Append(valDiags)
if valDiags.HasErrors() {
return "", diags
2016-11-14 00:04:21 -06:00
}
if !val.IsWhollyKnown() {
// FIXME: In future, once we've updated the result formatter to be
// cty-aware, we should just include unknown values as "(not yet known)"
// in the serialized result, allowing the rest (if any) to be seen.
diags = diags.Append(fmt.Errorf("Result depends on values that cannot be determined until after \"terraform apply\"."))
return "", diags
2016-11-14 00:04:21 -06:00
}
// Our formatter still wants an old-style raw interface{} value, so
// for now we'll just shim it.
// FIXME: Port the formatter to work with cty.Value directly.
legacyVal := hcl2shim.ConfigValueFromHCL2(val)
result, err := FormatResult(legacyVal)
2016-11-14 00:04:21 -06:00
if err != nil {
diags = diags.Append(err)
return "", diags
2016-11-14 00:04:21 -06:00
}
return result, diags
2016-11-14 00:04:21 -06:00
}
func (s *Session) handleHelp() (string, tfdiags.Diagnostics) {
2016-11-14 00:04:21 -06:00
text := `
The Terraform console allows you to experiment with Terraform interpolations.
You may access resources in the state (if you have one) just as you would
from a configuration. For example: "aws_instance.foo.id" would evaluate
to the ID of "aws_instance.foo" if it exists in your state.
Type in the interpolation to test and hit <enter> to see the result.
To exit the console, type "exit" and hit <enter>, or use Control-C or
Control-D.
`
return strings.TrimSpace(text), nil
}