679 lines
15 KiB
Go
679 lines
15 KiB
Go
package stdlib
|
|
|
|
import (
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/d5/tengo/objects"
|
|
)
|
|
|
|
var textModule = map[string]objects.Object{
|
|
// re_match(pattern, text) => bool/error
|
|
"re_match": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 2 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
s2, ok := objects.ToString(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
matched, err := regexp.MatchString(s1, s2)
|
|
if err != nil {
|
|
ret = wrapError(err)
|
|
return
|
|
}
|
|
|
|
if matched {
|
|
ret = objects.TrueValue
|
|
} else {
|
|
ret = objects.FalseValue
|
|
}
|
|
|
|
return
|
|
},
|
|
},
|
|
|
|
// re_find(pattern, text) => array(array({text:,begin:,end:}))/undefined
|
|
// re_find(pattern, text, maxCount) => array(array({text:,begin:,end:}))/undefined
|
|
"re_find": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
numArgs := len(args)
|
|
if numArgs != 2 && numArgs != 3 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
re, err := regexp.Compile(s1)
|
|
if err != nil {
|
|
ret = wrapError(err)
|
|
return
|
|
}
|
|
|
|
s2, ok := objects.ToString(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
if numArgs < 3 {
|
|
m := re.FindStringSubmatchIndex(s2)
|
|
if m == nil {
|
|
ret = objects.UndefinedValue
|
|
return
|
|
}
|
|
|
|
arr := &objects.Array{}
|
|
for i := 0; i < len(m); i += 2 {
|
|
arr.Value = append(arr.Value, &objects.ImmutableMap{Value: map[string]objects.Object{
|
|
"text": &objects.String{Value: s2[m[i]:m[i+1]]},
|
|
"begin": &objects.Int{Value: int64(m[i])},
|
|
"end": &objects.Int{Value: int64(m[i+1])},
|
|
}})
|
|
}
|
|
|
|
ret = &objects.Array{Value: []objects.Object{arr}}
|
|
|
|
return
|
|
}
|
|
|
|
i3, ok := objects.ToInt(args[2])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
m := re.FindAllStringSubmatchIndex(s2, i3)
|
|
if m == nil {
|
|
ret = objects.UndefinedValue
|
|
return
|
|
}
|
|
|
|
arr := &objects.Array{}
|
|
for _, m := range m {
|
|
subMatch := &objects.Array{}
|
|
for i := 0; i < len(m); i += 2 {
|
|
subMatch.Value = append(subMatch.Value, &objects.ImmutableMap{Value: map[string]objects.Object{
|
|
"text": &objects.String{Value: s2[m[i]:m[i+1]]},
|
|
"begin": &objects.Int{Value: int64(m[i])},
|
|
"end": &objects.Int{Value: int64(m[i+1])},
|
|
}})
|
|
}
|
|
|
|
arr.Value = append(arr.Value, subMatch)
|
|
}
|
|
|
|
ret = arr
|
|
|
|
return
|
|
},
|
|
},
|
|
|
|
// re_replace(pattern, text, repl) => string/error
|
|
"re_replace": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 3 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
s2, ok := objects.ToString(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
s3, ok := objects.ToString(args[2])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
re, err := regexp.Compile(s1)
|
|
if err != nil {
|
|
ret = wrapError(err)
|
|
} else {
|
|
ret = &objects.String{Value: re.ReplaceAllString(s2, s3)}
|
|
}
|
|
|
|
return
|
|
},
|
|
},
|
|
|
|
// re_split(pattern, text) => array(string)/error
|
|
// re_split(pattern, text, maxCount) => array(string)/error
|
|
"re_split": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
numArgs := len(args)
|
|
if numArgs != 2 && numArgs != 3 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
s2, ok := objects.ToString(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
var i3 = -1
|
|
if numArgs > 2 {
|
|
i3, ok = objects.ToInt(args[2])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
}
|
|
|
|
re, err := regexp.Compile(s1)
|
|
if err != nil {
|
|
ret = wrapError(err)
|
|
return
|
|
}
|
|
|
|
arr := &objects.Array{}
|
|
for _, s := range re.Split(s2, i3) {
|
|
arr.Value = append(arr.Value, &objects.String{Value: s})
|
|
}
|
|
|
|
ret = arr
|
|
|
|
return
|
|
},
|
|
},
|
|
|
|
// re_compile(pattern) => Regexp/error
|
|
"re_compile": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 1 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
re, err := regexp.Compile(s1)
|
|
if err != nil {
|
|
ret = wrapError(err)
|
|
} else {
|
|
ret = stringsRegexpImmutableMap(re)
|
|
}
|
|
|
|
return
|
|
},
|
|
},
|
|
|
|
// compare(a, b) => int
|
|
"compare": FuncASSRI(strings.Compare),
|
|
// contains(s, substr) => bool
|
|
"contains": FuncASSRB(strings.Contains),
|
|
// contains_any(s, chars) => bool
|
|
"contains_any": FuncASSRB(strings.ContainsAny),
|
|
// count(s, substr) => int
|
|
"count": FuncASSRI(strings.Count),
|
|
// "equal_fold(s, t) => bool
|
|
"equal_fold": FuncASSRB(strings.EqualFold),
|
|
// fields(s) => array(string)
|
|
"fields": FuncASRSs(strings.Fields),
|
|
// has_prefix(s, prefix) => bool
|
|
"has_prefix": FuncASSRB(strings.HasPrefix),
|
|
// has_suffix(s, suffix) => bool
|
|
"has_suffix": FuncASSRB(strings.HasSuffix),
|
|
// index(s, substr) => int
|
|
"index": FuncASSRI(strings.Index),
|
|
// index_any(s, chars) => int
|
|
"index_any": FuncASSRI(strings.IndexAny),
|
|
// join(arr, sep) => string
|
|
"join": FuncASsSRS(strings.Join),
|
|
// last_index(s, substr) => int
|
|
"last_index": FuncASSRI(strings.LastIndex),
|
|
// last_index_any(s, chars) => int
|
|
"last_index_any": FuncASSRI(strings.LastIndexAny),
|
|
// repeat(s, count) => string
|
|
"repeat": FuncASIRS(strings.Repeat),
|
|
// replace(s, old, new, n) => string
|
|
"replace": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 4 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
s2, ok := objects.ToString(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
s3, ok := objects.ToString(args[2])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
i4, ok := objects.ToInt(args[3])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
ret = &objects.String{Value: strings.Replace(s1, s2, s3, i4)}
|
|
|
|
return
|
|
},
|
|
},
|
|
// split(s, sep) => []string
|
|
"split": FuncASSRSs(strings.Split),
|
|
// split_after(s, sep) => []string
|
|
"split_after": FuncASSRSs(strings.SplitAfter),
|
|
// split_after_n(s, sep, n) => []string
|
|
"split_after_n": FuncASSIRSs(strings.SplitAfterN),
|
|
// split_n(s, sep, n) => []string
|
|
"split_n": FuncASSIRSs(strings.SplitN),
|
|
// title(s) => string
|
|
"title": FuncASRS(strings.Title),
|
|
// to_lower(s) => string
|
|
"to_lower": FuncASRS(strings.ToLower),
|
|
// to_title(s) => string
|
|
"to_title": FuncASRS(strings.ToTitle),
|
|
// to_upper(s) => string
|
|
"to_upper": FuncASRS(strings.ToUpper),
|
|
// trim_left(s, cutset) => string
|
|
"trim_left": FuncASSRS(strings.TrimLeft),
|
|
// trim_prefix(s, prefix) => string
|
|
"trim_prefix": FuncASSRS(strings.TrimPrefix),
|
|
// trim_right(s, cutset) => string
|
|
"trim_right": FuncASSRS(strings.TrimRight),
|
|
// trim_space(s) => string
|
|
"trim_space": FuncASRS(strings.TrimSpace),
|
|
// trim_suffix(s, suffix) => string
|
|
"trim_suffix": FuncASSRS(strings.TrimSuffix),
|
|
// atoi(str) => int/error
|
|
"atoi": FuncASRIE(strconv.Atoi),
|
|
// format_bool(b) => string
|
|
"format_bool": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 1 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
b1, ok := args[0].(*objects.Bool)
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
if b1 == objects.TrueValue {
|
|
ret = &objects.String{Value: "true"}
|
|
} else {
|
|
ret = &objects.String{Value: "false"}
|
|
}
|
|
|
|
return
|
|
},
|
|
},
|
|
// format_float(f, fmt, prec, bits) => string
|
|
"format_float": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 4 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
f1, ok := args[0].(*objects.Float)
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
s2, ok := objects.ToString(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
i3, ok := objects.ToInt(args[2])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
i4, ok := objects.ToInt(args[3])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
ret = &objects.String{Value: strconv.FormatFloat(f1.Value, s2[0], i3, i4)}
|
|
|
|
return
|
|
},
|
|
},
|
|
// format_int(i, base) => string
|
|
"format_int": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 2 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
i1, ok := args[0].(*objects.Int)
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
i2, ok := objects.ToInt(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
ret = &objects.String{Value: strconv.FormatInt(i1.Value, i2)}
|
|
|
|
return
|
|
},
|
|
},
|
|
// itoa(i) => string
|
|
"itoa": FuncAIRS(strconv.Itoa),
|
|
// parse_bool(str) => bool/error
|
|
"parse_bool": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 1 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := args[0].(*objects.String)
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
parsed, err := strconv.ParseBool(s1.Value)
|
|
if err != nil {
|
|
ret = wrapError(err)
|
|
return
|
|
}
|
|
|
|
if parsed {
|
|
ret = objects.TrueValue
|
|
} else {
|
|
ret = objects.FalseValue
|
|
}
|
|
|
|
return
|
|
},
|
|
},
|
|
// parse_float(str, bits) => float/error
|
|
"parse_float": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 2 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := args[0].(*objects.String)
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
i2, ok := objects.ToInt(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
parsed, err := strconv.ParseFloat(s1.Value, i2)
|
|
if err != nil {
|
|
ret = wrapError(err)
|
|
return
|
|
}
|
|
|
|
ret = &objects.Float{Value: parsed}
|
|
|
|
return
|
|
},
|
|
},
|
|
// parse_int(str, base, bits) => int/error
|
|
"parse_int": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 3 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := args[0].(*objects.String)
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
i2, ok := objects.ToInt(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
i3, ok := objects.ToInt(args[2])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
parsed, err := strconv.ParseInt(s1.Value, i2, i3)
|
|
if err != nil {
|
|
ret = wrapError(err)
|
|
return
|
|
}
|
|
|
|
ret = &objects.Int{Value: parsed}
|
|
|
|
return
|
|
},
|
|
},
|
|
// quote(str) => string
|
|
"quote": FuncASRS(strconv.Quote),
|
|
// unquote(str) => string/error
|
|
"unquote": FuncASRSE(strconv.Unquote),
|
|
}
|
|
|
|
func stringsRegexpImmutableMap(re *regexp.Regexp) *objects.ImmutableMap {
|
|
return &objects.ImmutableMap{
|
|
Value: map[string]objects.Object{
|
|
// match(text) => bool
|
|
"match": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 1 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
if re.MatchString(s1) {
|
|
ret = objects.TrueValue
|
|
} else {
|
|
ret = objects.FalseValue
|
|
}
|
|
|
|
return
|
|
},
|
|
},
|
|
|
|
// find(text) => array(array({text:,begin:,end:}))/undefined
|
|
// find(text, maxCount) => array(array({text:,begin:,end:}))/undefined
|
|
"find": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
numArgs := len(args)
|
|
if numArgs != 1 && numArgs != 2 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
if numArgs == 1 {
|
|
m := re.FindStringSubmatchIndex(s1)
|
|
if m == nil {
|
|
ret = objects.UndefinedValue
|
|
return
|
|
}
|
|
|
|
arr := &objects.Array{}
|
|
for i := 0; i < len(m); i += 2 {
|
|
arr.Value = append(arr.Value, &objects.ImmutableMap{Value: map[string]objects.Object{
|
|
"text": &objects.String{Value: s1[m[i]:m[i+1]]},
|
|
"begin": &objects.Int{Value: int64(m[i])},
|
|
"end": &objects.Int{Value: int64(m[i+1])},
|
|
}})
|
|
}
|
|
|
|
ret = &objects.Array{Value: []objects.Object{arr}}
|
|
|
|
return
|
|
}
|
|
|
|
i2, ok := objects.ToInt(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
m := re.FindAllStringSubmatchIndex(s1, i2)
|
|
if m == nil {
|
|
ret = objects.UndefinedValue
|
|
return
|
|
}
|
|
|
|
arr := &objects.Array{}
|
|
for _, m := range m {
|
|
subMatch := &objects.Array{}
|
|
for i := 0; i < len(m); i += 2 {
|
|
subMatch.Value = append(subMatch.Value, &objects.ImmutableMap{Value: map[string]objects.Object{
|
|
"text": &objects.String{Value: s1[m[i]:m[i+1]]},
|
|
"begin": &objects.Int{Value: int64(m[i])},
|
|
"end": &objects.Int{Value: int64(m[i+1])},
|
|
}})
|
|
}
|
|
|
|
arr.Value = append(arr.Value, subMatch)
|
|
}
|
|
|
|
ret = arr
|
|
|
|
return
|
|
},
|
|
},
|
|
|
|
// replace(src, repl) => string
|
|
"replace": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 2 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
s2, ok := objects.ToString(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
ret = &objects.String{Value: re.ReplaceAllString(s1, s2)}
|
|
|
|
return
|
|
},
|
|
},
|
|
|
|
// split(text) => array(string)
|
|
// split(text, maxCount) => array(string)
|
|
"split": &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
numArgs := len(args)
|
|
if numArgs != 1 && numArgs != 2 {
|
|
err = objects.ErrWrongNumArguments
|
|
return
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
|
|
var i2 = -1
|
|
if numArgs > 1 {
|
|
i2, ok = objects.ToInt(args[1])
|
|
if !ok {
|
|
err = objects.ErrInvalidTypeConversion
|
|
return
|
|
}
|
|
}
|
|
|
|
arr := &objects.Array{}
|
|
for _, s := range re.Split(s1, i2) {
|
|
arr.Value = append(arr.Value, &objects.String{Value: s})
|
|
}
|
|
|
|
ret = arr
|
|
|
|
return
|
|
},
|
|
},
|
|
},
|
|
}
|
|
}
|