github.com/paultyng/terraform@v0.6.11-0.20180227224804-66ff8f8bed40/command/format/diagnostic.go (about) 1 package format 2 3 import ( 4 "bytes" 5 "fmt" 6 7 "github.com/hashicorp/terraform/tfdiags" 8 "github.com/mitchellh/colorstring" 9 wordwrap "github.com/mitchellh/go-wordwrap" 10 ) 11 12 // Diagnostic formats a single diagnostic message. 13 // 14 // The width argument specifies at what column the diagnostic messages will 15 // be wrapped. If set to zero, messages will not be wrapped by this function 16 // at all. Although the long-form text parts of the message are wrapped, 17 // not all aspects of the message are guaranteed to fit within the specified 18 // terminal width. 19 func Diagnostic(diag tfdiags.Diagnostic, color *colorstring.Colorize, width int) string { 20 if diag == nil { 21 // No good reason to pass a nil diagnostic in here... 22 return "" 23 } 24 25 var buf bytes.Buffer 26 27 switch diag.Severity() { 28 case tfdiags.Error: 29 buf.WriteString(color.Color("\n[bold][red]Error: [reset]")) 30 case tfdiags.Warning: 31 buf.WriteString(color.Color("\n[bold][yellow]Warning: [reset]")) 32 default: 33 // Clear out any coloring that might be applied by Terraform's UI helper, 34 // so our result is not context-sensitive. 35 buf.WriteString(color.Color("\n[reset]")) 36 } 37 38 desc := diag.Description() 39 sourceRefs := diag.Source() 40 41 // We don't wrap the summary, since we expect it to be terse, and since 42 // this is where we put the text of a native Go error it may not always 43 // be pure text that lends itself well to word-wrapping. 44 if sourceRefs.Subject != nil { 45 fmt.Fprintf(&buf, color.Color("[bold]%s[reset] at %s\n\n"), desc.Summary, sourceRefs.Subject.StartString()) 46 } else { 47 fmt.Fprintf(&buf, color.Color("[bold]%s[reset]\n\n"), desc.Summary) 48 } 49 50 // TODO: also print out the relevant snippet of config source with the 51 // relevant section highlighted, so the user doesn't need to manually 52 // correlate back to config. Before we can do this, the HCL2 parser 53 // needs to be more deeply integrated so that we can use it to obtain 54 // the parsed source code and AST. 55 56 if desc.Detail != "" { 57 detail := desc.Detail 58 if width != 0 { 59 detail = wordwrap.WrapString(detail, uint(width)) 60 } 61 fmt.Fprintf(&buf, "%s\n", detail) 62 } 63 64 return buf.String() 65 }