2017-06-27 19:36:38 +02:00
|
|
|
// Package filters defines the standard Liquid filters.
|
2017-06-27 18:06:24 +02:00
|
|
|
package filters
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2017-06-28 21:33:13 +02:00
|
|
|
"html"
|
2017-06-28 19:28:55 +02:00
|
|
|
"math"
|
2017-06-29 02:49:27 +02:00
|
|
|
"reflect"
|
2017-06-28 21:33:13 +02:00
|
|
|
"regexp"
|
2017-06-27 18:06:24 +02:00
|
|
|
"strings"
|
2017-06-27 22:02:05 +02:00
|
|
|
"time"
|
2017-06-28 19:28:55 +02:00
|
|
|
"unicode"
|
2017-06-27 18:06:24 +02:00
|
|
|
|
2017-06-27 22:53:34 +02:00
|
|
|
"github.com/leekchan/timeutil"
|
2017-06-27 18:06:24 +02:00
|
|
|
"github.com/osteele/liquid/expressions"
|
|
|
|
"github.com/osteele/liquid/generics"
|
|
|
|
)
|
|
|
|
|
|
|
|
// DefineStandardFilters defines the standard Liquid filters.
|
|
|
|
func DefineStandardFilters() {
|
2017-06-27 22:02:05 +02:00
|
|
|
// values
|
2017-06-29 19:41:04 +02:00
|
|
|
expressions.DefineFilter("default", func(value, defaultValue interface{}) interface{} {
|
|
|
|
if value == nil || value == false || generics.IsEmpty(value) {
|
|
|
|
value = defaultValue
|
2017-06-27 22:02:05 +02:00
|
|
|
}
|
2017-06-29 19:41:04 +02:00
|
|
|
return value
|
2017-06-27 22:02:05 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
// dates
|
2017-06-29 19:41:04 +02:00
|
|
|
expressions.DefineFilter("date", func(date time.Time, format interface{}) interface{} {
|
2017-06-28 02:56:17 +02:00
|
|
|
form, ok := format.(string)
|
2017-06-27 22:53:34 +02:00
|
|
|
if !ok {
|
2017-06-28 02:56:17 +02:00
|
|
|
form = "%a, %b %d, %y"
|
2017-06-27 22:02:05 +02:00
|
|
|
}
|
2017-06-29 17:30:43 +02:00
|
|
|
// FIXME All the libraries I could find format 09:00 with "%-H" as "H" instead of "9".
|
|
|
|
// This renders it as "09" instead of "9", which is still bad but better.
|
|
|
|
form = strings.Replace(form, "%-", "%", -1)
|
2017-06-29 19:41:04 +02:00
|
|
|
return timeutil.Strftime(&date, form)
|
2017-06-27 22:02:05 +02:00
|
|
|
})
|
|
|
|
|
2017-06-27 18:06:24 +02:00
|
|
|
// lists
|
2017-06-29 19:41:04 +02:00
|
|
|
expressions.DefineFilter("compact", func(array []interface{}) interface{} {
|
2017-06-28 20:41:46 +02:00
|
|
|
out := []interface{}{}
|
2017-06-29 19:41:04 +02:00
|
|
|
for _, item := range array {
|
|
|
|
if item != nil {
|
|
|
|
out = append(out, item)
|
2017-06-28 20:41:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
})
|
2017-06-27 18:06:24 +02:00
|
|
|
expressions.DefineFilter("join", joinFilter)
|
2017-06-29 19:41:04 +02:00
|
|
|
expressions.DefineFilter("map", func(array []map[string]interface{}, key string) interface{} {
|
2017-06-28 20:41:46 +02:00
|
|
|
out := []interface{}{}
|
2017-06-29 19:41:04 +02:00
|
|
|
for _, obj := range array {
|
2017-06-28 20:41:46 +02:00
|
|
|
out = append(out, obj[key])
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
})
|
2017-06-27 23:29:50 +02:00
|
|
|
expressions.DefineFilter("reverse", reverseFilter)
|
2017-06-27 18:06:24 +02:00
|
|
|
expressions.DefineFilter("sort", sortFilter)
|
2017-06-28 20:41:46 +02:00
|
|
|
// https://shopify.github.io/liquid/ does not demonstrate first and last as filters,
|
|
|
|
// but https://help.shopify.com/themes/liquid/filters/array-filters does
|
2017-06-29 19:41:04 +02:00
|
|
|
expressions.DefineFilter("first", func(array []interface{}) interface{} {
|
|
|
|
if len(array) == 0 {
|
2017-06-28 20:41:46 +02:00
|
|
|
return nil
|
|
|
|
}
|
2017-06-29 19:41:04 +02:00
|
|
|
return array[0]
|
2017-06-28 20:41:46 +02:00
|
|
|
})
|
2017-06-29 19:41:04 +02:00
|
|
|
expressions.DefineFilter("last", func(array []interface{}) interface{} {
|
|
|
|
if len(array) == 0 {
|
2017-06-28 20:41:46 +02:00
|
|
|
return nil
|
|
|
|
}
|
2017-06-29 19:41:04 +02:00
|
|
|
return array[len(array)-1]
|
2017-06-28 20:41:46 +02:00
|
|
|
})
|
2017-06-27 18:06:24 +02:00
|
|
|
|
2017-06-28 19:28:55 +02:00
|
|
|
// numbers
|
|
|
|
expressions.DefineFilter("abs", math.Abs)
|
|
|
|
expressions.DefineFilter("ceil", math.Ceil)
|
|
|
|
expressions.DefineFilter("floor", math.Floor)
|
2017-06-28 22:43:18 +02:00
|
|
|
expressions.DefineFilter("modulo", math.Mod)
|
|
|
|
expressions.DefineFilter("minus", func(a, b float64) float64 {
|
|
|
|
return a - b
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("plus", func(a, b float64) float64 {
|
|
|
|
return a + b
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("times", func(a, b float64) float64 {
|
|
|
|
return a * b
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("divided_by", func(a float64, b interface{}) interface{} {
|
|
|
|
switch bt := b.(type) {
|
|
|
|
case int, int16, int32, int64:
|
|
|
|
return int(a) / bt.(int)
|
|
|
|
case float32, float64:
|
2017-06-29 18:26:04 +02:00
|
|
|
return a / b.(float64)
|
2017-06-28 22:43:18 +02:00
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("round", func(n float64, places interface{}) float64 {
|
|
|
|
pl, ok := places.(int)
|
|
|
|
if !ok {
|
|
|
|
pl = 0
|
|
|
|
}
|
|
|
|
exp := math.Pow10(pl)
|
|
|
|
return math.Floor(n*exp+0.5) / exp
|
|
|
|
})
|
2017-06-28 19:28:55 +02:00
|
|
|
|
2017-06-28 20:41:46 +02:00
|
|
|
// sequences
|
|
|
|
expressions.DefineFilter("size", generics.Length)
|
|
|
|
|
2017-06-27 18:06:24 +02:00
|
|
|
// strings
|
2017-06-28 19:28:55 +02:00
|
|
|
expressions.DefineFilter("append", func(s, suffix string) string {
|
|
|
|
return s + suffix
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("capitalize", func(s, suffix string) string {
|
|
|
|
if len(s) < 1 {
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
return strings.ToUpper(s[:1]) + s[1:]
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("downcase", func(s, suffix string) string {
|
|
|
|
return strings.ToLower(s)
|
|
|
|
})
|
2017-06-28 21:33:13 +02:00
|
|
|
expressions.DefineFilter("escape", html.EscapeString)
|
|
|
|
expressions.DefineFilter("escape_once", func(s, suffix string) string {
|
|
|
|
return html.EscapeString(html.UnescapeString(s))
|
|
|
|
})
|
|
|
|
// TODO test case for this
|
|
|
|
expressions.DefineFilter("newline_to_br", func(s string) string {
|
|
|
|
return strings.Replace(s, "\n", "<br />", -1)
|
|
|
|
})
|
2017-06-28 19:28:55 +02:00
|
|
|
expressions.DefineFilter("prepend", func(s, prefix string) string {
|
|
|
|
return prefix + s
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("remove", func(s, old string) string {
|
|
|
|
return strings.Replace(s, old, "", -1)
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("remove_first", func(s, old string) string {
|
|
|
|
return strings.Replace(s, old, "", 1)
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("replace", func(s, old, new string) string {
|
|
|
|
return strings.Replace(s, old, new, -1)
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("replace_first", func(s, old, new string) string {
|
|
|
|
return strings.Replace(s, old, new, 1)
|
|
|
|
})
|
2017-06-28 20:41:46 +02:00
|
|
|
expressions.DefineFilter("slice", func(s string, start int, length interface{}) string {
|
|
|
|
n, ok := length.(int)
|
|
|
|
if !ok {
|
|
|
|
n = 1
|
|
|
|
}
|
|
|
|
if start < 0 {
|
|
|
|
start = len(s) + start
|
|
|
|
}
|
|
|
|
if start >= len(s) {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
if start+n > len(s) {
|
|
|
|
return s[start:]
|
|
|
|
}
|
|
|
|
return s[start : start+n]
|
|
|
|
})
|
2017-06-27 18:06:24 +02:00
|
|
|
expressions.DefineFilter("split", splitFilter)
|
2017-06-28 21:33:13 +02:00
|
|
|
expressions.DefineFilter("strip_html", func(s string) string {
|
|
|
|
// TODO this probably isn't sufficient
|
|
|
|
return regexp.MustCompile(`<.*?>`).ReplaceAllString(s, "")
|
|
|
|
})
|
|
|
|
// TODO test case for this
|
|
|
|
expressions.DefineFilter("strip_newlines", func(s string) string {
|
|
|
|
return strings.Replace(s, "\n", "", -1)
|
|
|
|
})
|
2017-06-28 19:28:55 +02:00
|
|
|
expressions.DefineFilter("strip", strings.TrimSpace)
|
|
|
|
expressions.DefineFilter("lstrip", func(s string) string {
|
|
|
|
return strings.TrimLeftFunc(s, unicode.IsSpace)
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("rstrip", func(s string) string {
|
|
|
|
return strings.TrimRightFunc(s, unicode.IsSpace)
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("truncate", func(s string, n int, ellipsis interface{}) string {
|
|
|
|
el, ok := ellipsis.(string)
|
|
|
|
if !ok {
|
|
|
|
el = "..."
|
|
|
|
}
|
|
|
|
if len(s) > n {
|
|
|
|
s = s[:n-len(el)] + el
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("upcase", func(s, suffix string) string {
|
|
|
|
return strings.ToUpper(s)
|
|
|
|
})
|
2017-06-27 18:06:24 +02:00
|
|
|
|
2017-06-29 02:49:27 +02:00
|
|
|
// debugging extensions
|
|
|
|
// inspect is from Jekyll
|
|
|
|
expressions.DefineFilter("inspect", func(value interface{}) string {
|
|
|
|
s, err := json.Marshal(value)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Sprintf("%#v", value)
|
|
|
|
}
|
|
|
|
return string(s)
|
|
|
|
})
|
|
|
|
expressions.DefineFilter("type", func(value interface{}) string {
|
|
|
|
return reflect.TypeOf(value).String()
|
|
|
|
})
|
2017-06-27 18:06:24 +02:00
|
|
|
}
|
|
|
|
|
2017-06-29 19:41:04 +02:00
|
|
|
func joinFilter(array []interface{}, sep interface{}) interface{} {
|
|
|
|
a := make([]string, len(array))
|
2017-06-27 18:06:24 +02:00
|
|
|
s := ", "
|
|
|
|
if sep != nil {
|
|
|
|
s = fmt.Sprint(sep)
|
|
|
|
}
|
2017-06-29 19:41:04 +02:00
|
|
|
for i, x := range array {
|
2017-06-27 18:06:24 +02:00
|
|
|
a[i] = fmt.Sprint(x)
|
|
|
|
}
|
|
|
|
return strings.Join(a, s)
|
|
|
|
}
|
|
|
|
|
2017-06-29 19:41:04 +02:00
|
|
|
func reverseFilter(array []interface{}) interface{} {
|
|
|
|
out := make([]interface{}, len(array))
|
|
|
|
for i, x := range array {
|
2017-06-27 23:29:50 +02:00
|
|
|
out[len(out)-1-i] = x
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2017-06-29 19:41:04 +02:00
|
|
|
func sortFilter(array []interface{}, key interface{}) []interface{} {
|
|
|
|
out := make([]interface{}, len(array))
|
|
|
|
copy(out, array)
|
2017-06-27 18:06:24 +02:00
|
|
|
if key == nil {
|
|
|
|
generics.Sort(out)
|
|
|
|
} else {
|
2017-06-29 14:22:12 +02:00
|
|
|
generics.SortByProperty(out, key.(string), true)
|
2017-06-27 18:06:24 +02:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2017-06-29 19:41:04 +02:00
|
|
|
func splitFilter(s, sep string) interface{} {
|
|
|
|
out := strings.Split(s, sep)
|
|
|
|
// This matches Jekyll's observed behavior.
|
|
|
|
// TODO test case
|
|
|
|
if len(out) > 0 && out[len(out)-1] == "" {
|
|
|
|
out = out[:len(out)-1]
|
|
|
|
}
|
|
|
|
return out
|
2017-06-27 18:06:24 +02:00
|
|
|
}
|