package python

import (
	"context"
	"errors"
	"fmt"
	"io/fs"
	"os"
	"path/filepath"
	"runtime"
	"sort"
	"strings"

	"github.com/databricks/cli/libs/log"
	"github.com/databricks/cli/libs/process"
	"golang.org/x/mod/semver"
)

var ErrNoPythonInterpreters = errors.New("no python3 interpreters found")

const (
	officialMswinPython  = "(Python Official) https://python.org/downloads/windows"
	microsoftStorePython = "(Microsoft Store) https://apps.microsoft.com/store/search?publisher=Python%20Software%20Foundation"
)

const worldWriteable = 0o002

type Interpreter struct {
	Version string
	Path    string
}

func (i Interpreter) String() string {
	return fmt.Sprintf("%s (%s)", i.Version, i.Path)
}

type allInterpreters []Interpreter

func (a allInterpreters) Latest() Interpreter {
	return a[len(a)-1]
}

func (a allInterpreters) AtLeast(minimalVersion string) (*Interpreter, error) {
	canonicalMinimalVersion := semver.Canonical("v" + strings.TrimPrefix(minimalVersion, "v"))
	if canonicalMinimalVersion == "" {
		return nil, fmt.Errorf("invalid SemVer: %s", minimalVersion)
	}
	for _, interpreter := range a {
		cmp := semver.Compare(interpreter.Version, canonicalMinimalVersion)
		if cmp < 0 {
			continue
		}
		return &interpreter, nil
	}
	return nil, fmt.Errorf("cannot find Python greater or equal to %s", canonicalMinimalVersion)
}

func DetectInterpreters(ctx context.Context) (allInterpreters, error) {
	found := allInterpreters{}
	seen := map[string]bool{}
	executables, err := pythonicExecutablesFromPathEnvironment(ctx)
	if err != nil {
		return nil, err
	}
	log.Debugf(ctx, "found %d potential alternative Python versions in $PATH", len(executables))
	for _, resolved := range executables {
		if seen[resolved] {
			continue
		}
		seen[resolved] = true
		// probe the binary version by executing it, like `python --version`
		// and parsing the output.
		//
		// Keep in mind, that mswin installations get python.exe and pythonw.exe,
		// which are slightly different: see https://stackoverflow.com/a/30313091
		out, err := process.Background(ctx, []string{resolved, "--version"})
		var processErr *process.ProcessError
		if errors.As(err, &processErr) {
			log.Debugf(ctx, "failed to check version for %s: %s", resolved, processErr.Err)
			continue
		}
		if err != nil {
			log.Debugf(ctx, "failed to check version for %s: %s", resolved, err)
			continue
		}
		version := validPythonVersion(ctx, resolved, out)
		if version == "" {
			continue
		}
		found = append(found, Interpreter{
			Version: version,
			Path:    resolved,
		})
	}
	if runtime.GOOS == "windows" && len(found) == 0 {
		return nil, fmt.Errorf("%w. Install them from %s or %s and restart the shell",
			ErrNoPythonInterpreters, officialMswinPython, microsoftStorePython)
	}
	if len(found) == 0 {
		return nil, ErrNoPythonInterpreters
	}
	sort.Slice(found, func(i, j int) bool {
		a := found[i].Version
		b := found[j].Version
		cmp := semver.Compare(a, b)
		if cmp != 0 {
			return cmp < 0
		}
		return a < b
	})
	return found, nil
}

func pythonicExecutablesFromPathEnvironment(ctx context.Context) (out []string, err error) {
	paths := strings.Split(os.Getenv("PATH"), string(os.PathListSeparator))
	for _, prefix := range paths {
		info, err := os.Stat(prefix)
		if errors.Is(err, fs.ErrNotExist) {
			// some directories in $PATH may not exist
			continue
		}
		if errors.Is(err, fs.ErrPermission) {
			// some directories we cannot list
			continue
		}
		if err != nil {
			return nil, fmt.Errorf("stat %s: %w", prefix, err)
		}
		if !info.IsDir() {
			continue
		}
		perm := info.Mode().Perm()
		if runtime.GOOS != "windows" && perm&worldWriteable != 0 {
			// we try not to run any python binary that sits in a writable folder by all users.
			// this is mainly to avoid breaking the security model on a multi-user system.
			// If the PATH is pointing somewhere untrusted it is the user fault, but we can
			// help here.
			//
			// See https://github.com/databricks/cli/pull/805#issuecomment-1735403952
			log.Debugf(ctx, "%s is world-writeable (%s), skipping for security reasons", prefix, perm)
			continue
		}
		entries, err := os.ReadDir(prefix)
		if errors.Is(err, fs.ErrPermission) {
			// some directories we cannot list
			continue
		}
		if err != nil {
			return nil, fmt.Errorf("listing %s: %w", prefix, err)
		}
		for _, v := range entries {
			if v.IsDir() {
				continue
			}
			if strings.Contains(v.Name(), "-") {
				// skip python3-config, python3.10-config, etc
				continue
			}
			// If Python3 is installed on Windows through GUI installer app that was
			// downloaded from https://python.org/downloads/windows, it may appear
			// in $PATH as `python`, even though it means Python 2.7 in all other
			// operating systems (macOS, Linux).
			//
			// See https://github.com/databrickslabs/ucx/issues/281
			if !strings.HasPrefix(v.Name(), "python") {
				continue
			}
			bin := filepath.Join(prefix, v.Name())
			resolved, err := filepath.EvalSymlinks(bin)
			if err != nil {
				log.Debugf(ctx, "cannot resolve symlink for %s: %s", bin, resolved)
				continue
			}
			out = append(out, resolved)
		}
	}
	return out, nil
}

func validPythonVersion(ctx context.Context, resolved, out string) string {
	out = strings.TrimSpace(out)
	log.Debugf(ctx, "%s --version: %s", resolved, out)

	words := strings.Split(out, " ")
	// The Python distribution from the Windows Store is available in $PATH as `python.exe`
	// and `python3.exe`, even though it symlinks to a real file packaged with some versions of Windows:
	// /c/Program Files/WindowsApps/Microsoft.DesktopAppInstaller_.../AppInstallerPythonRedirector.exe.
	// Executing the `python` command from this distribution opens the Windows Store, allowing users to
	// download and install Python. Once installed, it replaces the `python.exe` and `python3.exe`` stub
	// with the genuine Python executable. Additionally, once user installs from the main installer at
	// https://python.org/downloads/windows, it does not replace this stub.
	//
	// However, a drawback is that if this initial stub is run with any command line arguments, it quietly
	// fails to execute. According to https://github.com/databrickslabs/ucx/issues/281, it can be
	// detected by seeing just the "Python" output without any version info from the `python --version`
	// command execution.
	//
	// See https://github.com/pypa/packaging-problems/issues/379
	// See https://bugs.python.org/issue41327
	if len(words) < 2 {
		log.Debugf(ctx, "%s --version: stub from Windows Store", resolved)
		return ""
	}

	if words[0] != "Python" {
		log.Debugf(ctx, "%s --version: not a Python", resolved)
		return ""
	}

	lastWord := words[len(words)-1]
	version := semver.Canonical("v" + lastWord)
	if version == "" {
		log.Debugf(ctx, "%s --version: invalid SemVer: %s", resolved, lastWord)
		return ""
	}

	return version
}