// Package filters is an internal package that defines the standard Liquid filters. package filters import ( "encoding/json" "fmt" "html" "math" "net/url" "reflect" "regexp" "strings" "time" "unicode" "unicode/utf8" "github.com/danog/liquid/values" "github.com/osteele/tuesday" ) // A FilterDictionary holds filters. type FilterDictionary interface { AddFilter(string, interface{}) } // AddStandardFilters defines the standard Liquid filters. func AddStandardFilters(fd FilterDictionary) { // nolint: gocyclo // value filters fd.AddFilter("default", func(value, defaultValue interface{}) interface{} { if value == nil || value == false || values.IsEmpty(value) { value = defaultValue } return value }) // array filters fd.AddFilter("compact", func(a []interface{}) (result []interface{}) { for _, item := range a { if item != nil { result = append(result, item) } } return }) fd.AddFilter("concat", func(a, b []interface{}) (result []interface{}) { result = make([]interface{}, 0, len(a)+len(b)) return append(append(result, a...), b...) }) fd.AddFilter("join", joinFilter) fd.AddFilter("map", func(a []map[string]interface{}, key string) (result []interface{}) { for _, obj := range a { result = append(result, obj[key]) } return result }) fd.AddFilter("reverse", reverseFilter) fd.AddFilter("sort", sortFilter) // https://shopify.github.io/liquid/ does not demonstrate first and last as filters, // but https://help.shopify.com/themes/liquid/filters/array-filters does fd.AddFilter("first", func(a []interface{}) interface{} { if len(a) == 0 { return nil } return a[0] }) fd.AddFilter("last", func(a []interface{}) interface{} { if len(a) == 0 { return nil } return a[len(a)-1] }) fd.AddFilter("uniq", uniqFilter) // date filters fd.AddFilter("date", func(t time.Time, format func(string) string) (string, error) { f := format("%a, %b %d, %y") return tuesday.Strftime(f, t) }) // number filters fd.AddFilter("abs", math.Abs) fd.AddFilter("ceil", func(a float64) int { return int(math.Ceil(a)) }) fd.AddFilter("floor", func(a float64) int { return int(math.Floor(a)) }) fd.AddFilter("modulo", math.Mod) fd.AddFilter("minus", func(a, b float64) float64 { return a - b }) fd.AddFilter("plus", func(a, b float64) float64 { return a + b }) fd.AddFilter("times", func(a, b float64) float64 { return a * b }) fd.AddFilter("divided_by", func(a float64, b interface{}) interface{} { switch q := b.(type) { case int, int16, int32, int64: return int(a) / q.(int) case float32, float64: return a / b.(float64) default: return nil } }) fd.AddFilter("round", func(n float64, places func(int) int) float64 { pl := places(0) exp := math.Pow10(pl) return math.Floor(n*exp+0.5) / exp }) // sequence filters fd.AddFilter("size", values.Length) // string filters fd.AddFilter("append", func(s, suffix string) string { return s + suffix }) fd.AddFilter("capitalize", func(s, suffix string) string { if len(s) == 0 { return s } return strings.ToUpper(s[:1]) + s[1:] }) fd.AddFilter("downcase", func(s, suffix string) string { return strings.ToLower(s) }) fd.AddFilter("escape", html.EscapeString) fd.AddFilter("escape_once", func(s, suffix string) string { return html.EscapeString(html.UnescapeString(s)) }) fd.AddFilter("newline_to_br", func(s string) string { return strings.Replace(s, "\n", "
", -1) }) fd.AddFilter("prepend", func(s, prefix string) string { return prefix + s }) fd.AddFilter("remove", func(s, old string) string { return strings.Replace(s, old, "", -1) }) fd.AddFilter("remove_first", func(s, old string) string { return strings.Replace(s, old, "", 1) }) fd.AddFilter("replace", func(s, old, new string) string { return strings.Replace(s, old, new, -1) }) fd.AddFilter("replace_first", func(s, old, new string) string { return strings.Replace(s, old, new, 1) }) fd.AddFilter("sort_natural", sortNaturalFilter) fd.AddFilter("slice", func(s string, start int, length func(int) int) string { // runes aren't bytes; don't use slice n := length(1) if start < 0 { start = utf8.RuneCountInString(s) + start } p := regexp.MustCompile(fmt.Sprintf(`^.{%d}(.{0,%d}).*$`, start, n)) return p.ReplaceAllString(s, "$1") }) fd.AddFilter("split", splitFilter) fd.AddFilter("strip_html", func(s string) string { // TODO this probably isn't sufficient return regexp.MustCompile(`<.*?>`).ReplaceAllString(s, "") }) fd.AddFilter("strip_newlines", func(s string) string { return strings.Replace(s, "\n", "", -1) }) fd.AddFilter("strip", strings.TrimSpace) fd.AddFilter("lstrip", func(s string) string { return strings.TrimLeftFunc(s, unicode.IsSpace) }) fd.AddFilter("rstrip", func(s string) string { return strings.TrimRightFunc(s, unicode.IsSpace) }) fd.AddFilter("truncate", func(s string, length func(int) int, ellipsis func(string) string) string { n := length(50) el := ellipsis("...") // runes aren't bytes; don't use slice re := regexp.MustCompile(fmt.Sprintf(`^(.{%d})..{%d,}`, n-len(el), len(el))) return re.ReplaceAllString(s, `$1`+el) }) fd.AddFilter("truncatewords", func(s string, length func(int) int, ellipsis func(string) string) string { el := ellipsis("...") n := length(15) re := regexp.MustCompile(fmt.Sprintf(`^(?:\s*\S+){%d}`, n)) m := re.FindString(s) if m == "" { return s } return m + el }) fd.AddFilter("upcase", func(s, suffix string) string { return strings.ToUpper(s) }) fd.AddFilter("url_encode", url.QueryEscape) fd.AddFilter("url_decode", url.QueryUnescape) // debugging filters // inspect is from Jekyll fd.AddFilter("inspect", func(value interface{}) string { s, err := json.Marshal(value) if err != nil { return fmt.Sprintf("%#v", value) } return string(s) }) fd.AddFilter("type", func(value interface{}) string { return fmt.Sprintf("%T", value) }) } func joinFilter(a []interface{}, sep func(string) string) interface{} { ss := make([]string, 0, len(a)) s := sep(" ") for _, v := range a { if v != nil { ss = append(ss, fmt.Sprint(v)) } } return strings.Join(ss, s) } func reverseFilter(a []interface{}) interface{} { result := make([]interface{}, len(a)) for i, x := range a { result[len(result)-1-i] = x } return result } var wsre = regexp.MustCompile(`[[:space:]]+`) func splitFilter(s, sep string) interface{} { result := strings.Split(s, sep) if sep == " " { // Special case for Ruby, therefore Liquid result = wsre.Split(s, -1) } // This matches Ruby / Liquid / Jekyll's observed behavior. for len(result) > 0 && result[len(result)-1] == "" { result = result[:len(result)-1] } return result } func uniqFilter(a []interface{}) (result []interface{}) { seenMap := map[interface{}]bool{} seen := func(item interface{}) bool { if k := reflect.TypeOf(item).Kind(); k < reflect.Array || k == reflect.Ptr || k == reflect.UnsafePointer { if seenMap[item] { return true } seenMap[item] = true return false } // the O(n^2) case: for _, other := range result { if eqItems(item, other) { return true } } return false } for _, item := range a { if !seen(item) { result = append(result, item) } } return } func eqItems(a, b interface{}) bool { if reflect.TypeOf(a).Comparable() && reflect.TypeOf(b).Comparable() { return a == b } return reflect.DeepEqual(a, b) }