oh-my-posh/src/engine/config.go
2022-03-27 15:42:42 +02:00

298 lines
7.3 KiB
Go

package engine
import (
"bytes"
json2 "encoding/json"
"fmt"
"io"
"oh-my-posh/color"
"oh-my-posh/environment"
"oh-my-posh/properties"
"os"
"path/filepath"
"strings"
"github.com/gookit/config/v2"
"github.com/gookit/config/v2/json"
"github.com/gookit/config/v2/toml"
yaml "github.com/gookit/config/v2/yamlv3"
"github.com/mitchellh/mapstructure"
)
const (
JSON string = "json"
YAML string = "yaml"
TOML string = "toml"
configVersion = 2
)
// Config holds all the theme for rendering the prompt
type Config struct {
Version int `json:"version"`
FinalSpace bool `json:"final_space,omitempty"`
OSC99 bool `json:"osc99,omitempty"`
ConsoleTitleTemplate string `json:"console_title_template,omitempty"`
TerminalBackground string `json:"terminal_background,omitempty"`
Blocks []*Block `json:"blocks,omitempty"`
Tooltips []*Segment `json:"tooltips,omitempty"`
TransientPrompt *Segment `json:"transient_prompt,omitempty"`
ValidLine *Segment `json:"valid_line,omitempty"`
ErrorLine *Segment `json:"error_line,omitempty"`
SecondaryPrompt *Segment `json:"secondary_prompt,omitempty"`
DebugPrompt *Segment `json:"debug_prompt,omitempty"`
Palette color.Palette `json:"palette,omitempty"`
Output string `json:"-"`
format string
origin string
eval bool
updated bool
}
// MakeColors creates instance of AnsiColors to use in AnsiWriter according to
// environment and configuration.
func (cfg *Config) MakeColors(env environment.Environment) color.AnsiColors {
cacheDisabled := env.Getenv("OMP_CACHE_DISABLED") == "1"
return color.MakeColors(cfg.Palette, !cacheDisabled)
}
func (cfg *Config) print(message string) {
if cfg.eval {
fmt.Printf("echo \"%s\"", message)
return
}
fmt.Println(message)
}
func (cfg *Config) exitWithError(err error) {
if err == nil {
return
}
defer os.Exit(1)
message := "Oh My Posh Error:\n\n" + err.Error()
if cfg.eval {
fmt.Printf("echo \"%s\"\n", message)
return
}
cfg.print(message)
}
// LoadConfig returns the default configuration including possible user overrides
func LoadConfig(env environment.Environment) *Config {
cfg := loadConfig(env)
// only migrate automatically when the switch isn't set
if !env.Flags().Migrate && cfg.Version != configVersion {
cfg.BackupAndMigrate(env)
}
return cfg
}
func loadConfig(env environment.Environment) *Config {
var cfg Config
configFile := env.Flags().Config
if configFile == "" {
return defaultConfig()
}
if _, err := os.Stat(configFile); os.IsNotExist(err) {
cfg.exitWithError(err)
}
cfg.origin = configFile
cfg.format = strings.TrimPrefix(filepath.Ext(configFile), ".")
if cfg.format == "yml" {
cfg.format = YAML
}
config.AddDriver(yaml.Driver)
config.AddDriver(json.Driver)
config.AddDriver(toml.Driver)
config.WithOptions(func(opt *config.Options) {
opt.DecoderConfig = &mapstructure.DecoderConfig{
TagName: "json",
}
})
err := config.LoadFiles(configFile)
cfg.exitWithError(err)
err = config.BindStruct("", &cfg)
cfg.exitWithError(err)
// initialize default values
if cfg.TransientPrompt == nil {
cfg.TransientPrompt = &Segment{}
}
return &cfg
}
func (cfg *Config) sync() {
if !cfg.updated {
return
}
var structMap map[string]interface{}
inrec, err := json2.Marshal(cfg)
if err != nil {
return
}
err = json2.Unmarshal(inrec, &structMap)
if err != nil {
return
}
// remove empty structs
for k, v := range structMap {
if smap, OK := v.(map[string]interface{}); OK && len(smap) == 0 {
delete(structMap, k)
}
}
config.SetData(structMap)
}
func (cfg *Config) Export(format string) string {
cfg.sync()
if len(format) != 0 {
cfg.format = format
}
config.AddDriver(yaml.Driver)
config.AddDriver(toml.Driver)
var result bytes.Buffer
if cfg.format == JSON {
data := config.Data()
data["$schema"] = "https://raw.githubusercontent.com/JanDeDobbeleer/oh-my-posh/main/themes/schema.json"
jsonEncoder := json2.NewEncoder(&result)
jsonEncoder.SetEscapeHTML(false)
jsonEncoder.SetIndent("", " ")
err := jsonEncoder.Encode(data)
cfg.exitWithError(err)
return escapeGlyphs(result.String())
}
_, err := config.DumpTo(&result, cfg.format)
cfg.exitWithError(err)
if cfg.format == YAML {
prefix := "# yaml-language-server: $schema=https://raw.githubusercontent.com/JanDeDobbeleer/oh-my-posh/main/themes/schema.json\n\n"
return prefix + result.String()
}
prefix := "#:schema https://raw.githubusercontent.com/JanDeDobbeleer/oh-my-posh/main/themes/schema.json\n\n"
return prefix + escapeGlyphs(result.String())
}
func (cfg *Config) BackupAndMigrate(env environment.Environment) {
origin := cfg.backup()
cfg.Migrate(env)
cfg.Write(cfg.format)
cfg.print(fmt.Sprintf("\nOh My Posh config migrated to version %d\nBackup config available at %s\n\n", cfg.Version, origin))
}
func (cfg *Config) Write(format string) {
content := cfg.Export(format)
destination := cfg.Output
if len(destination) == 0 {
destination = cfg.origin
}
f, err := os.OpenFile(destination, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0755)
cfg.exitWithError(err)
_, err = f.WriteString(content)
cfg.exitWithError(err)
if err := f.Close(); err != nil {
cfg.exitWithError(err)
}
}
func (cfg *Config) backup() string {
dst := cfg.origin + ".bak"
source, err := os.Open(cfg.origin)
cfg.exitWithError(err)
defer source.Close()
destination, err := os.Create(dst)
cfg.exitWithError(err)
defer destination.Close()
_, err = io.Copy(destination, source)
cfg.exitWithError(err)
return dst
}
func escapeGlyphs(s string) string {
var builder strings.Builder
for _, r := range s {
// exclude regular characters and emoji
if r < 0x1000 || r > 0x10000 {
builder.WriteRune(r)
continue
}
quoted := fmt.Sprintf("\\u%04x", r)
builder.WriteString(quoted)
}
return builder.String()
}
func defaultConfig() *Config {
cfg := &Config{
Version: 1,
FinalSpace: true,
Blocks: []*Block{
{
Type: Prompt,
Alignment: Left,
Segments: []*Segment{
{
Type: SESSION,
Style: Diamond,
Background: "#c386f1",
Foreground: "#ffffff",
LeadingDiamond: "\uE0B6",
TrailingDiamond: "\uE0B0",
},
{
Type: PATH,
Style: Powerline,
PowerlineSymbol: "\uE0B0",
Background: "#ff479c",
Foreground: "#ffffff",
Properties: properties.Map{
properties.Style: "folder",
},
},
{
Type: SHELL,
Style: Powerline,
PowerlineSymbol: "\uE0B0",
Background: "#0077c2",
Foreground: "#ffffff",
},
{
Type: TEXT,
Style: Powerline,
PowerlineSymbol: "\uE0B0",
Background: "#ffffff",
Foreground: "#111111",
Template: " no config ",
},
{
Type: EXIT,
Style: Diamond,
Background: "#2e9599",
Foreground: "#ffffff",
LeadingDiamond: "<transparent,background>\uE0B0</>",
TrailingDiamond: "\uE0B4",
BackgroundTemplates: []string{
"{{ if gt .Code 0 }}#f1184c{{ end }}",
},
Template: " \uE23A",
Properties: properties.Map{
properties.AlwaysEnabled: true,
},
},
},
},
},
}
return cfg
}