oh-my-posh/src/segment_language.go

172 lines
4 KiB
Go
Raw Normal View History

2020-11-14 11:04:04 -08:00
package main
import (
"errors"
"fmt"
"strings"
)
type loadContext func()
type inContext func() bool
type version struct {
full string
major string
minor string
patch string
regex string
urlTemplate string
}
func (v *version) parse(versionInfo string) error {
values := findNamedRegexMatch(v.regex, versionInfo)
if len(values) == 0 {
return errors.New("cannot parse version string")
}
v.full = values["version"]
v.major = values["major"]
v.minor = values["minor"]
v.patch = values["patch"]
return nil
}
2020-11-14 11:04:04 -08:00
type language struct {
props *properties
env environmentInfo
extensions []string
commands []string
executable string
2020-11-14 11:04:04 -08:00
versionParam string
version *version
exitCode int
loadContext loadContext
inContext inContext
2020-11-14 11:04:04 -08:00
}
const (
// DisplayMode sets the display mode (always, when_in_context, never)
DisplayMode Property = "display_mode"
2020-12-31 23:00:08 -08:00
// DisplayModeAlways displays the segment always
DisplayModeAlways string = "always"
// DisplayModeFiles displays the segment when the current folder contains certain extensions
DisplayModeFiles string = "files"
// DisplayModeEnvironment displays the segment when the environment has a language's context
DisplayModeEnvironment string = "environment"
// DisplayModeContext displays the segment when the environment or files is active
DisplayModeContext string = "context"
2021-01-05 03:28:33 -08:00
// MissingCommandTextProperty sets the text to display when the command is not present in the system
MissingCommandTextProperty Property = "missing_command_text"
2021-01-05 03:28:33 -08:00
// MissingCommandText displays empty string by default
MissingCommandText string = ""
)
2020-11-14 11:04:04 -08:00
func (l *language) string() string {
if !l.props.getBool(DisplayVersion, true) {
return ""
}
if !l.hasCommand() {
return l.props.getString(MissingCommandTextProperty, MissingCommandText)
}
err := l.setVersion()
if err != nil {
return ""
}
// build release notes hyperlink
if l.props.getBool(EnableHyperlink, false) && l.version.urlTemplate != "" {
version, err := TruncatingSprintf(l.version.urlTemplate, l.version.full, l.version.major, l.version.minor, l.version.patch)
if err != nil {
return l.version.full
}
return version
}
return l.version.full
2020-11-14 11:04:04 -08:00
}
func (l *language) enabled() bool {
l.loadLanguageContext()
displayMode := l.props.getString(DisplayMode, DisplayModeFiles)
switch displayMode {
case DisplayModeAlways:
return true
case DisplayModeEnvironment:
return l.inLanguageContext()
case DisplayModeFiles:
return l.hasLanguageFiles()
case DisplayModeContext:
fallthrough
default:
return l.hasLanguageFiles() || l.inLanguageContext()
}
}
// hasLanguageFiles will return true at least one file matching the extensions is found
func (l *language) hasLanguageFiles() bool {
2020-11-14 11:04:04 -08:00
for i, extension := range l.extensions {
if l.env.hasFiles(extension) {
break
}
if i == len(l.extensions)-1 {
return false
}
}
return true
}
// setVersion parses the version string returned by the command
func (l *language) setVersion() error {
versionInfo, err := l.env.runCommand(l.executable, l.versionParam)
if exitErr, ok := err.(*commandError); ok {
l.exitCode = exitErr.exitCode
return errors.New("error executing command")
}
err = l.version.parse(versionInfo)
if err != nil {
return err
}
return nil
}
2020-12-31 11:07:59 -08:00
// hasCommand checks if one of the commands exists and sets it as executable
func (l *language) hasCommand() bool {
2020-11-14 11:04:04 -08:00
for i, command := range l.commands {
if l.env.hasCommand(command) {
l.executable = command
2020-11-14 11:04:04 -08:00
break
}
if i == len(l.commands)-1 {
return false
}
}
return true
}
func (l *language) loadLanguageContext() {
if l.loadContext == nil {
return
}
l.loadContext()
}
func (l *language) inLanguageContext() bool {
if l.inContext == nil {
return false
}
return l.inContext()
}
func TruncatingSprintf(str string, args ...interface{}) (string, error) {
n := strings.Count(str, "%s")
if n > len(args) {
return "", errors.New("Too many parameters")
}
if n == 0 {
return fmt.Sprintf(str, args...), nil
}
return fmt.Sprintf(str, args[:n]...), nil
}