json 1.8.6 → 2.7.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (107) hide show
  1. checksums.yaml +5 -5
  2. data/{CHANGES → CHANGES.md} +292 -96
  3. data/LICENSE +56 -0
  4. data/README.md +185 -114
  5. data/ext/json/ext/fbuffer/fbuffer.h +0 -3
  6. data/ext/json/ext/generator/generator.c +328 -117
  7. data/ext/json/ext/generator/generator.h +8 -8
  8. data/ext/json/ext/parser/extconf.rb +29 -0
  9. data/ext/json/ext/parser/parser.c +540 -569
  10. data/ext/json/ext/parser/parser.h +10 -6
  11. data/ext/json/ext/parser/parser.rl +269 -261
  12. data/ext/json/extconf.rb +1 -1
  13. data/json.gemspec +0 -0
  14. data/lib/json/add/bigdecimal.rb +40 -10
  15. data/lib/json/add/complex.rb +32 -9
  16. data/lib/json/add/core.rb +1 -0
  17. data/lib/json/add/date.rb +27 -7
  18. data/lib/json/add/date_time.rb +26 -9
  19. data/lib/json/add/exception.rb +25 -7
  20. data/lib/json/add/ostruct.rb +32 -9
  21. data/lib/json/add/range.rb +33 -8
  22. data/lib/json/add/rational.rb +30 -8
  23. data/lib/json/add/regexp.rb +28 -10
  24. data/lib/json/add/set.rb +48 -0
  25. data/lib/json/add/struct.rb +29 -7
  26. data/lib/json/add/symbol.rb +28 -5
  27. data/lib/json/add/time.rb +27 -6
  28. data/lib/json/common.rb +402 -188
  29. data/lib/json/ext.rb +0 -6
  30. data/lib/json/generic_object.rb +11 -6
  31. data/lib/json/pure/generator.rb +120 -137
  32. data/lib/json/pure/parser.rb +64 -86
  33. data/lib/json/pure.rb +2 -8
  34. data/lib/json/version.rb +2 -1
  35. data/lib/json.rb +559 -29
  36. metadata +18 -129
  37. data/.gitignore +0 -17
  38. data/.travis.yml +0 -18
  39. data/Gemfile +0 -7
  40. data/README-json-jruby.markdown +0 -33
  41. data/Rakefile +0 -402
  42. data/TODO +0 -1
  43. data/VERSION +0 -1
  44. data/data/example.json +0 -1
  45. data/data/index.html +0 -38
  46. data/data/prototype.js +0 -4184
  47. data/diagrams/.keep +0 -0
  48. data/install.rb +0 -23
  49. data/java/src/json/ext/ByteListTranscoder.java +0 -166
  50. data/java/src/json/ext/Generator.java +0 -446
  51. data/java/src/json/ext/GeneratorMethods.java +0 -231
  52. data/java/src/json/ext/GeneratorService.java +0 -42
  53. data/java/src/json/ext/GeneratorState.java +0 -542
  54. data/java/src/json/ext/OptionsReader.java +0 -113
  55. data/java/src/json/ext/Parser.java +0 -2644
  56. data/java/src/json/ext/Parser.rl +0 -968
  57. data/java/src/json/ext/ParserService.java +0 -34
  58. data/java/src/json/ext/RuntimeInfo.java +0 -120
  59. data/java/src/json/ext/StringDecoder.java +0 -166
  60. data/java/src/json/ext/StringEncoder.java +0 -111
  61. data/java/src/json/ext/Utils.java +0 -88
  62. data/json-java.gemspec +0 -38
  63. data/json_pure.gemspec +0 -37
  64. data/lib/json/ext/.keep +0 -0
  65. data/tests/fixtures/fail1.json +0 -1
  66. data/tests/fixtures/fail10.json +0 -1
  67. data/tests/fixtures/fail11.json +0 -1
  68. data/tests/fixtures/fail12.json +0 -1
  69. data/tests/fixtures/fail13.json +0 -1
  70. data/tests/fixtures/fail14.json +0 -1
  71. data/tests/fixtures/fail18.json +0 -1
  72. data/tests/fixtures/fail19.json +0 -1
  73. data/tests/fixtures/fail2.json +0 -1
  74. data/tests/fixtures/fail20.json +0 -1
  75. data/tests/fixtures/fail21.json +0 -1
  76. data/tests/fixtures/fail22.json +0 -1
  77. data/tests/fixtures/fail23.json +0 -1
  78. data/tests/fixtures/fail24.json +0 -1
  79. data/tests/fixtures/fail25.json +0 -1
  80. data/tests/fixtures/fail27.json +0 -2
  81. data/tests/fixtures/fail28.json +0 -2
  82. data/tests/fixtures/fail3.json +0 -1
  83. data/tests/fixtures/fail4.json +0 -1
  84. data/tests/fixtures/fail5.json +0 -1
  85. data/tests/fixtures/fail6.json +0 -1
  86. data/tests/fixtures/fail7.json +0 -1
  87. data/tests/fixtures/fail8.json +0 -1
  88. data/tests/fixtures/fail9.json +0 -1
  89. data/tests/fixtures/pass1.json +0 -56
  90. data/tests/fixtures/pass15.json +0 -1
  91. data/tests/fixtures/pass16.json +0 -1
  92. data/tests/fixtures/pass17.json +0 -1
  93. data/tests/fixtures/pass2.json +0 -1
  94. data/tests/fixtures/pass26.json +0 -1
  95. data/tests/fixtures/pass3.json +0 -6
  96. data/tests/setup_variant.rb +0 -11
  97. data/tests/test_json.rb +0 -519
  98. data/tests/test_json_addition.rb +0 -196
  99. data/tests/test_json_encoding.rb +0 -65
  100. data/tests/test_json_fixtures.rb +0 -35
  101. data/tests/test_json_generate.rb +0 -348
  102. data/tests/test_json_generic_object.rb +0 -75
  103. data/tests/test_json_string_matching.rb +0 -39
  104. data/tests/test_json_unicode.rb +0 -72
  105. data/tools/diff.sh +0 -18
  106. data/tools/fuzz.rb +0 -139
  107. data/tools/server.rb +0 -62
@@ -1,542 +0,0 @@
1
- /*
2
- * This code is copyrighted work by Daniel Luz <dev at mernen dot com>.
3
- *
4
- * Distributed under the Ruby license: https://www.ruby-lang.org/en/about/license.txt
5
- */
6
- package json.ext;
7
-
8
- import org.jruby.Ruby;
9
- import org.jruby.RubyBoolean;
10
- import org.jruby.RubyClass;
11
- import org.jruby.RubyHash;
12
- import org.jruby.RubyInteger;
13
- import org.jruby.RubyNumeric;
14
- import org.jruby.RubyObject;
15
- import org.jruby.RubyString;
16
- import org.jruby.anno.JRubyMethod;
17
- import org.jruby.runtime.Block;
18
- import org.jruby.runtime.ObjectAllocator;
19
- import org.jruby.runtime.ThreadContext;
20
- import org.jruby.runtime.Visibility;
21
- import org.jruby.runtime.builtin.IRubyObject;
22
- import org.jruby.util.ByteList;
23
-
24
- /**
25
- * The <code>JSON::Ext::Generator::State</code> class.
26
- *
27
- * <p>This class is used to create State instances, that are use to hold data
28
- * while generating a JSON text from a a Ruby data structure.
29
- *
30
- * @author mernen
31
- */
32
- public class GeneratorState extends RubyObject {
33
- /**
34
- * The indenting unit string. Will be repeated several times for larger
35
- * indenting levels.
36
- */
37
- private ByteList indent = ByteList.EMPTY_BYTELIST;
38
- /**
39
- * The spacing to be added after a semicolon on a JSON object.
40
- * @see #spaceBefore
41
- */
42
- private ByteList space = ByteList.EMPTY_BYTELIST;
43
- /**
44
- * The spacing to be added before a semicolon on a JSON object.
45
- * @see #space
46
- */
47
- private ByteList spaceBefore = ByteList.EMPTY_BYTELIST;
48
- /**
49
- * Any suffix to be added after the comma for each element on a JSON object.
50
- * It is assumed to be a newline, if set.
51
- */
52
- private ByteList objectNl = ByteList.EMPTY_BYTELIST;
53
- /**
54
- * Any suffix to be added after the comma for each element on a JSON Array.
55
- * It is assumed to be a newline, if set.
56
- */
57
- private ByteList arrayNl = ByteList.EMPTY_BYTELIST;
58
-
59
- /**
60
- * The maximum level of nesting of structures allowed.
61
- * <code>0</code> means disabled.
62
- */
63
- private int maxNesting = DEFAULT_MAX_NESTING;
64
- static final int DEFAULT_MAX_NESTING = 100;
65
- /**
66
- * Whether special float values (<code>NaN</code>, <code>Infinity</code>,
67
- * <code>-Infinity</code>) are accepted.
68
- * If set to <code>false</code>, an exception will be thrown upon
69
- * encountering one.
70
- */
71
- private boolean allowNaN = DEFAULT_ALLOW_NAN;
72
- static final boolean DEFAULT_ALLOW_NAN = false;
73
- /**
74
- * If set to <code>true</code> all JSON documents generated do not contain
75
- * any other characters than ASCII characters.
76
- */
77
- private boolean asciiOnly = DEFAULT_ASCII_ONLY;
78
- static final boolean DEFAULT_ASCII_ONLY = false;
79
- /**
80
- * If set to <code>true</code> all JSON values generated might not be
81
- * RFC-conform JSON documents.
82
- */
83
- private boolean quirksMode = DEFAULT_QUIRKS_MODE;
84
- static final boolean DEFAULT_QUIRKS_MODE = false;
85
- /**
86
- * The initial buffer length of this state. (This isn't really used on all
87
- * non-C implementations.)
88
- */
89
- private int bufferInitialLength = DEFAULT_BUFFER_INITIAL_LENGTH;
90
- static final int DEFAULT_BUFFER_INITIAL_LENGTH = 1024;
91
-
92
- /**
93
- * The current depth (inside a #to_json call)
94
- */
95
- private int depth = 0;
96
-
97
- static final ObjectAllocator ALLOCATOR = new ObjectAllocator() {
98
- public IRubyObject allocate(Ruby runtime, RubyClass klazz) {
99
- return new GeneratorState(runtime, klazz);
100
- }
101
- };
102
-
103
- public GeneratorState(Ruby runtime, RubyClass metaClass) {
104
- super(runtime, metaClass);
105
- }
106
-
107
- /**
108
- * <code>State.from_state(opts)</code>
109
- *
110
- * <p>Creates a State object from <code>opts</code>, which ought to be
111
- * {@link RubyHash Hash} to create a new <code>State</code> instance
112
- * configured by <codes>opts</code>, something else to create an
113
- * unconfigured instance. If <code>opts</code> is a <code>State</code>
114
- * object, it is just returned.
115
- * @param clazzParam The receiver of the method call
116
- * ({@link RubyClass} <code>State</code>)
117
- * @param opts The object to use as a base for the new <code>State</code>
118
- * @param block The block passed to the method
119
- * @return A <code>GeneratorState</code> as determined above
120
- */
121
- @JRubyMethod(meta=true)
122
- public static IRubyObject from_state(ThreadContext context,
123
- IRubyObject klass, IRubyObject opts) {
124
- return fromState(context, opts);
125
- }
126
-
127
- static GeneratorState fromState(ThreadContext context, IRubyObject opts) {
128
- return fromState(context, RuntimeInfo.forRuntime(context.getRuntime()), opts);
129
- }
130
-
131
- static GeneratorState fromState(ThreadContext context, RuntimeInfo info,
132
- IRubyObject opts) {
133
- RubyClass klass = info.generatorStateClass.get();
134
- if (opts != null) {
135
- // if the given parameter is a Generator::State, return itself
136
- if (klass.isInstance(opts)) return (GeneratorState)opts;
137
-
138
- // if the given parameter is a Hash, pass it to the instantiator
139
- if (context.getRuntime().getHash().isInstance(opts)) {
140
- return (GeneratorState)klass.newInstance(context,
141
- new IRubyObject[] {opts}, Block.NULL_BLOCK);
142
- }
143
- }
144
-
145
- // for other values, return the safe prototype
146
- return (GeneratorState)info.getSafeStatePrototype(context).dup();
147
- }
148
-
149
- /**
150
- * <code>State#initialize(opts = {})</code>
151
- *
152
- * Instantiates a new <code>State</code> object, configured by <code>opts</code>.
153
- *
154
- * <code>opts</code> can have the following keys:
155
- *
156
- * <dl>
157
- * <dt><code>:indent</code>
158
- * <dd>a {@link RubyString String} used to indent levels (default: <code>""</code>)
159
- * <dt><code>:space</code>
160
- * <dd>a String that is put after a <code>':'</code> or <code>','</code>
161
- * delimiter (default: <code>""</code>)
162
- * <dt><code>:space_before</code>
163
- * <dd>a String that is put before a <code>":"</code> pair delimiter
164
- * (default: <code>""</code>)
165
- * <dt><code>:object_nl</code>
166
- * <dd>a String that is put at the end of a JSON object (default: <code>""</code>)
167
- * <dt><code>:array_nl</code>
168
- * <dd>a String that is put at the end of a JSON array (default: <code>""</code>)
169
- * <dt><code>:allow_nan</code>
170
- * <dd><code>true</code> if <code>NaN</code>, <code>Infinity</code>, and
171
- * <code>-Infinity</code> should be generated, otherwise an exception is
172
- * thrown if these values are encountered.
173
- * This options defaults to <code>false</code>.
174
- */
175
- @JRubyMethod(optional=1, visibility=Visibility.PRIVATE)
176
- public IRubyObject initialize(ThreadContext context, IRubyObject[] args) {
177
- configure(context, args.length > 0 ? args[0] : null);
178
- return this;
179
- }
180
-
181
- @JRubyMethod
182
- public IRubyObject initialize_copy(ThreadContext context, IRubyObject vOrig) {
183
- Ruby runtime = context.getRuntime();
184
- if (!(vOrig instanceof GeneratorState)) {
185
- throw runtime.newTypeError(vOrig, getType());
186
- }
187
- GeneratorState orig = (GeneratorState)vOrig;
188
- this.indent = orig.indent;
189
- this.space = orig.space;
190
- this.spaceBefore = orig.spaceBefore;
191
- this.objectNl = orig.objectNl;
192
- this.arrayNl = orig.arrayNl;
193
- this.maxNesting = orig.maxNesting;
194
- this.allowNaN = orig.allowNaN;
195
- this.asciiOnly = orig.asciiOnly;
196
- this.quirksMode = orig.quirksMode;
197
- this.bufferInitialLength = orig.bufferInitialLength;
198
- this.depth = orig.depth;
199
- return this;
200
- }
201
-
202
- /**
203
- * Generates a valid JSON document from object <code>obj</code> and returns
204
- * the result. If no valid JSON document can be created this method raises
205
- * a GeneratorError exception.
206
- */
207
- @JRubyMethod
208
- public IRubyObject generate(ThreadContext context, IRubyObject obj) {
209
- RubyString result = Generator.generateJson(context, obj, this);
210
- if (!quirksMode && !objectOrArrayLiteral(result)) {
211
- throw Utils.newException(context, Utils.M_GENERATOR_ERROR,
212
- "only generation of JSON objects or arrays allowed");
213
- }
214
- RuntimeInfo info = RuntimeInfo.forRuntime(context.getRuntime());
215
- if (info.encodingsSupported()) {
216
- result.force_encoding(context, info.utf8.get());
217
- }
218
- return result;
219
- }
220
-
221
- /**
222
- * Ensures the given string is in the form "[...]" or "{...}", being
223
- * possibly surrounded by white space.
224
- * The string's encoding must be ASCII-compatible.
225
- * @param value
226
- * @return
227
- */
228
- private static boolean objectOrArrayLiteral(RubyString value) {
229
- ByteList bl = value.getByteList();
230
- int len = bl.length();
231
-
232
- for (int pos = 0; pos < len - 1; pos++) {
233
- int b = bl.get(pos);
234
- if (Character.isWhitespace(b)) continue;
235
-
236
- // match the opening brace
237
- switch (b) {
238
- case '[':
239
- return matchClosingBrace(bl, pos, len, ']');
240
- case '{':
241
- return matchClosingBrace(bl, pos, len, '}');
242
- default:
243
- return false;
244
- }
245
- }
246
- return false;
247
- }
248
-
249
- private static boolean matchClosingBrace(ByteList bl, int pos, int len,
250
- int brace) {
251
- for (int endPos = len - 1; endPos > pos; endPos--) {
252
- int b = bl.get(endPos);
253
- if (Character.isWhitespace(b)) continue;
254
- return b == brace;
255
- }
256
- return false;
257
- }
258
-
259
- @JRubyMethod(name="[]", required=1)
260
- public IRubyObject op_aref(ThreadContext context, IRubyObject vName) {
261
- String name = vName.asJavaString();
262
- if (getMetaClass().isMethodBound(name, true)) {
263
- return send(context, vName, Block.NULL_BLOCK);
264
- } else {
265
- IRubyObject value = getInstanceVariables().getInstanceVariable("@" + name);
266
- return value == null ? context.nil : value;
267
- }
268
- }
269
-
270
- @JRubyMethod(name="[]=", required=2)
271
- public IRubyObject op_aset(ThreadContext context, IRubyObject vName, IRubyObject value) {
272
- String name = vName.asJavaString();
273
- String nameWriter = name + "=";
274
- if (getMetaClass().isMethodBound(nameWriter, true)) {
275
- return send(context, context.getRuntime().newString(nameWriter), value, Block.NULL_BLOCK);
276
- } else {
277
- getInstanceVariables().setInstanceVariable("@" + name, value);
278
- }
279
- return context.getRuntime().getNil();
280
- }
281
-
282
- public ByteList getIndent() {
283
- return indent;
284
- }
285
-
286
- @JRubyMethod(name="indent")
287
- public RubyString indent_get(ThreadContext context) {
288
- return context.getRuntime().newString(indent);
289
- }
290
-
291
- @JRubyMethod(name="indent=")
292
- public IRubyObject indent_set(ThreadContext context, IRubyObject indent) {
293
- this.indent = prepareByteList(context, indent);
294
- return indent;
295
- }
296
-
297
- public ByteList getSpace() {
298
- return space;
299
- }
300
-
301
- @JRubyMethod(name="space")
302
- public RubyString space_get(ThreadContext context) {
303
- return context.getRuntime().newString(space);
304
- }
305
-
306
- @JRubyMethod(name="space=")
307
- public IRubyObject space_set(ThreadContext context, IRubyObject space) {
308
- this.space = prepareByteList(context, space);
309
- return space;
310
- }
311
-
312
- public ByteList getSpaceBefore() {
313
- return spaceBefore;
314
- }
315
-
316
- @JRubyMethod(name="space_before")
317
- public RubyString space_before_get(ThreadContext context) {
318
- return context.getRuntime().newString(spaceBefore);
319
- }
320
-
321
- @JRubyMethod(name="space_before=")
322
- public IRubyObject space_before_set(ThreadContext context,
323
- IRubyObject spaceBefore) {
324
- this.spaceBefore = prepareByteList(context, spaceBefore);
325
- return spaceBefore;
326
- }
327
-
328
- public ByteList getObjectNl() {
329
- return objectNl;
330
- }
331
-
332
- @JRubyMethod(name="object_nl")
333
- public RubyString object_nl_get(ThreadContext context) {
334
- return context.getRuntime().newString(objectNl);
335
- }
336
-
337
- @JRubyMethod(name="object_nl=")
338
- public IRubyObject object_nl_set(ThreadContext context,
339
- IRubyObject objectNl) {
340
- this.objectNl = prepareByteList(context, objectNl);
341
- return objectNl;
342
- }
343
-
344
- public ByteList getArrayNl() {
345
- return arrayNl;
346
- }
347
-
348
- @JRubyMethod(name="array_nl")
349
- public RubyString array_nl_get(ThreadContext context) {
350
- return context.getRuntime().newString(arrayNl);
351
- }
352
-
353
- @JRubyMethod(name="array_nl=")
354
- public IRubyObject array_nl_set(ThreadContext context,
355
- IRubyObject arrayNl) {
356
- this.arrayNl = prepareByteList(context, arrayNl);
357
- return arrayNl;
358
- }
359
-
360
- @JRubyMethod(name="check_circular?")
361
- public RubyBoolean check_circular_p(ThreadContext context) {
362
- return context.getRuntime().newBoolean(maxNesting != 0);
363
- }
364
-
365
- /**
366
- * Returns the maximum level of nesting configured for this state.
367
- */
368
- public int getMaxNesting() {
369
- return maxNesting;
370
- }
371
-
372
- @JRubyMethod(name="max_nesting")
373
- public RubyInteger max_nesting_get(ThreadContext context) {
374
- return context.getRuntime().newFixnum(maxNesting);
375
- }
376
-
377
- @JRubyMethod(name="max_nesting=")
378
- public IRubyObject max_nesting_set(IRubyObject max_nesting) {
379
- maxNesting = RubyNumeric.fix2int(max_nesting);
380
- return max_nesting;
381
- }
382
-
383
- public boolean allowNaN() {
384
- return allowNaN;
385
- }
386
-
387
- @JRubyMethod(name="allow_nan?")
388
- public RubyBoolean allow_nan_p(ThreadContext context) {
389
- return context.getRuntime().newBoolean(allowNaN);
390
- }
391
-
392
- public boolean asciiOnly() {
393
- return asciiOnly;
394
- }
395
-
396
- @JRubyMethod(name="ascii_only?")
397
- public RubyBoolean ascii_only_p(ThreadContext context) {
398
- return context.getRuntime().newBoolean(asciiOnly);
399
- }
400
-
401
- @JRubyMethod(name="quirks_mode")
402
- public RubyBoolean quirks_mode_get(ThreadContext context) {
403
- return context.getRuntime().newBoolean(quirksMode);
404
- }
405
-
406
- @JRubyMethod(name="quirks_mode=")
407
- public IRubyObject quirks_mode_set(IRubyObject quirks_mode) {
408
- quirksMode = quirks_mode.isTrue();
409
- return quirks_mode.getRuntime().newBoolean(quirksMode);
410
- }
411
-
412
- @JRubyMethod(name="buffer_initial_length")
413
- public RubyInteger buffer_initial_length_get(ThreadContext context) {
414
- return context.getRuntime().newFixnum(bufferInitialLength);
415
- }
416
-
417
- @JRubyMethod(name="buffer_initial_length=")
418
- public IRubyObject buffer_initial_length_set(IRubyObject buffer_initial_length) {
419
- int newLength = RubyNumeric.fix2int(buffer_initial_length);
420
- if (newLength > 0) bufferInitialLength = newLength;
421
- return buffer_initial_length;
422
- }
423
-
424
- @JRubyMethod(name="quirks_mode?")
425
- public RubyBoolean quirks_mode_p(ThreadContext context) {
426
- return context.getRuntime().newBoolean(quirksMode);
427
- }
428
-
429
- public int getDepth() {
430
- return depth;
431
- }
432
-
433
- @JRubyMethod(name="depth")
434
- public RubyInteger depth_get(ThreadContext context) {
435
- return context.getRuntime().newFixnum(depth);
436
- }
437
-
438
- @JRubyMethod(name="depth=")
439
- public IRubyObject depth_set(IRubyObject vDepth) {
440
- depth = RubyNumeric.fix2int(vDepth);
441
- return vDepth;
442
- }
443
-
444
- private ByteList prepareByteList(ThreadContext context, IRubyObject value) {
445
- RubyString str = value.convertToString();
446
- RuntimeInfo info = RuntimeInfo.forRuntime(context.getRuntime());
447
- if (info.encodingsSupported() && str.encoding(context) != info.utf8.get()) {
448
- str = (RubyString)str.encode(context, info.utf8.get());
449
- }
450
- return str.getByteList().dup();
451
- }
452
-
453
- /**
454
- * <code>State#configure(opts)</code>
455
- *
456
- * <p>Configures this State instance with the {@link RubyHash Hash}
457
- * <code>opts</code>, and returns itself.
458
- * @param vOpts The options hash
459
- * @return The receiver
460
- */
461
- @JRubyMethod(alias = "merge")
462
- public IRubyObject configure(ThreadContext context, IRubyObject vOpts) {
463
- OptionsReader opts = new OptionsReader(context, vOpts);
464
-
465
- ByteList indent = opts.getString("indent");
466
- if (indent != null) this.indent = indent;
467
-
468
- ByteList space = opts.getString("space");
469
- if (space != null) this.space = space;
470
-
471
- ByteList spaceBefore = opts.getString("space_before");
472
- if (spaceBefore != null) this.spaceBefore = spaceBefore;
473
-
474
- ByteList arrayNl = opts.getString("array_nl");
475
- if (arrayNl != null) this.arrayNl = arrayNl;
476
-
477
- ByteList objectNl = opts.getString("object_nl");
478
- if (objectNl != null) this.objectNl = objectNl;
479
-
480
- maxNesting = opts.getInt("max_nesting", DEFAULT_MAX_NESTING);
481
- allowNaN = opts.getBool("allow_nan", DEFAULT_ALLOW_NAN);
482
- asciiOnly = opts.getBool("ascii_only", DEFAULT_ASCII_ONLY);
483
- quirksMode = opts.getBool("quirks_mode", DEFAULT_QUIRKS_MODE);
484
- bufferInitialLength = opts.getInt("buffer_initial_length", DEFAULT_BUFFER_INITIAL_LENGTH);
485
-
486
- depth = opts.getInt("depth", 0);
487
-
488
- return this;
489
- }
490
-
491
- /**
492
- * <code>State#to_h()</code>
493
- *
494
- * <p>Returns the configuration instance variables as a hash, that can be
495
- * passed to the configure method.
496
- * @return the hash
497
- */
498
- @JRubyMethod(alias = "to_hash")
499
- public RubyHash to_h(ThreadContext context) {
500
- Ruby runtime = context.getRuntime();
501
- RubyHash result = RubyHash.newHash(runtime);
502
-
503
- result.op_aset(context, runtime.newSymbol("indent"), indent_get(context));
504
- result.op_aset(context, runtime.newSymbol("space"), space_get(context));
505
- result.op_aset(context, runtime.newSymbol("space_before"), space_before_get(context));
506
- result.op_aset(context, runtime.newSymbol("object_nl"), object_nl_get(context));
507
- result.op_aset(context, runtime.newSymbol("array_nl"), array_nl_get(context));
508
- result.op_aset(context, runtime.newSymbol("allow_nan"), allow_nan_p(context));
509
- result.op_aset(context, runtime.newSymbol("ascii_only"), ascii_only_p(context));
510
- result.op_aset(context, runtime.newSymbol("quirks_mode"), quirks_mode_p(context));
511
- result.op_aset(context, runtime.newSymbol("max_nesting"), max_nesting_get(context));
512
- result.op_aset(context, runtime.newSymbol("depth"), depth_get(context));
513
- result.op_aset(context, runtime.newSymbol("buffer_initial_length"), buffer_initial_length_get(context));
514
- for (String name: getInstanceVariableNameList()) {
515
- result.op_aset(context, runtime.newSymbol(name.substring(1)), getInstanceVariables().getInstanceVariable(name));
516
- }
517
- return result;
518
- }
519
-
520
- public int increaseDepth() {
521
- depth++;
522
- checkMaxNesting();
523
- return depth;
524
- }
525
-
526
- public int decreaseDepth() {
527
- return --depth;
528
- }
529
-
530
- /**
531
- * Checks if the current depth is allowed as per this state's options.
532
- * @param context
533
- * @param depth The corrent depth
534
- */
535
- private void checkMaxNesting() {
536
- if (maxNesting != 0 && depth > maxNesting) {
537
- depth--;
538
- throw Utils.newException(getRuntime().getCurrentContext(),
539
- Utils.M_NESTING_ERROR, "nesting of " + depth + " is too deep");
540
- }
541
- }
542
- }
@@ -1,113 +0,0 @@
1
- /*
2
- * This code is copyrighted work by Daniel Luz <dev at mernen dot com>.
3
- *
4
- * Distributed under the Ruby license: https://www.ruby-lang.org/en/about/license.txt
5
- */
6
- package json.ext;
7
-
8
- import org.jruby.Ruby;
9
- import org.jruby.RubyClass;
10
- import org.jruby.RubyHash;
11
- import org.jruby.RubyNumeric;
12
- import org.jruby.RubyString;
13
- import org.jruby.runtime.ThreadContext;
14
- import org.jruby.runtime.builtin.IRubyObject;
15
- import org.jruby.util.ByteList;
16
-
17
- final class OptionsReader {
18
- private final ThreadContext context;
19
- private final Ruby runtime;
20
- private final RubyHash opts;
21
- private RuntimeInfo info;
22
-
23
- OptionsReader(ThreadContext context, IRubyObject vOpts) {
24
- this.context = context;
25
- this.runtime = context.getRuntime();
26
- if (vOpts == null || vOpts.isNil()) {
27
- opts = null;
28
- } else if (vOpts.respondsTo("to_hash")) {
29
- opts = vOpts.convertToHash();
30
- } else if (vOpts.respondsTo("to_h")) {
31
- opts = vOpts.callMethod(context, "to_h").convertToHash();
32
- } else {
33
- opts = vOpts.convertToHash(); /* Should just raise the correct TypeError */
34
- }
35
- }
36
-
37
- private RuntimeInfo getRuntimeInfo() {
38
- if (info != null) return info;
39
- info = RuntimeInfo.forRuntime(runtime);
40
- return info;
41
- }
42
-
43
- /**
44
- * Efficiently looks up items with a {@link RubySymbol Symbol} key
45
- * @param key The Symbol name to look up for
46
- * @return The item in the {@link RubyHash Hash}, or <code>null</code>
47
- * if not found
48
- */
49
- IRubyObject get(String key) {
50
- return opts == null ? null : opts.fastARef(runtime.newSymbol(key));
51
- }
52
-
53
- boolean getBool(String key, boolean defaultValue) {
54
- IRubyObject value = get(key);
55
- return value == null ? defaultValue : value.isTrue();
56
- }
57
-
58
- int getInt(String key, int defaultValue) {
59
- IRubyObject value = get(key);
60
- if (value == null) return defaultValue;
61
- if (!value.isTrue()) return 0;
62
- return RubyNumeric.fix2int(value);
63
- }
64
-
65
- /**
66
- * Reads the setting from the options hash. If no entry is set for this
67
- * key or if it evaluates to <code>false</code>, returns null; attempts to
68
- * coerce the value to {@link RubyString String} otherwise.
69
- * @param key The Symbol name to look up for
70
- * @return <code>null</code> if the key is not in the Hash or if
71
- * its value evaluates to <code>false</code>
72
- * @throws RaiseException <code>TypeError</code> if the value does not
73
- * evaluate to <code>false</code> and can't be
74
- * converted to string
75
- */
76
- ByteList getString(String key) {
77
- RubyString str = getString(key, null);
78
- return str == null ? null : str.getByteList().dup();
79
- }
80
-
81
- RubyString getString(String key, RubyString defaultValue) {
82
- IRubyObject value = get(key);
83
- if (value == null || !value.isTrue()) return defaultValue;
84
-
85
- RubyString str = value.convertToString();
86
- RuntimeInfo info = getRuntimeInfo();
87
- if (info.encodingsSupported() && str.encoding(context) != info.utf8.get()) {
88
- str = (RubyString)str.encode(context, info.utf8.get());
89
- }
90
- return str;
91
- }
92
-
93
- /**
94
- * Reads the setting from the options hash. If it is <code>nil</code> or
95
- * undefined, returns the default value given.
96
- * If not, ensures it is a RubyClass instance and shares the same
97
- * allocator as the default value (i.e. for the basic types which have
98
- * their specific allocators, this ensures the passed value is
99
- * a subclass of them).
100
- */
101
- RubyClass getClass(String key, RubyClass defaultValue) {
102
- IRubyObject value = get(key);
103
-
104
- if (value == null || value.isNil()) return defaultValue;
105
- return (RubyClass)value;
106
- }
107
-
108
- public RubyHash getHash(String key) {
109
- IRubyObject value = get(key);
110
- if (value == null || value.isNil()) return new RubyHash(runtime);
111
- return (RubyHash) value;
112
- }
113
- }