2024-02-24 21:00:04 +00:00
|
|
|
package otto
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
)
|
|
|
|
|
|
|
|
// RegExp
|
|
|
|
|
|
|
|
func builtinRegExp(call FunctionCall) Value {
|
|
|
|
pattern := call.Argument(0)
|
|
|
|
flags := call.Argument(1)
|
2024-04-04 14:46:14 +00:00
|
|
|
if obj := pattern.object(); obj != nil {
|
|
|
|
if obj.class == classRegExpName && flags.IsUndefined() {
|
2024-02-24 21:00:04 +00:00
|
|
|
return pattern
|
|
|
|
}
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return objectValue(call.runtime.newRegExp(pattern, flags))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinNewRegExp(obj *object, argumentList []Value) Value {
|
|
|
|
return objectValue(obj.runtime.newRegExp(
|
2024-02-24 21:00:04 +00:00
|
|
|
valueOfArrayIndex(argumentList, 0),
|
|
|
|
valueOfArrayIndex(argumentList, 1),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinRegExpToString(call FunctionCall) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
thisObject := call.thisObject()
|
|
|
|
source := thisObject.get("source").string()
|
|
|
|
flags := []byte{}
|
|
|
|
if thisObject.get("global").bool() {
|
|
|
|
flags = append(flags, 'g')
|
|
|
|
}
|
|
|
|
if thisObject.get("ignoreCase").bool() {
|
|
|
|
flags = append(flags, 'i')
|
|
|
|
}
|
|
|
|
if thisObject.get("multiline").bool() {
|
|
|
|
flags = append(flags, 'm')
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return stringValue(fmt.Sprintf("/%s/%s", source, flags))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinRegExpExec(call FunctionCall) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
thisObject := call.thisObject()
|
|
|
|
target := call.Argument(0).string()
|
|
|
|
match, result := execRegExp(thisObject, target)
|
|
|
|
if !match {
|
|
|
|
return nullValue
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return objectValue(execResultToArray(call.runtime, target, result))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinRegExpTest(call FunctionCall) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
thisObject := call.thisObject()
|
|
|
|
target := call.Argument(0).string()
|
|
|
|
match, result := execRegExp(thisObject, target)
|
|
|
|
|
|
|
|
if !match {
|
2024-04-04 14:46:14 +00:00
|
|
|
return boolValue(match)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Match extract and assign input, $_ and $1 -> $9 on global RegExp.
|
2024-04-04 14:46:14 +00:00
|
|
|
input := stringValue(target)
|
|
|
|
call.runtime.global.RegExp.defineProperty("$_", input, 0o100, false)
|
|
|
|
call.runtime.global.RegExp.defineProperty("input", input, 0o100, false)
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
var start int
|
|
|
|
n := 1
|
|
|
|
re := call.runtime.global.RegExp
|
2024-04-04 14:46:14 +00:00
|
|
|
empty := stringValue("")
|
2024-02-24 21:00:04 +00:00
|
|
|
for i, v := range result[2:] {
|
|
|
|
if i%2 == 0 {
|
|
|
|
start = v
|
|
|
|
} else {
|
2024-04-04 14:46:14 +00:00
|
|
|
if v == -1 {
|
|
|
|
// No match for this part.
|
|
|
|
re.defineProperty(fmt.Sprintf("$%d", n), empty, 0o100, false)
|
|
|
|
} else {
|
|
|
|
re.defineProperty(fmt.Sprintf("$%d", n), stringValue(target[start:v]), 0o100, false)
|
|
|
|
}
|
2024-02-24 21:00:04 +00:00
|
|
|
n++
|
|
|
|
if n == 10 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if n <= 9 {
|
|
|
|
// Erase remaining.
|
|
|
|
for i := n; i <= 9; i++ {
|
2024-04-04 14:46:14 +00:00
|
|
|
re.defineProperty(fmt.Sprintf("$%d", i), empty, 0o100, false)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
return boolValue(match)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func builtinRegExpCompile(call FunctionCall) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
// This (useless) function is deprecated, but is here to provide some
|
|
|
|
// semblance of compatibility.
|
|
|
|
// Caveat emptor: it may not be around for long.
|
|
|
|
return Value{}
|
|
|
|
}
|