2023-03-29 18:44:19 +00:00
|
|
|
package databrickscfg
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-06-02 11:49:39 +00:00
|
|
|
"errors"
|
2023-03-29 18:44:19 +00:00
|
|
|
"fmt"
|
2024-06-03 12:39:36 +00:00
|
|
|
"io/fs"
|
2023-03-29 18:44:19 +00:00
|
|
|
"strings"
|
|
|
|
|
2023-05-16 16:35:39 +00:00
|
|
|
"github.com/databricks/cli/libs/log"
|
2023-03-29 18:44:19 +00:00
|
|
|
"github.com/databricks/databricks-sdk-go/config"
|
|
|
|
"gopkg.in/ini.v1"
|
|
|
|
)
|
|
|
|
|
|
|
|
var ResolveProfileFromHost = profileFromHostLoader{}
|
|
|
|
|
2023-06-02 11:49:39 +00:00
|
|
|
var errNoMatchingProfiles = errors.New("no matching config profiles found")
|
|
|
|
|
|
|
|
type errMultipleProfiles []string
|
|
|
|
|
|
|
|
func (e errMultipleProfiles) Error() string {
|
2025-01-07 10:49:23 +00:00
|
|
|
return "multiple profiles matched: " + strings.Join(e, ", ")
|
2023-06-02 11:49:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func findMatchingProfile(configFile *config.File, matcher func(*ini.Section) bool) (*ini.Section, error) {
|
|
|
|
// Look for sections in the configuration file that match the configured host.
|
|
|
|
var matching []*ini.Section
|
|
|
|
for _, section := range configFile.Sections() {
|
|
|
|
if !matcher(section) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
matching = append(matching, section)
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there are no matching sections, we don't do anything.
|
|
|
|
if len(matching) == 0 {
|
|
|
|
return nil, errNoMatchingProfiles
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there are multiple matching sections, let the user know it is impossible
|
|
|
|
// to unambiguously select a profile to use.
|
|
|
|
if len(matching) > 1 {
|
|
|
|
var names errMultipleProfiles
|
|
|
|
for _, section := range matching {
|
|
|
|
names = append(names, section.Name())
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, names
|
|
|
|
}
|
|
|
|
|
|
|
|
return matching[0], nil
|
|
|
|
}
|
|
|
|
|
2023-03-29 18:44:19 +00:00
|
|
|
type profileFromHostLoader struct{}
|
|
|
|
|
|
|
|
func (l profileFromHostLoader) Name() string {
|
|
|
|
return "resolve-profile-from-host"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l profileFromHostLoader) Configure(cfg *config.Config) error {
|
|
|
|
// Skip an attempt to resolve a profile from the host if any authentication
|
|
|
|
// is already configured (either directly, through environment variables, or
|
|
|
|
// if a profile was specified).
|
|
|
|
if cfg.Host == "" || l.isAnyAuthConfigured(cfg) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-06-02 11:49:39 +00:00
|
|
|
ctx := context.Background()
|
2023-04-03 19:33:21 +00:00
|
|
|
configFile, err := config.LoadFile(cfg.ConfigFile)
|
2023-03-29 18:44:19 +00:00
|
|
|
if err != nil {
|
2024-06-03 12:39:36 +00:00
|
|
|
if errors.Is(err, fs.ErrNotExist) {
|
2023-03-29 18:44:19 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return fmt.Errorf("cannot parse config file: %w", err)
|
|
|
|
}
|
|
|
|
// Normalized version of the configured host.
|
|
|
|
host := normalizeHost(cfg.Host)
|
2023-06-02 11:49:39 +00:00
|
|
|
match, err := findMatchingProfile(configFile, func(s *ini.Section) bool {
|
|
|
|
key, err := s.GetKey("host")
|
2023-03-29 18:44:19 +00:00
|
|
|
if err != nil {
|
2023-06-02 11:49:39 +00:00
|
|
|
log.Tracef(ctx, "section %s: %s", s.Name(), err)
|
|
|
|
return false
|
2023-03-29 18:44:19 +00:00
|
|
|
}
|
|
|
|
|
2023-06-02 11:49:39 +00:00
|
|
|
// Check if this section matches the normalized host
|
|
|
|
return normalizeHost(key.Value()) == host
|
|
|
|
})
|
|
|
|
if err == errNoMatchingProfiles {
|
2023-03-29 18:44:19 +00:00
|
|
|
return nil
|
|
|
|
}
|
2023-06-02 11:49:39 +00:00
|
|
|
if err, ok := err.(errMultipleProfiles); ok {
|
2023-03-29 18:44:19 +00:00
|
|
|
return fmt.Errorf(
|
2023-07-12 12:09:25 +00:00
|
|
|
"%s: %w: please set DATABRICKS_CONFIG_PROFILE or provide --profile flag to specify one",
|
2023-06-02 11:49:39 +00:00
|
|
|
host, err)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2023-03-29 18:44:19 +00:00
|
|
|
}
|
|
|
|
|
2023-06-02 11:49:39 +00:00
|
|
|
log.Debugf(ctx, "Loading profile %s because of host match", match.Name())
|
2024-04-03 08:14:04 +00:00
|
|
|
err = config.ConfigAttributes.ResolveFromStringMapWithSource(cfg, match.KeysHash(), config.Source{
|
|
|
|
Type: config.SourceFile,
|
|
|
|
Name: configFile.Path(),
|
|
|
|
})
|
2023-03-29 18:44:19 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%s %s profile: %w", configFile.Path(), match.Name(), err)
|
|
|
|
}
|
|
|
|
|
2023-10-20 13:10:31 +00:00
|
|
|
cfg.Profile = match.Name()
|
2023-03-29 18:44:19 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l profileFromHostLoader) isAnyAuthConfigured(cfg *config.Config) bool {
|
2023-12-18 09:57:07 +00:00
|
|
|
// If any of the auth-specific attributes are set, we can skip profile resolution.
|
2023-03-29 18:44:19 +00:00
|
|
|
for _, a := range config.ConfigAttributes {
|
2024-04-03 08:14:04 +00:00
|
|
|
if !a.HasAuthAttribute() {
|
2023-03-29 18:44:19 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !a.IsZero(cfg) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
2023-12-18 09:57:07 +00:00
|
|
|
// If the auth type is set, we can skip profile resolution.
|
|
|
|
// For example, to force "azure-cli", only the host and the auth type will be set.
|
|
|
|
return cfg.AuthType != ""
|
2023-03-29 18:44:19 +00:00
|
|
|
}
|