2024-02-24 21:00:04 +00:00
|
|
|
package otto
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/hex"
|
2024-04-04 14:46:14 +00:00
|
|
|
"errors"
|
2024-02-24 21:00:04 +00:00
|
|
|
"math"
|
|
|
|
"net/url"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"unicode/utf16"
|
|
|
|
"unicode/utf8"
|
|
|
|
)
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
// Global.
|
|
|
|
func builtinGlobalEval(call FunctionCall) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
src := call.Argument(0)
|
|
|
|
if !src.IsString() {
|
|
|
|
return src
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
rt := call.runtime
|
|
|
|
program := rt.cmplParseOrThrow(src.string(), nil)
|
2024-02-24 21:00:04 +00:00
|
|
|
if !call.eval {
|
|
|
|
// Not a direct call to eval, so we enter the global ExecutionContext
|
2024-04-04 14:46:14 +00:00
|
|
|
rt.enterGlobalScope()
|
|
|
|
defer rt.leaveScope()
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
returnValue := rt.cmplEvaluateNodeProgram(program, true)
|
2024-02-24 21:00:04 +00:00
|
|
|
if returnValue.isEmpty() {
|
|
|
|
return Value{}
|
|
|
|
}
|
|
|
|
return returnValue
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalIsNaN(call FunctionCall) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
value := call.Argument(0).float64()
|
2024-04-04 14:46:14 +00:00
|
|
|
return boolValue(math.IsNaN(value))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalIsFinite(call FunctionCall) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
value := call.Argument(0).float64()
|
2024-04-04 14:46:14 +00:00
|
|
|
return boolValue(!math.IsNaN(value) && !math.IsInf(value, 0))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func digitValue(chr rune) int {
|
|
|
|
switch {
|
|
|
|
case '0' <= chr && chr <= '9':
|
|
|
|
return int(chr - '0')
|
|
|
|
case 'a' <= chr && chr <= 'z':
|
|
|
|
return int(chr - 'a' + 10)
|
|
|
|
case 'A' <= chr && chr <= 'Z':
|
|
|
|
return int(chr - 'A' + 10)
|
|
|
|
}
|
|
|
|
return 36 // Larger than any legal digit value
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalParseInt(call FunctionCall) Value {
|
|
|
|
input := strings.Trim(call.Argument(0).string(), builtinStringTrimWhitespace)
|
2024-02-24 21:00:04 +00:00
|
|
|
if len(input) == 0 {
|
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
|
|
|
|
radix := int(toInt32(call.Argument(1)))
|
|
|
|
|
|
|
|
negative := false
|
|
|
|
switch input[0] {
|
|
|
|
case '+':
|
|
|
|
input = input[1:]
|
|
|
|
case '-':
|
|
|
|
negative = true
|
|
|
|
input = input[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
strip := true
|
|
|
|
if radix == 0 {
|
|
|
|
radix = 10
|
|
|
|
} else {
|
|
|
|
if radix < 2 || radix > 36 {
|
|
|
|
return NaNValue()
|
|
|
|
} else if radix != 16 {
|
|
|
|
strip = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch len(input) {
|
|
|
|
case 0:
|
|
|
|
return NaNValue()
|
|
|
|
case 1:
|
|
|
|
default:
|
|
|
|
if strip {
|
|
|
|
if input[0] == '0' && (input[1] == 'x' || input[1] == 'X') {
|
|
|
|
input = input[2:]
|
|
|
|
radix = 16
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
base := radix
|
|
|
|
index := 0
|
|
|
|
for ; index < len(input); index++ {
|
|
|
|
digit := digitValue(rune(input[index])) // If not ASCII, then an error anyway
|
|
|
|
if digit >= base {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
input = input[0:index]
|
|
|
|
|
|
|
|
value, err := strconv.ParseInt(input, radix, 64)
|
|
|
|
if err != nil {
|
2024-04-04 14:46:14 +00:00
|
|
|
if errors.Is(err, strconv.ErrRange) {
|
2024-02-24 21:00:04 +00:00
|
|
|
base := float64(base)
|
|
|
|
// Could just be a very large number (e.g. 0x8000000000000000)
|
|
|
|
var value float64
|
|
|
|
for _, chr := range input {
|
|
|
|
digit := float64(digitValue(chr))
|
|
|
|
if digit >= base {
|
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
value = value*base + digit
|
|
|
|
}
|
|
|
|
if negative {
|
|
|
|
value *= -1
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return float64Value(value)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
if negative {
|
|
|
|
value *= -1
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
return int64Value(value)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
var (
|
|
|
|
parseFloatMatchBadSpecial = regexp.MustCompile(`[\+\-]?(?:[Ii]nf$|infinity)`)
|
|
|
|
parseFloatMatchValid = regexp.MustCompile(`[0-9eE\+\-\.]|Infinity`)
|
|
|
|
)
|
2024-02-24 21:00:04 +00:00
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalParseFloat(call FunctionCall) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
// Caveat emptor: This implementation does NOT match the specification
|
2024-04-04 14:46:14 +00:00
|
|
|
input := strings.Trim(call.Argument(0).string(), builtinStringTrimWhitespace)
|
2024-02-24 21:00:04 +00:00
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
if parseFloatMatchBadSpecial.MatchString(input) {
|
2024-02-24 21:00:04 +00:00
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
value, err := strconv.ParseFloat(input, 64)
|
|
|
|
if err != nil {
|
|
|
|
for end := len(input); end > 0; end-- {
|
|
|
|
input := input[0:end]
|
2024-04-04 14:46:14 +00:00
|
|
|
if !parseFloatMatchValid.MatchString(input) {
|
2024-02-24 21:00:04 +00:00
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
value, err = strconv.ParseFloat(input, 64)
|
|
|
|
if err == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return float64Value(value)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// encodeURI/decodeURI
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func encodeDecodeURI(call FunctionCall, escape *regexp.Regexp) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
value := call.Argument(0)
|
|
|
|
var input []uint16
|
|
|
|
switch vl := value.value.(type) {
|
|
|
|
case []uint16:
|
|
|
|
input = vl
|
|
|
|
default:
|
|
|
|
input = utf16.Encode([]rune(value.string()))
|
|
|
|
}
|
|
|
|
if len(input) == 0 {
|
2024-04-04 14:46:14 +00:00
|
|
|
return stringValue("")
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
output := []byte{}
|
|
|
|
length := len(input)
|
|
|
|
encode := make([]byte, 4)
|
|
|
|
for index := 0; index < length; {
|
|
|
|
value := input[index]
|
|
|
|
decode := utf16.Decode(input[index : index+1])
|
|
|
|
if value >= 0xDC00 && value <= 0xDFFF {
|
|
|
|
panic(call.runtime.panicURIError("URI malformed"))
|
|
|
|
}
|
|
|
|
if value >= 0xD800 && value <= 0xDBFF {
|
2024-04-04 14:46:14 +00:00
|
|
|
index++
|
2024-02-24 21:00:04 +00:00
|
|
|
if index >= length {
|
|
|
|
panic(call.runtime.panicURIError("URI malformed"))
|
|
|
|
}
|
|
|
|
// input = ..., value, value1, ...
|
|
|
|
value1 := input[index]
|
|
|
|
if value1 < 0xDC00 || value1 > 0xDFFF {
|
|
|
|
panic(call.runtime.panicURIError("URI malformed"))
|
|
|
|
}
|
|
|
|
decode = []rune{((rune(value) - 0xD800) * 0x400) + (rune(value1) - 0xDC00) + 0x10000}
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
index++
|
2024-02-24 21:00:04 +00:00
|
|
|
size := utf8.EncodeRune(encode, decode[0])
|
|
|
|
encode := encode[0:size]
|
|
|
|
output = append(output, encode...)
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
|
|
|
|
bytes := escape.ReplaceAllFunc(output, func(target []byte) []byte {
|
|
|
|
// Probably a better way of doing this
|
|
|
|
if target[0] == ' ' {
|
|
|
|
return []byte("%20")
|
|
|
|
}
|
|
|
|
return []byte(url.QueryEscape(string(target)))
|
|
|
|
})
|
|
|
|
return stringValue(string(bytes))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
var encodeURIRegexp = regexp.MustCompile(`([^~!@#$&*()=:/,;?+'])`)
|
2024-02-24 21:00:04 +00:00
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalEncodeURI(call FunctionCall) Value {
|
|
|
|
return encodeDecodeURI(call, encodeURIRegexp)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
var encodeURIComponentRegexp = regexp.MustCompile(`([^~!*()'])`)
|
2024-02-24 21:00:04 +00:00
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalEncodeURIComponent(call FunctionCall) Value {
|
|
|
|
return encodeDecodeURI(call, encodeURIComponentRegexp)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
// 3B/2F/3F/3A/40/26/3D/2B/24/2C/23.
|
|
|
|
var decodeURIGuard = regexp.MustCompile(`(?i)(?:%)(3B|2F|3F|3A|40|26|3D|2B|24|2C|23)`)
|
2024-02-24 21:00:04 +00:00
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func decodeURI(input string, reserve bool) (string, bool) {
|
2024-02-24 21:00:04 +00:00
|
|
|
if reserve {
|
2024-04-04 14:46:14 +00:00
|
|
|
input = decodeURIGuard.ReplaceAllString(input, "%25$1")
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
input = strings.ReplaceAll(input, "+", "%2B") // Ugly hack to make QueryUnescape work with our use case
|
2024-02-24 21:00:04 +00:00
|
|
|
output, err := url.QueryUnescape(input)
|
|
|
|
if err != nil || !utf8.ValidString(output) {
|
|
|
|
return "", true
|
|
|
|
}
|
|
|
|
return output, false
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalDecodeURI(call FunctionCall) Value {
|
|
|
|
output, err := decodeURI(call.Argument(0).string(), true)
|
2024-02-24 21:00:04 +00:00
|
|
|
if err {
|
|
|
|
panic(call.runtime.panicURIError("URI malformed"))
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return stringValue(output)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalDecodeURIComponent(call FunctionCall) Value {
|
|
|
|
output, err := decodeURI(call.Argument(0).string(), false)
|
2024-02-24 21:00:04 +00:00
|
|
|
if err {
|
|
|
|
panic(call.runtime.panicURIError("URI malformed"))
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return stringValue(output)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// escape/unescape
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinShouldEscape(chr byte) bool {
|
2024-02-24 21:00:04 +00:00
|
|
|
if 'A' <= chr && chr <= 'Z' || 'a' <= chr && chr <= 'z' || '0' <= chr && chr <= '9' {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return !strings.ContainsRune("*_+-./", rune(chr))
|
|
|
|
}
|
|
|
|
|
|
|
|
const escapeBase16 = "0123456789ABCDEF"
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinEscape(input string) string {
|
2024-02-24 21:00:04 +00:00
|
|
|
output := make([]byte, 0, len(input))
|
|
|
|
length := len(input)
|
|
|
|
for index := 0; index < length; {
|
2024-04-04 14:46:14 +00:00
|
|
|
if builtinShouldEscape(input[index]) {
|
2024-02-24 21:00:04 +00:00
|
|
|
chr, width := utf8.DecodeRuneInString(input[index:])
|
|
|
|
chr16 := utf16.Encode([]rune{chr})[0]
|
|
|
|
if 256 > chr16 {
|
|
|
|
output = append(output, '%',
|
|
|
|
escapeBase16[chr16>>4],
|
|
|
|
escapeBase16[chr16&15],
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
output = append(output, '%', 'u',
|
|
|
|
escapeBase16[chr16>>12],
|
|
|
|
escapeBase16[(chr16>>8)&15],
|
|
|
|
escapeBase16[(chr16>>4)&15],
|
|
|
|
escapeBase16[chr16&15],
|
|
|
|
)
|
|
|
|
}
|
|
|
|
index += width
|
|
|
|
} else {
|
|
|
|
output = append(output, input[index])
|
2024-04-04 14:46:14 +00:00
|
|
|
index++
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return string(output)
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinUnescape(input string) string {
|
2024-02-24 21:00:04 +00:00
|
|
|
output := make([]rune, 0, len(input))
|
|
|
|
length := len(input)
|
|
|
|
for index := 0; index < length; {
|
|
|
|
if input[index] == '%' {
|
|
|
|
if index <= length-6 && input[index+1] == 'u' {
|
|
|
|
byte16, err := hex.DecodeString(input[index+2 : index+6])
|
|
|
|
if err == nil {
|
|
|
|
value := uint16(byte16[0])<<8 + uint16(byte16[1])
|
|
|
|
chr := utf16.Decode([]uint16{value})[0]
|
|
|
|
output = append(output, chr)
|
|
|
|
index += 6
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if index <= length-3 {
|
|
|
|
byte8, err := hex.DecodeString(input[index+1 : index+3])
|
|
|
|
if err == nil {
|
|
|
|
value := uint16(byte8[0])
|
|
|
|
chr := utf16.Decode([]uint16{value})[0]
|
|
|
|
output = append(output, chr)
|
|
|
|
index += 3
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
output = append(output, rune(input[index]))
|
2024-04-04 14:46:14 +00:00
|
|
|
index++
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
return string(output)
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalEscape(call FunctionCall) Value {
|
|
|
|
return stringValue(builtinEscape(call.Argument(0).string()))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinGlobalUnescape(call FunctionCall) Value {
|
|
|
|
return stringValue(builtinUnescape(call.Argument(0).string()))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|