/src/mongo/gotools/src/github.com/mongodb/mongo-tools/legacy/json/new.go

https://github.com/mongodb/mongo · Go · 98 lines · 69 code · 13 blank · 16 comment · 15 complexity · 3d74696302ebb9fbe6870b0266ea9c4f MD5 · raw file

  1. // Copyright (C) MongoDB, Inc. 2014-present.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License"); you may
  4. // not use this file except in compliance with the License. You may obtain
  5. // a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
  6. package json
  7. import (
  8. "fmt"
  9. "reflect"
  10. )
  11. // Transition functions for recognizing new.
  12. // Adapted from encoding/json/scanner.go.
  13. // stateNe is the state after reading `ne`.
  14. func stateNe(s *scanner, c int) int {
  15. if c == 'w' {
  16. s.step = stateNew
  17. return scanContinue
  18. }
  19. return s.error(c, "in literal new (expecting 'w')")
  20. }
  21. // stateNew is the state after reading `new`.
  22. // Ensures that there is a space after the new keyword.
  23. func stateNew(s *scanner, c int) int {
  24. if c <= ' ' && isSpace(rune(c)) {
  25. s.step = stateBeginObjectValue
  26. return scanContinue
  27. }
  28. return s.error(c, "expected space after new keyword")
  29. }
  30. // stateBeginObjectValue is the state after reading `new`.
  31. func stateBeginObjectValue(s *scanner, c int) int {
  32. if c <= ' ' && isSpace(rune(c)) {
  33. return scanSkipSpace
  34. }
  35. switch c {
  36. case 'B': // beginning of BinData or Boolean
  37. s.step = stateB
  38. case 'D': // beginning of Date
  39. s.step = stateD
  40. case 'N': // beginning of NumberInt or NumberLong
  41. s.step = stateNumberUpperN
  42. case 'O': // beginning of ObjectId
  43. s.step = stateO
  44. case 'R': // beginning of RegExp
  45. s.step = stateR
  46. case 'T': // beginning of Timestamp
  47. s.step = stateUpperT
  48. default:
  49. return s.error(c, "looking for beginning of value")
  50. }
  51. return scanBeginLiteral
  52. }
  53. // stateNumberUpperN is the state after reading `N`.
  54. func stateNumberUpperN(s *scanner, c int) int {
  55. if c == 'u' {
  56. s.step = stateUpperNu
  57. return scanContinue
  58. }
  59. return s.error(c, "in literal NumberInt or NumberLong (expecting 'u')")
  60. }
  61. // Decodes a literal stored in the underlying byte data into v.
  62. func (d *decodeState) storeNewLiteral(v reflect.Value, fromQuoted bool) {
  63. op := d.scanWhile(scanSkipSpace)
  64. if op != scanBeginLiteral {
  65. d.error(fmt.Errorf("expected beginning of constructor"))
  66. }
  67. // Read constructor identifier
  68. start := d.off - 1
  69. op = d.scanWhile(scanContinue)
  70. if op != scanBeginCtor {
  71. d.error(fmt.Errorf("expected beginning of constructor"))
  72. }
  73. // Back up so d.ctor can have the byte we just read.
  74. d.off--
  75. d.scan.undo(op)
  76. d.literalStore(d.data[start:d.off-1], v, fromQuoted)
  77. }
  78. // Returns a literal from the underlying byte data.
  79. func (d *decodeState) getNewLiteral() interface{} {
  80. op := d.scanWhile(scanSkipSpace)
  81. if op != scanBeginLiteral {
  82. d.error(fmt.Errorf("expected beginning of constructor"))
  83. }
  84. return d.literalInterface()
  85. }