string.go 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. package goja
  2. import (
  3. "io"
  4. "strconv"
  5. "strings"
  6. "unicode/utf16"
  7. "unicode/utf8"
  8. "github.com/dop251/goja/unistring"
  9. )
  10. const (
  11. __proto__ = "__proto__"
  12. )
  13. var (
  14. stringTrue valueString = asciiString("true")
  15. stringFalse valueString = asciiString("false")
  16. stringNull valueString = asciiString("null")
  17. stringUndefined valueString = asciiString("undefined")
  18. stringObjectC valueString = asciiString("object")
  19. stringFunction valueString = asciiString("function")
  20. stringBoolean valueString = asciiString("boolean")
  21. stringString valueString = asciiString("string")
  22. stringSymbol valueString = asciiString("symbol")
  23. stringNumber valueString = asciiString("number")
  24. stringNaN valueString = asciiString("NaN")
  25. stringInfinity = asciiString("Infinity")
  26. stringNegInfinity = asciiString("-Infinity")
  27. stringBound_ valueString = asciiString("bound ")
  28. stringEmpty valueString = asciiString("")
  29. stringError valueString = asciiString("Error")
  30. stringAggregateError valueString = asciiString("AggregateError")
  31. stringTypeError valueString = asciiString("TypeError")
  32. stringReferenceError valueString = asciiString("ReferenceError")
  33. stringSyntaxError valueString = asciiString("SyntaxError")
  34. stringRangeError valueString = asciiString("RangeError")
  35. stringEvalError valueString = asciiString("EvalError")
  36. stringURIError valueString = asciiString("URIError")
  37. stringGoError valueString = asciiString("GoError")
  38. stringObjectNull valueString = asciiString("[object Null]")
  39. stringObjectUndefined valueString = asciiString("[object Undefined]")
  40. stringInvalidDate valueString = asciiString("Invalid Date")
  41. )
  42. type valueString interface {
  43. Value
  44. charAt(int) rune
  45. length() int
  46. concat(valueString) valueString
  47. substring(start, end int) valueString
  48. compareTo(valueString) int
  49. reader(start int) io.RuneReader
  50. utf16Reader(start int) io.RuneReader
  51. utf16Runes() []rune
  52. index(valueString, int) int
  53. lastIndex(valueString, int) int
  54. toLower() valueString
  55. toUpper() valueString
  56. toTrimmedUTF8() string
  57. }
  58. type stringIterObject struct {
  59. baseObject
  60. reader io.RuneReader
  61. }
  62. func isUTF16FirstSurrogate(r rune) bool {
  63. return r >= 0xD800 && r <= 0xDBFF
  64. }
  65. func isUTF16SecondSurrogate(r rune) bool {
  66. return r >= 0xDC00 && r <= 0xDFFF
  67. }
  68. func (si *stringIterObject) next() Value {
  69. if si.reader == nil {
  70. return si.val.runtime.createIterResultObject(_undefined, true)
  71. }
  72. r, _, err := si.reader.ReadRune()
  73. if err == io.EOF {
  74. si.reader = nil
  75. return si.val.runtime.createIterResultObject(_undefined, true)
  76. }
  77. return si.val.runtime.createIterResultObject(stringFromRune(r), false)
  78. }
  79. func stringFromRune(r rune) valueString {
  80. if r < utf8.RuneSelf {
  81. var sb strings.Builder
  82. sb.Grow(1)
  83. sb.WriteByte(byte(r))
  84. return asciiString(sb.String())
  85. }
  86. var sb unicodeStringBuilder
  87. if r <= 0xFFFF {
  88. sb.Grow(1)
  89. } else {
  90. sb.Grow(2)
  91. }
  92. sb.WriteRune(r)
  93. return sb.String()
  94. }
  95. func (r *Runtime) createStringIterator(s valueString) Value {
  96. o := &Object{runtime: r}
  97. si := &stringIterObject{
  98. reader: &lenientUtf16Decoder{utf16Reader: s.utf16Reader(0)},
  99. }
  100. si.class = classStringIterator
  101. si.val = o
  102. si.extensible = true
  103. o.self = si
  104. si.prototype = r.global.StringIteratorPrototype
  105. si.init()
  106. return o
  107. }
  108. type stringObject struct {
  109. baseObject
  110. value valueString
  111. length int
  112. lengthProp valueProperty
  113. }
  114. func newStringValue(s string) valueString {
  115. utf16Size := 0
  116. ascii := true
  117. for _, chr := range s {
  118. utf16Size++
  119. if chr >= utf8.RuneSelf {
  120. ascii = false
  121. if chr > 0xFFFF {
  122. utf16Size++
  123. }
  124. }
  125. }
  126. if ascii {
  127. return asciiString(s)
  128. }
  129. buf := make([]uint16, utf16Size+1)
  130. buf[0] = unistring.BOM
  131. c := 1
  132. for _, chr := range s {
  133. if chr <= 0xFFFF {
  134. buf[c] = uint16(chr)
  135. } else {
  136. first, second := utf16.EncodeRune(chr)
  137. buf[c] = uint16(first)
  138. c++
  139. buf[c] = uint16(second)
  140. }
  141. c++
  142. }
  143. return unicodeString(buf)
  144. }
  145. func stringValueFromRaw(raw unistring.String) valueString {
  146. if b := raw.AsUtf16(); b != nil {
  147. return unicodeString(b)
  148. }
  149. return asciiString(raw)
  150. }
  151. func (s *stringObject) init() {
  152. s.baseObject.init()
  153. s.setLength()
  154. }
  155. func (s *stringObject) setLength() {
  156. if s.value != nil {
  157. s.length = s.value.length()
  158. }
  159. s.lengthProp.value = intToValue(int64(s.length))
  160. s._put("length", &s.lengthProp)
  161. }
  162. func (s *stringObject) getStr(name unistring.String, receiver Value) Value {
  163. if i := strToGoIdx(name); i >= 0 && i < s.length {
  164. return s._getIdx(i)
  165. }
  166. return s.baseObject.getStr(name, receiver)
  167. }
  168. func (s *stringObject) getIdx(idx valueInt, receiver Value) Value {
  169. i := int(idx)
  170. if i >= 0 && i < s.length {
  171. return s._getIdx(i)
  172. }
  173. return s.baseObject.getStr(idx.string(), receiver)
  174. }
  175. func (s *stringObject) getOwnPropStr(name unistring.String) Value {
  176. if i := strToGoIdx(name); i >= 0 && i < s.length {
  177. val := s._getIdx(i)
  178. return &valueProperty{
  179. value: val,
  180. enumerable: true,
  181. }
  182. }
  183. return s.baseObject.getOwnPropStr(name)
  184. }
  185. func (s *stringObject) getOwnPropIdx(idx valueInt) Value {
  186. i := int64(idx)
  187. if i >= 0 {
  188. if i < int64(s.length) {
  189. val := s._getIdx(int(i))
  190. return &valueProperty{
  191. value: val,
  192. enumerable: true,
  193. }
  194. }
  195. return nil
  196. }
  197. return s.baseObject.getOwnPropStr(idx.string())
  198. }
  199. func (s *stringObject) _getIdx(idx int) Value {
  200. return s.value.substring(idx, idx+1)
  201. }
  202. func (s *stringObject) setOwnStr(name unistring.String, val Value, throw bool) bool {
  203. if i := strToGoIdx(name); i >= 0 && i < s.length {
  204. s.val.runtime.typeErrorResult(throw, "Cannot assign to read only property '%d' of a String", i)
  205. return false
  206. }
  207. return s.baseObject.setOwnStr(name, val, throw)
  208. }
  209. func (s *stringObject) setOwnIdx(idx valueInt, val Value, throw bool) bool {
  210. i := int64(idx)
  211. if i >= 0 && i < int64(s.length) {
  212. s.val.runtime.typeErrorResult(throw, "Cannot assign to read only property '%d' of a String", i)
  213. return false
  214. }
  215. return s.baseObject.setOwnStr(idx.string(), val, throw)
  216. }
  217. func (s *stringObject) setForeignStr(name unistring.String, val, receiver Value, throw bool) (bool, bool) {
  218. return s._setForeignStr(name, s.getOwnPropStr(name), val, receiver, throw)
  219. }
  220. func (s *stringObject) setForeignIdx(idx valueInt, val, receiver Value, throw bool) (bool, bool) {
  221. return s._setForeignIdx(idx, s.getOwnPropIdx(idx), val, receiver, throw)
  222. }
  223. func (s *stringObject) defineOwnPropertyStr(name unistring.String, descr PropertyDescriptor, throw bool) bool {
  224. if i := strToGoIdx(name); i >= 0 && i < s.length {
  225. _, ok := s._defineOwnProperty(name, &valueProperty{enumerable: true}, descr, throw)
  226. return ok
  227. }
  228. return s.baseObject.defineOwnPropertyStr(name, descr, throw)
  229. }
  230. func (s *stringObject) defineOwnPropertyIdx(idx valueInt, descr PropertyDescriptor, throw bool) bool {
  231. i := int64(idx)
  232. if i >= 0 && i < int64(s.length) {
  233. s.val.runtime.typeErrorResult(throw, "Cannot redefine property: %d", i)
  234. return false
  235. }
  236. return s.baseObject.defineOwnPropertyStr(idx.string(), descr, throw)
  237. }
  238. type stringPropIter struct {
  239. str valueString // separate, because obj can be the singleton
  240. obj *stringObject
  241. idx, length int
  242. }
  243. func (i *stringPropIter) next() (propIterItem, iterNextFunc) {
  244. if i.idx < i.length {
  245. name := strconv.Itoa(i.idx)
  246. i.idx++
  247. return propIterItem{name: asciiString(name), enumerable: _ENUM_TRUE}, i.next
  248. }
  249. return i.obj.baseObject.iterateStringKeys()()
  250. }
  251. func (s *stringObject) iterateStringKeys() iterNextFunc {
  252. return (&stringPropIter{
  253. str: s.value,
  254. obj: s,
  255. length: s.length,
  256. }).next
  257. }
  258. func (s *stringObject) stringKeys(all bool, accum []Value) []Value {
  259. for i := 0; i < s.length; i++ {
  260. accum = append(accum, asciiString(strconv.Itoa(i)))
  261. }
  262. return s.baseObject.stringKeys(all, accum)
  263. }
  264. func (s *stringObject) deleteStr(name unistring.String, throw bool) bool {
  265. if i := strToGoIdx(name); i >= 0 && i < s.length {
  266. s.val.runtime.typeErrorResult(throw, "Cannot delete property '%d' of a String", i)
  267. return false
  268. }
  269. return s.baseObject.deleteStr(name, throw)
  270. }
  271. func (s *stringObject) deleteIdx(idx valueInt, throw bool) bool {
  272. i := int64(idx)
  273. if i >= 0 && i < int64(s.length) {
  274. s.val.runtime.typeErrorResult(throw, "Cannot delete property '%d' of a String", i)
  275. return false
  276. }
  277. return s.baseObject.deleteStr(idx.string(), throw)
  278. }
  279. func (s *stringObject) hasOwnPropertyStr(name unistring.String) bool {
  280. if i := strToGoIdx(name); i >= 0 && i < s.length {
  281. return true
  282. }
  283. return s.baseObject.hasOwnPropertyStr(name)
  284. }
  285. func (s *stringObject) hasOwnPropertyIdx(idx valueInt) bool {
  286. i := int64(idx)
  287. if i >= 0 && i < int64(s.length) {
  288. return true
  289. }
  290. return s.baseObject.hasOwnPropertyStr(idx.string())
  291. }