2023-08-07 13:14:25 +00:00
|
|
|
package template
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-06-03 12:39:36 +00:00
|
|
|
"errors"
|
2023-11-22 12:25:16 +00:00
|
|
|
"fmt"
|
2023-08-25 09:03:42 +00:00
|
|
|
"io/fs"
|
2023-08-30 14:01:08 +00:00
|
|
|
|
|
|
|
"github.com/databricks/cli/libs/cmdio"
|
2024-11-20 10:11:31 +00:00
|
|
|
"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"
|
2023-08-07 13:14:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
libraryDirName = "library"
|
|
|
|
templateDirName = "template"
|
|
|
|
schemaFileName = "databricks_template_schema.json"
|
2024-12-12 09:28:42 +00:00
|
|
|
)
|
2023-08-07 13:14:25 +00:00
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
type Template struct {
|
|
|
|
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.
|
|
|
|
func (t *Template) resolveTemplateInput(ctx context.Context) error {
|
|
|
|
if _, err := fs.Stat(t.TemplateFS, schemaFileName); errors.Is(err, fs.ErrNotExist) {
|
2024-11-20 09:28:35 +00:00
|
|
|
return fmt.Errorf("not a bundle template: expected to find a template schema file at %s", schemaFileName)
|
2023-11-22 12:25:16 +00:00
|
|
|
}
|
|
|
|
|
2024-12-27 06:05:04 +00:00
|
|
|
var err error
|
|
|
|
t.config, err = newConfig(ctx, t.TemplateFS, schemaFileName)
|
2023-08-07 13:14:25 +00:00
|
|
|
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)
|
2023-08-07 13:14:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-20 09:28:35 +00:00
|
|
|
helpers := loadHelpers(ctx)
|
2024-12-27 06:05:04 +00:00
|
|
|
t.renderer, err = newRenderer(ctx, t.config.values, helpers, t.TemplateFS, templateDirName, libraryDirName)
|
2023-10-19 07:08:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-06-04 08:57:13 +00:00
|
|
|
// Print welcome message
|
2024-12-27 06:05:04 +00:00
|
|
|
welcome := t.config.schema.WelcomeMessage
|
2024-06-04 08:57:13 +00:00
|
|
|
if welcome != "" {
|
2024-12-27 06:05:04 +00:00
|
|
|
welcome, err = t.renderer.executeTemplate(welcome)
|
2024-06-04 08:57:13 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cmdio.LogString(ctx, welcome)
|
|
|
|
}
|
|
|
|
|
2023-08-07 13:14:25 +00:00
|
|
|
// 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)
|
2023-08-07 13:14:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-12-27 06:05:04 +00:00
|
|
|
return t.config.validate()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Template) printSuccessMessage(ctx context.Context) error {
|
|
|
|
success := t.config.schema.SuccessMessage
|
|
|
|
if success == "" {
|
|
|
|
cmdio.LogString(ctx, "✨ Successfully initialized template")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
success, err := t.renderer.executeTemplate(success)
|
2023-08-07 13:14:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-12-27 06:05:04 +00:00
|
|
|
cmdio.LogString(ctx, success)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Template) logTelemetry(ctx context.Context) error {
|
|
|
|
// 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.
|
2024-12-27 06:05:04 +00:00
|
|
|
templateEnumArgs := map[string]string{}
|
2024-12-27 07:57:24 +00:00
|
|
|
if t.IsDatabricksOwned {
|
2024-12-27 06:05:04 +00:00
|
|
|
templateEnumArgs = t.config.enumValues()
|
2024-12-27 09:02:42 +00:00
|
|
|
} else {
|
|
|
|
t.Name = "custom"
|
2024-12-27 06:05:04 +00:00
|
|
|
}
|
2023-08-07 13:14:25 +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:50:14 +00:00
|
|
|
return telemetry.Log(ctx, event)
|
2024-12-27 06:05:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This function materializes the input templates as a project, using user defined
|
|
|
|
// configurations.
|
|
|
|
func (t *Template) Materialize(ctx context.Context) error {
|
|
|
|
err := t.resolveTemplateInput(ctx)
|
2023-08-07 13:14:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-08-25 09:03:42 +00:00
|
|
|
|
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()
|
2023-08-25 09:03:42 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-10-19 07:08:36 +00:00
|
|
|
|
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
|
2023-10-19 07:08:36 +00:00
|
|
|
}
|
2024-12-27 06:05:04 +00:00
|
|
|
|
|
|
|
err = t.printSuccessMessage(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return t.logTelemetry(ctx)
|
2023-08-25 09:03:42 +00:00
|
|
|
}
|