2024-07-01 09:01:10 +00:00
|
|
|
package render
|
|
|
|
|
|
|
|
import (
|
2024-10-18 06:45:47 +00:00
|
|
|
"context"
|
2024-07-01 09:01:10 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"path/filepath"
|
2024-10-18 06:45:47 +00:00
|
|
|
"sort"
|
2024-07-01 09:01:10 +00:00
|
|
|
"strings"
|
|
|
|
"text/template"
|
|
|
|
|
|
|
|
"github.com/databricks/cli/bundle"
|
2024-10-11 14:39:53 +00:00
|
|
|
"github.com/databricks/cli/libs/cmdio"
|
2024-07-01 09:01:10 +00:00
|
|
|
"github.com/databricks/cli/libs/diag"
|
|
|
|
"github.com/databricks/databricks-sdk-go/service/iam"
|
|
|
|
"github.com/fatih/color"
|
|
|
|
)
|
|
|
|
|
|
|
|
var renderFuncMap = template.FuncMap{
|
|
|
|
"red": color.RedString,
|
|
|
|
"green": color.GreenString,
|
|
|
|
"blue": color.BlueString,
|
|
|
|
"yellow": color.YellowString,
|
|
|
|
"magenta": color.MagentaString,
|
|
|
|
"cyan": color.CyanString,
|
2024-12-05 15:37:24 +00:00
|
|
|
"bold": func(format string, a ...any) string {
|
2024-07-01 09:01:10 +00:00
|
|
|
return color.New(color.Bold).Sprintf(format, a...)
|
|
|
|
},
|
2024-12-05 15:37:24 +00:00
|
|
|
"italic": func(format string, a ...any) string {
|
2024-07-01 09:01:10 +00:00
|
|
|
return color.New(color.Italic).Sprintf(format, a...)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2024-10-18 06:45:47 +00:00
|
|
|
const summaryHeaderTemplate = `{{- if .Name -}}
|
2024-07-01 09:01:10 +00:00
|
|
|
Name: {{ .Name | bold }}
|
|
|
|
{{- if .Target }}
|
|
|
|
Target: {{ .Target | bold }}
|
|
|
|
{{- end }}
|
|
|
|
{{- if or .User .Host .Path }}
|
|
|
|
Workspace:
|
|
|
|
{{- if .Host }}
|
|
|
|
Host: {{ .Host | bold }}
|
|
|
|
{{- end }}
|
|
|
|
{{- if .User }}
|
|
|
|
User: {{ .User | bold }}
|
|
|
|
{{- end }}
|
|
|
|
{{- if .Path }}
|
|
|
|
Path: {{ .Path | bold }}
|
|
|
|
{{- end }}
|
|
|
|
{{- end }}
|
2024-10-18 06:45:47 +00:00
|
|
|
{{ end -}}`
|
|
|
|
|
|
|
|
const resourcesTemplate = `Resources:
|
|
|
|
{{- range . }}
|
|
|
|
{{ .GroupName }}:
|
|
|
|
{{- range .Resources }}
|
|
|
|
{{ .Key | bold }}:
|
|
|
|
Name: {{ .Name }}
|
|
|
|
URL: {{ if .URL }}{{ .URL | cyan }}{{ else }}{{ "(not deployed)" | cyan }}{{ end }}
|
|
|
|
{{- end }}
|
|
|
|
{{- end }}
|
|
|
|
`
|
2024-07-01 09:01:10 +00:00
|
|
|
|
2024-10-18 06:45:47 +00:00
|
|
|
type ResourceGroup struct {
|
|
|
|
GroupName string
|
|
|
|
Resources []ResourceInfo
|
|
|
|
}
|
2024-07-01 09:01:10 +00:00
|
|
|
|
2024-10-18 06:45:47 +00:00
|
|
|
type ResourceInfo struct {
|
|
|
|
Key string
|
|
|
|
Name string
|
|
|
|
URL string
|
|
|
|
}
|
2024-07-01 09:01:10 +00:00
|
|
|
|
|
|
|
func pluralize(n int, singular, plural string) string {
|
|
|
|
if n == 1 {
|
|
|
|
return fmt.Sprintf("%d %s", n, singular)
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%d %s", n, plural)
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildTrailer(diags diag.Diagnostics) string {
|
|
|
|
parts := []string{}
|
|
|
|
if errors := len(diags.Filter(diag.Error)); errors > 0 {
|
|
|
|
parts = append(parts, color.RedString(pluralize(errors, "error", "errors")))
|
|
|
|
}
|
|
|
|
if warnings := len(diags.Filter(diag.Warning)); warnings > 0 {
|
|
|
|
parts = append(parts, color.YellowString(pluralize(warnings, "warning", "warnings")))
|
|
|
|
}
|
2024-10-07 09:16:20 +00:00
|
|
|
if recommendations := len(diags.Filter(diag.Recommendation)); recommendations > 0 {
|
|
|
|
parts = append(parts, color.BlueString(pluralize(recommendations, "recommendation", "recommendations")))
|
|
|
|
}
|
|
|
|
switch {
|
|
|
|
case len(parts) >= 3:
|
|
|
|
first := strings.Join(parts[:len(parts)-1], ", ")
|
|
|
|
last := parts[len(parts)-1]
|
2024-10-18 06:45:47 +00:00
|
|
|
return fmt.Sprintf("Found %s, and %s\n", first, last)
|
2024-10-07 09:16:20 +00:00
|
|
|
case len(parts) == 2:
|
2024-10-18 06:45:47 +00:00
|
|
|
return fmt.Sprintf("Found %s and %s\n", parts[0], parts[1])
|
2024-10-07 09:16:20 +00:00
|
|
|
case len(parts) == 1:
|
2024-10-18 06:45:47 +00:00
|
|
|
return fmt.Sprintf("Found %s\n", parts[0])
|
2024-10-07 09:16:20 +00:00
|
|
|
default:
|
|
|
|
// No diagnostics to print.
|
2024-10-18 06:45:47 +00:00
|
|
|
return color.GreenString("Validation OK!\n")
|
2024-07-01 09:01:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-18 06:45:47 +00:00
|
|
|
func renderSummaryHeaderTemplate(out io.Writer, b *bundle.Bundle) error {
|
2024-07-01 09:01:10 +00:00
|
|
|
if b == nil {
|
2024-10-18 06:45:47 +00:00
|
|
|
return renderSummaryHeaderTemplate(out, &bundle.Bundle{})
|
2024-07-01 09:01:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
currentUser := &iam.User{}
|
|
|
|
|
|
|
|
if b.Config.Workspace.CurrentUser != nil {
|
|
|
|
if b.Config.Workspace.CurrentUser.User != nil {
|
|
|
|
currentUser = b.Config.Workspace.CurrentUser.User
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-18 06:45:47 +00:00
|
|
|
t := template.Must(template.New("summary").Funcs(renderFuncMap).Parse(summaryHeaderTemplate))
|
2024-07-01 09:01:10 +00:00
|
|
|
err := t.Execute(out, map[string]any{
|
2024-10-18 06:45:47 +00:00
|
|
|
"Name": b.Config.Bundle.Name,
|
|
|
|
"Target": b.Config.Bundle.Target,
|
|
|
|
"User": currentUser.UserName,
|
|
|
|
"Path": b.Config.Workspace.RootPath,
|
|
|
|
"Host": b.Config.Workspace.Host,
|
2024-07-01 09:01:10 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-10-18 06:45:47 +00:00
|
|
|
func renderDiagnosticsOnly(out io.Writer, b *bundle.Bundle, diags diag.Diagnostics) error {
|
2024-07-01 09:01:10 +00:00
|
|
|
for _, d := range diags {
|
2024-07-23 17:20:11 +00:00
|
|
|
for i := range d.Locations {
|
|
|
|
if b == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make location relative to bundle root
|
|
|
|
if d.Locations[i].File != "" {
|
2024-09-27 10:03:05 +00:00
|
|
|
out, err := filepath.Rel(b.BundleRootPath, d.Locations[i].File)
|
2024-07-23 17:20:11 +00:00
|
|
|
// if we can't relativize the path, just use path as-is
|
|
|
|
if err == nil {
|
|
|
|
d.Locations[i].File = out
|
|
|
|
}
|
2024-07-01 09:01:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-11 14:39:53 +00:00
|
|
|
return cmdio.RenderDiagnostics(out, diags)
|
2024-07-01 09:01:10 +00:00
|
|
|
}
|
|
|
|
|
2024-07-10 11:14:57 +00:00
|
|
|
// RenderOptions contains options for rendering diagnostics.
|
|
|
|
type RenderOptions struct {
|
|
|
|
// variable to include leading new line
|
|
|
|
|
|
|
|
RenderSummaryTable bool
|
|
|
|
}
|
|
|
|
|
2024-10-18 06:45:47 +00:00
|
|
|
// RenderDiagnostics renders the diagnostics in a human-readable format.
|
|
|
|
func RenderDiagnostics(out io.Writer, b *bundle.Bundle, diags diag.Diagnostics, opts RenderOptions) error {
|
|
|
|
err := renderDiagnosticsOnly(out, b, diags)
|
2024-07-01 09:01:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to render diagnostics: %w", err)
|
|
|
|
}
|
|
|
|
|
2024-07-10 11:14:57 +00:00
|
|
|
if opts.RenderSummaryTable {
|
2024-10-18 06:45:47 +00:00
|
|
|
if b != nil {
|
|
|
|
err = renderSummaryHeaderTemplate(out, b)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to render summary: %w", err)
|
|
|
|
}
|
|
|
|
_, err = io.WriteString(out, "\n")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2024-12-11 12:26:00 +00:00
|
|
|
}
|
2024-07-10 11:14:57 +00:00
|
|
|
}
|
2024-10-18 06:45:47 +00:00
|
|
|
trailer := buildTrailer(diags)
|
|
|
|
_, err = io.WriteString(out, trailer)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2024-12-11 12:26:00 +00:00
|
|
|
}
|
2024-07-01 09:01:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2024-10-18 06:45:47 +00:00
|
|
|
|
|
|
|
func RenderSummary(ctx context.Context, out io.Writer, b *bundle.Bundle) error {
|
|
|
|
if err := renderSummaryHeaderTemplate(out, b); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var resourceGroups []ResourceGroup
|
|
|
|
|
|
|
|
for _, group := range b.Config.Resources.AllResources() {
|
|
|
|
resources := make([]ResourceInfo, 0, len(group.Resources))
|
|
|
|
for key, resource := range group.Resources {
|
|
|
|
resources = append(resources, ResourceInfo{
|
|
|
|
Key: key,
|
|
|
|
Name: resource.GetName(),
|
|
|
|
URL: resource.GetURL(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(resources) > 0 {
|
|
|
|
resourceGroups = append(resourceGroups, ResourceGroup{
|
|
|
|
GroupName: group.Description.PluralTitle,
|
|
|
|
Resources: resources,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := renderResourcesTemplate(out, resourceGroups); err != nil {
|
|
|
|
return fmt.Errorf("failed to render resources template: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper function to sort and render resource groups using the template
|
|
|
|
func renderResourcesTemplate(out io.Writer, resourceGroups []ResourceGroup) error {
|
|
|
|
// Sort everything to ensure consistent output
|
|
|
|
sort.Slice(resourceGroups, func(i, j int) bool {
|
|
|
|
return resourceGroups[i].GroupName < resourceGroups[j].GroupName
|
|
|
|
})
|
|
|
|
for _, group := range resourceGroups {
|
|
|
|
sort.Slice(group.Resources, func(i, j int) bool {
|
|
|
|
return group.Resources[i].Key < group.Resources[j].Key
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
t := template.Must(template.New("resources").Funcs(renderFuncMap).Parse(resourcesTemplate))
|
|
|
|
|
|
|
|
return t.Execute(out, resourceGroups)
|
|
|
|
}
|