databricks-cli/libs/template/materialize.go

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

158 lines
4.0 KiB
Go
Raw Normal View History

package template
import (
"context"
"errors"
"fmt"
"io/fs"
"github.com/databricks/cli/libs/cmdio"
"github.com/databricks/cli/libs/filer"
2024-12-27 06:05:04 +00:00
"github.com/databricks/cli/libs/telemetry"
"github.com/databricks/cli/libs/telemetry/events"
)
2024-12-27 06:05:04 +00:00
type TemplateOpts struct {
// file path containing user defined config values
ConfigFilePath string
// root of the template definition
TemplateFS fs.FS
// filer to use for writing the initialized template
OutputFiler filer.Filer
2024-12-27 09:02:42 +00:00
// If true, we'll include the enum template args in the telemetry payload.
2024-12-27 06:05:04 +00:00
IsDatabricksOwned bool
// Name of the template. For non-Databricks owned templates, this is set to
2024-12-27 09:02:42 +00:00
// "custom".
2024-12-27 06:05:04 +00:00
Name string
}
2025-01-03 10:43:53 +00:00
type TemplateX struct {
2024-12-27 06:05:04 +00:00
TemplateOpts
// internal object used to prompt user for config values and store them.
config *config
// internal object user to render the template.
renderer *renderer
}
// This function resolves input to use to materialize the template in two steps.
// 1. First, this function loads any user specified input configuration if the user
// has provided a config file path.
// 2. For any values that are required by the template but not provided in the config
// file, this function prompts the user for them.
2025-01-03 10:43:53 +00:00
func (t *TemplateX) resolveTemplateInput(ctx context.Context) error {
2024-12-27 06:05:04 +00:00
if _, err := fs.Stat(t.TemplateFS, schemaFileName); errors.Is(err, fs.ErrNotExist) {
return fmt.Errorf("not a bundle template: expected to find a template schema file at %s", schemaFileName)
}
2024-12-27 06:05:04 +00:00
var err error
t.config, err = newConfig(ctx, t.TemplateFS, schemaFileName)
if err != nil {
return err
}
// Read and assign config values from file
2024-12-27 06:05:04 +00:00
if t.ConfigFilePath != "" {
err = t.config.assignValuesFromFile(t.ConfigFilePath)
if err != nil {
return err
}
}
helpers := loadHelpers(ctx)
2024-12-27 06:05:04 +00:00
t.renderer, err = newRenderer(ctx, t.config.values, helpers, t.TemplateFS, templateDirName, libraryDirName)
if err != nil {
return err
}
// Print welcome message
2024-12-27 06:05:04 +00:00
welcome := t.config.schema.WelcomeMessage
if welcome != "" {
2024-12-27 06:05:04 +00:00
welcome, err = t.renderer.executeTemplate(welcome)
if err != nil {
return err
}
cmdio.LogString(ctx, welcome)
}
// Prompt user for any missing config values. Assign default values if
// terminal is not TTY
2024-12-27 06:05:04 +00:00
err = t.config.promptOrAssignDefaultValues(t.renderer)
if err != nil {
return err
}
2024-12-27 06:05:04 +00:00
return t.config.validate()
}
2025-01-03 10:43:53 +00:00
func (t *TemplateX) printSuccessMessage(ctx context.Context) error {
2024-12-27 06:05:04 +00:00
success := t.config.schema.SuccessMessage
if success == "" {
cmdio.LogString(ctx, "✨ Successfully initialized template")
return nil
}
success, err := t.renderer.executeTemplate(success)
if err != nil {
return err
}
2024-12-27 06:05:04 +00:00
cmdio.LogString(ctx, success)
return nil
}
2025-01-03 10:43:53 +00:00
func (t *TemplateX) logTelemetry(ctx context.Context) error {
2024-12-27 06:05:04 +00:00
// Only log telemetry input for Databricks owned templates. This is to prevent
2024-12-27 09:02:42 +00:00
// accidentally collecting PII from custom user templates.
2025-01-02 09:04:28 +00:00
templateEnumArgs := []events.BundleInitTemplateEnumArg{}
2024-12-27 07:57:24 +00:00
if t.IsDatabricksOwned {
2025-01-02 09:04:28 +00:00
for k, v := range t.config.enumValues() {
templateEnumArgs = append(templateEnumArgs, events.BundleInitTemplateEnumArg{
Key: k,
Value: v,
})
}
2024-12-27 09:02:42 +00:00
} else {
t.Name = "custom"
2024-12-27 06:05:04 +00:00
}
2024-12-27 06:05:04 +00:00
event := telemetry.DatabricksCliLog{
BundleInitEvent: &events.BundleInitEvent{
Uuid: bundleUuid,
TemplateName: t.Name,
TemplateEnumArgs: templateEnumArgs,
},
}
2024-12-30 06:59:18 +00:00
telemetry.Log(ctx, event)
return nil
2024-12-27 06:05:04 +00:00
}
// This function materializes the input templates as a project, using user defined
// configurations.
2025-01-03 10:43:53 +00:00
func (t *TemplateX) Materialize(ctx context.Context) error {
2024-12-27 06:05:04 +00:00
err := t.resolveTemplateInput(ctx)
if err != nil {
return err
}
2024-12-27 06:05:04 +00:00
// Walk the template file tree and compute in-memory representations of the
// output files.
err = t.renderer.walk()
if err != nil {
return err
}
2024-12-27 06:05:04 +00:00
// Flush the output files to disk.
err = t.renderer.persistToDisk(ctx, t.OutputFiler)
if err != nil {
return err
}
2024-12-27 06:05:04 +00:00
err = t.printSuccessMessage(ctx)
if err != nil {
return err
}
return t.logTelemetry(ctx)
}