package template import ( "bytes" "errors" "oh-my-posh/platform" "oh-my-posh/regex" "strings" "text/template" ) const ( // Errors to show when the template handling fails InvalidTemplate = "invalid template text" IncorrectTemplate = "unable to create text based on template" ) type Text struct { Template string Context interface{} Env platform.Environment TemplatesResult string } type Data interface{} type Context struct { *platform.TemplateCache // Simple container to hold ANY object Data Templates string } func (c *Context) init(t *Text) { c.Data = t.Context c.Templates = t.TemplatesResult if cache := t.Env.TemplateCache(); cache != nil { c.TemplateCache = cache return } } func (t *Text) Render() (string, error) { t.cleanTemplate() tmpl, err := template.New(t.Template).Funcs(funcMap()).Parse(t.Template) if err != nil { t.Env.Log(platform.Error, "Render", err.Error()) return "", errors.New(InvalidTemplate) } context := &Context{} context.init(t) buffer := new(bytes.Buffer) defer buffer.Reset() err = tmpl.Execute(buffer, context) if err != nil { t.Env.Log(platform.Error, "Render", err.Error()) return "", errors.New(IncorrectTemplate) } text := buffer.String() // issue with missingkey=zero ignored for map[string]interface{} // https://github.com/golang/go/issues/24963 text = strings.ReplaceAll(text, "", "") return text, nil } func (t *Text) cleanTemplate() { knownVariables := []string{ "Root", "PWD", "Folder", "Shell", "ShellVersion", "UserName", "HostName", "Env", "Data", "Code", "OS", "WSL", "Segments", "Templates", } knownVariable := func(variable string) bool { variable = strings.TrimPrefix(variable, ".") splitted := strings.Split(variable, ".") if len(splitted) == 0 { return true } variable = splitted[0] // check if alphanumeric if !regex.MatchString(`^[a-zA-Z0-9]+$`, variable) { return true } for _, b := range knownVariables { if variable == b { return true } } return false } var result string var property string var inProperty bool for _, char := range t.Template { switch char { case '.': var lastChar rune if len(result) > 0 { lastChar = rune(result[len(result)-1]) } // only replace if we're in a valid property start // with a space, { or ( character switch lastChar { case ' ', '{', '(': property += string(char) inProperty = true default: result += string(char) } case ' ', '}', ')': // space or } if !inProperty { result += string(char) continue } // end of a variable, needs to be appended if !knownVariable(property) { result += ".Data" + property } else { result += property } property = "" result += string(char) inProperty = false default: if inProperty { property += string(char) continue } result += string(char) } } // return the result and remaining unresolved property t.Template = result + property }