mirror of https://github.com/databricks/cli.git
116 lines
3.3 KiB
Go
116 lines
3.3 KiB
Go
package trampoline
|
|
|
|
import (
|
|
"context"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/databricks/cli/bundle"
|
|
"github.com/databricks/cli/bundle/config"
|
|
"github.com/databricks/cli/bundle/libraries"
|
|
"github.com/databricks/cli/libs/diag"
|
|
"github.com/databricks/cli/libs/log"
|
|
"github.com/databricks/databricks-sdk-go"
|
|
"golang.org/x/mod/semver"
|
|
)
|
|
|
|
type wrapperWarning struct{}
|
|
|
|
func WrapperWarning() bundle.Mutator {
|
|
return &wrapperWarning{}
|
|
}
|
|
|
|
func (m *wrapperWarning) Apply(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
|
if isPythonWheelWrapperOn(b) {
|
|
if config.IsExplicitlyEnabled(b.Config.Presets.SourceLinkedDeployment) {
|
|
return diag.Warningf("Python wheel notebook wrapper is not available when using source-linked deployment mode. You can disable this mode by setting 'presets.source_linked_deployment: false'")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
if hasIncompatibleWheelTasks(ctx, b) {
|
|
return diag.Errorf("Python wheel tasks require compute with DBR 13.3+ to include local libraries. Please change your cluster configuration or use the experimental 'python_wheel_wrapper' setting. See https://docs.databricks.com/dev-tools/bundles/python-wheel.html for more information.")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func isPythonWheelWrapperOn(b *bundle.Bundle) bool {
|
|
return b.Config.Experimental != nil && b.Config.Experimental.PythonWheelWrapper
|
|
}
|
|
|
|
func hasIncompatibleWheelTasks(ctx context.Context, b *bundle.Bundle) bool {
|
|
tasks := libraries.FindTasksWithLocalLibraries(b)
|
|
for _, task := range tasks {
|
|
if task.NewCluster != nil {
|
|
if lowerThanExpectedVersion(task.NewCluster.SparkVersion) {
|
|
return true
|
|
}
|
|
}
|
|
|
|
if task.JobClusterKey != "" {
|
|
for _, job := range b.Config.Resources.Jobs {
|
|
for _, cluster := range job.JobClusters {
|
|
if task.JobClusterKey == cluster.JobClusterKey && cluster.NewCluster.SparkVersion != "" {
|
|
if lowerThanExpectedVersion(cluster.NewCluster.SparkVersion) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if task.ExistingClusterId != "" {
|
|
version, err := getSparkVersionForCluster(ctx, b.WorkspaceClient(), task.ExistingClusterId)
|
|
// If there's error getting spark version for cluster, do not mark it as incompatible
|
|
if err != nil {
|
|
log.Warnf(ctx, "unable to get spark version for cluster %s, err: %s", task.ExistingClusterId, err.Error())
|
|
return false
|
|
}
|
|
|
|
if lowerThanExpectedVersion(version) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func lowerThanExpectedVersion(sparkVersion string) bool {
|
|
parts := strings.Split(sparkVersion, ".")
|
|
if len(parts) < 2 {
|
|
return false
|
|
}
|
|
|
|
if parts[1][0] == 'x' { // treat versions like 13.x as the very latest minor (13.99)
|
|
parts[1] = "99"
|
|
}
|
|
|
|
// if any of the version parts are not numbers, we can't compare
|
|
// so consider it as compatible version
|
|
if _, err := strconv.Atoi(parts[0]); err != nil {
|
|
return false
|
|
}
|
|
|
|
if _, err := strconv.Atoi(parts[1]); err != nil {
|
|
return false
|
|
}
|
|
|
|
v := "v" + parts[0] + "." + parts[1]
|
|
return semver.Compare(v, "v13.1") < 0
|
|
}
|
|
|
|
// Name implements bundle.Mutator.
|
|
func (m *wrapperWarning) Name() string {
|
|
return "PythonWrapperWarning"
|
|
}
|
|
|
|
func getSparkVersionForCluster(ctx context.Context, w *databricks.WorkspaceClient, clusterId string) (string, error) {
|
|
details, err := w.Clusters.GetByClusterId(ctx, clusterId)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return details.SparkVersion, nil
|
|
}
|