oj 3.13.23 → 3.16.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (178) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +86 -0
  3. data/README.md +2 -2
  4. data/ext/oj/buf.h +7 -6
  5. data/ext/oj/cache.c +29 -26
  6. data/ext/oj/cache.h +3 -2
  7. data/ext/oj/cache8.c +10 -9
  8. data/ext/oj/circarray.c +7 -5
  9. data/ext/oj/circarray.h +2 -2
  10. data/ext/oj/code.c +5 -12
  11. data/ext/oj/code.h +2 -2
  12. data/ext/oj/compat.c +20 -60
  13. data/ext/oj/custom.c +26 -59
  14. data/ext/oj/debug.c +3 -9
  15. data/ext/oj/dump.c +103 -53
  16. data/ext/oj/dump.h +1 -4
  17. data/ext/oj/dump_compat.c +557 -592
  18. data/ext/oj/dump_leaf.c +3 -5
  19. data/ext/oj/dump_object.c +42 -48
  20. data/ext/oj/dump_strict.c +10 -22
  21. data/ext/oj/encoder.c +1 -1
  22. data/ext/oj/err.c +2 -13
  23. data/ext/oj/err.h +9 -12
  24. data/ext/oj/extconf.rb +16 -7
  25. data/ext/oj/fast.c +63 -98
  26. data/ext/oj/intern.c +62 -47
  27. data/ext/oj/intern.h +3 -7
  28. data/ext/oj/mem.c +318 -0
  29. data/ext/oj/mem.h +53 -0
  30. data/ext/oj/mimic_json.c +54 -38
  31. data/ext/oj/object.c +33 -43
  32. data/ext/oj/odd.c +8 -6
  33. data/ext/oj/odd.h +4 -4
  34. data/ext/oj/oj.c +245 -216
  35. data/ext/oj/oj.h +83 -81
  36. data/ext/oj/parse.c +109 -153
  37. data/ext/oj/parse.h +21 -24
  38. data/ext/oj/parser.c +80 -67
  39. data/ext/oj/parser.h +9 -8
  40. data/ext/oj/rails.c +71 -94
  41. data/ext/oj/reader.c +9 -14
  42. data/ext/oj/reader.h +4 -2
  43. data/ext/oj/resolve.c +3 -4
  44. data/ext/oj/rxclass.c +6 -5
  45. data/ext/oj/rxclass.h +1 -1
  46. data/ext/oj/saj.c +13 -15
  47. data/ext/oj/saj2.c +37 -49
  48. data/ext/oj/saj2.h +1 -1
  49. data/ext/oj/scp.c +6 -20
  50. data/ext/oj/sparse.c +22 -70
  51. data/ext/oj/stream_writer.c +46 -48
  52. data/ext/oj/strict.c +22 -56
  53. data/ext/oj/string_writer.c +64 -40
  54. data/ext/oj/trace.h +31 -4
  55. data/ext/oj/usual.c +125 -114
  56. data/ext/oj/usual.h +7 -6
  57. data/ext/oj/util.h +1 -1
  58. data/ext/oj/val_stack.c +13 -2
  59. data/ext/oj/val_stack.h +8 -7
  60. data/ext/oj/wab.c +25 -57
  61. data/lib/oj/active_support_helper.rb +1 -3
  62. data/lib/oj/bag.rb +7 -1
  63. data/lib/oj/easy_hash.rb +4 -5
  64. data/lib/oj/error.rb +0 -1
  65. data/lib/oj/json.rb +162 -150
  66. data/lib/oj/mimic.rb +7 -7
  67. data/lib/oj/schandler.rb +5 -4
  68. data/lib/oj/state.rb +8 -5
  69. data/lib/oj/version.rb +1 -2
  70. data/lib/oj.rb +2 -0
  71. data/pages/InstallOptions.md +20 -0
  72. data/pages/Options.md +4 -0
  73. metadata +46 -121
  74. data/test/_test_active.rb +0 -76
  75. data/test/_test_active_mimic.rb +0 -96
  76. data/test/_test_mimic_rails.rb +0 -126
  77. data/test/activerecord/result_test.rb +0 -32
  78. data/test/activesupport4/decoding_test.rb +0 -108
  79. data/test/activesupport4/encoding_test.rb +0 -531
  80. data/test/activesupport4/test_helper.rb +0 -41
  81. data/test/activesupport5/abstract_unit.rb +0 -45
  82. data/test/activesupport5/decoding_test.rb +0 -133
  83. data/test/activesupport5/encoding_test.rb +0 -500
  84. data/test/activesupport5/encoding_test_cases.rb +0 -98
  85. data/test/activesupport5/test_helper.rb +0 -72
  86. data/test/activesupport5/time_zone_test_helpers.rb +0 -39
  87. data/test/activesupport6/abstract_unit.rb +0 -44
  88. data/test/activesupport6/decoding_test.rb +0 -133
  89. data/test/activesupport6/encoding_test.rb +0 -507
  90. data/test/activesupport6/encoding_test_cases.rb +0 -98
  91. data/test/activesupport6/test_common.rb +0 -17
  92. data/test/activesupport6/test_helper.rb +0 -163
  93. data/test/activesupport6/time_zone_test_helpers.rb +0 -39
  94. data/test/activesupport7/abstract_unit.rb +0 -49
  95. data/test/activesupport7/decoding_test.rb +0 -125
  96. data/test/activesupport7/encoding_test.rb +0 -486
  97. data/test/activesupport7/encoding_test_cases.rb +0 -104
  98. data/test/activesupport7/time_zone_test_helpers.rb +0 -47
  99. data/test/bar.rb +0 -11
  100. data/test/baz.rb +0 -16
  101. data/test/bug.rb +0 -16
  102. data/test/files.rb +0 -29
  103. data/test/foo.rb +0 -77
  104. data/test/helper.rb +0 -42
  105. data/test/isolated/shared.rb +0 -308
  106. data/test/isolated/test_mimic_after.rb +0 -13
  107. data/test/isolated/test_mimic_alone.rb +0 -12
  108. data/test/isolated/test_mimic_as_json.rb +0 -45
  109. data/test/isolated/test_mimic_before.rb +0 -13
  110. data/test/isolated/test_mimic_define.rb +0 -28
  111. data/test/isolated/test_mimic_rails_after.rb +0 -22
  112. data/test/isolated/test_mimic_rails_before.rb +0 -21
  113. data/test/isolated/test_mimic_redefine.rb +0 -15
  114. data/test/json_gem/json_addition_test.rb +0 -216
  115. data/test/json_gem/json_common_interface_test.rb +0 -153
  116. data/test/json_gem/json_encoding_test.rb +0 -107
  117. data/test/json_gem/json_ext_parser_test.rb +0 -20
  118. data/test/json_gem/json_fixtures_test.rb +0 -35
  119. data/test/json_gem/json_generator_test.rb +0 -396
  120. data/test/json_gem/json_generic_object_test.rb +0 -90
  121. data/test/json_gem/json_parser_test.rb +0 -477
  122. data/test/json_gem/json_string_matching_test.rb +0 -42
  123. data/test/json_gem/test_helper.rb +0 -30
  124. data/test/mem.rb +0 -33
  125. data/test/perf.rb +0 -107
  126. data/test/perf_compat.rb +0 -130
  127. data/test/perf_dump.rb +0 -50
  128. data/test/perf_fast.rb +0 -164
  129. data/test/perf_file.rb +0 -64
  130. data/test/perf_object.rb +0 -138
  131. data/test/perf_once.rb +0 -58
  132. data/test/perf_parser.rb +0 -189
  133. data/test/perf_saj.rb +0 -109
  134. data/test/perf_scp.rb +0 -152
  135. data/test/perf_simple.rb +0 -287
  136. data/test/perf_strict.rb +0 -139
  137. data/test/perf_wab.rb +0 -131
  138. data/test/prec.rb +0 -23
  139. data/test/sample/change.rb +0 -14
  140. data/test/sample/dir.rb +0 -19
  141. data/test/sample/doc.rb +0 -36
  142. data/test/sample/file.rb +0 -48
  143. data/test/sample/group.rb +0 -16
  144. data/test/sample/hasprops.rb +0 -16
  145. data/test/sample/layer.rb +0 -12
  146. data/test/sample/line.rb +0 -20
  147. data/test/sample/oval.rb +0 -10
  148. data/test/sample/rect.rb +0 -10
  149. data/test/sample/shape.rb +0 -35
  150. data/test/sample/text.rb +0 -20
  151. data/test/sample.rb +0 -54
  152. data/test/sample_json.rb +0 -37
  153. data/test/test_compat.rb +0 -540
  154. data/test/test_custom.rb +0 -544
  155. data/test/test_debian.rb +0 -53
  156. data/test/test_fast.rb +0 -530
  157. data/test/test_file.rb +0 -255
  158. data/test/test_gc.rb +0 -60
  159. data/test/test_generate.rb +0 -21
  160. data/test/test_hash.rb +0 -39
  161. data/test/test_integer_range.rb +0 -72
  162. data/test/test_null.rb +0 -376
  163. data/test/test_object.rb +0 -1025
  164. data/test/test_parser.rb +0 -11
  165. data/test/test_parser_debug.rb +0 -27
  166. data/test/test_parser_saj.rb +0 -335
  167. data/test/test_parser_usual.rb +0 -217
  168. data/test/test_rails.rb +0 -35
  169. data/test/test_saj.rb +0 -186
  170. data/test/test_scp.rb +0 -431
  171. data/test/test_strict.rb +0 -435
  172. data/test/test_various.rb +0 -752
  173. data/test/test_wab.rb +0 -309
  174. data/test/test_writer.rb +0 -380
  175. data/test/tests.rb +0 -33
  176. data/test/tests_mimic.rb +0 -23
  177. data/test/tests_mimic_addition.rb +0 -16
  178. data/test/zoo.rb +0 -13
data/ext/oj/dump_compat.c CHANGED
@@ -7,26 +7,24 @@
7
7
  #include "trace.h"
8
8
 
9
9
  // Workaround in case INFINITY is not defined in math.h or if the OS is CentOS
10
- #define OJ_INFINITY (1.0/0.0)
10
+ #define OJ_INFINITY (1.0 / 0.0)
11
11
 
12
- bool oj_use_hash_alt = false;
13
- bool oj_use_array_alt = false;
12
+ bool oj_use_hash_alt = false;
13
+ bool oj_use_array_alt = false;
14
14
 
15
- static bool use_struct_alt = false;
16
- static bool use_exception_alt = false;
17
- static bool use_bignum_alt = false;
15
+ static bool use_struct_alt = false;
16
+ static bool use_exception_alt = false;
17
+ static bool use_bignum_alt = false;
18
18
 
19
- static void
20
- raise_json_err(const char *msg, const char *err_classname) {
19
+ static void raise_json_err(const char *msg, const char *err_classname) {
21
20
  rb_raise(oj_get_json_err_class(err_classname), "%s", msg);
22
21
  }
23
22
 
24
- static void
25
- dump_obj_classname(const char *classname, int depth, Out out) {
26
- int d2 = depth + 1;
27
- size_t len = strlen(classname);
28
- size_t sep_len = out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2;
29
- size_t size = d2 * out->indent + 10 + len + out->opts->create_id_len + sep_len;
23
+ static void dump_obj_classname(const char *classname, int depth, Out out) {
24
+ int d2 = depth + 1;
25
+ size_t len = strlen(classname);
26
+ size_t sep_len = out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2;
27
+ size_t size = d2 * out->indent + 10 + len + out->opts->create_id_len + sep_len;
30
28
 
31
29
  assure_size(out, size);
32
30
  *out->cur++ = '{';
@@ -35,93 +33,88 @@ dump_obj_classname(const char *classname, int depth, Out out) {
35
33
  APPEND_CHARS(out->cur, out->opts->create_id, out->opts->create_id_len);
36
34
  *out->cur++ = '"';
37
35
  if (0 < out->opts->dump_opts.before_size) {
38
- APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
36
+ APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
39
37
  }
40
38
  *out->cur++ = ':';
41
39
  if (0 < out->opts->dump_opts.after_size) {
42
- APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
40
+ APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
43
41
  }
44
42
  *out->cur++ = '"';
45
43
  APPEND_CHARS(out->cur, classname, len);
46
44
  *out->cur++ = '"';
47
45
  }
48
46
 
49
- static void
50
- dump_values_array(VALUE *values, int depth, Out out) {
51
- size_t size;
52
- int d2 = depth + 1;
47
+ static void dump_values_array(VALUE *values, int depth, Out out) {
48
+ size_t size;
49
+ int d2 = depth + 1;
53
50
 
54
51
  assure_size(out, d2 * out->indent + 3);
55
52
  *out->cur++ = '[';
56
53
  if (Qundef == *values) {
57
- *out->cur++ = ']';
54
+ *out->cur++ = ']';
58
55
  } else {
59
- if (out->opts->dump_opts.use) {
60
- size = d2 * out->opts->dump_opts.indent_size + out->opts->dump_opts.array_size + 2;
61
- size += out->opts->dump_opts.array_size;
62
- size += out->opts->dump_opts.indent_size;
63
- } else {
64
- size = d2 * out->indent + 3;
65
- }
66
- for (; Qundef != *values; values++) {
67
- assure_size(out, size);
68
- if (out->opts->dump_opts.use) {
69
- if (0 < out->opts->dump_opts.array_size) {
70
- APPEND_CHARS(out->cur, out->opts->dump_opts.array_nl, out->opts->dump_opts.array_size);
71
- }
72
- if (0 < out->opts->dump_opts.indent_size) {
73
- int i;
74
-
75
- for (i = d2; 0 < i; i--) {
76
- APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
77
- }
78
- }
79
- } else {
80
- fill_indent(out, d2);
81
- }
82
- oj_dump_compat_val(*values, d2, out, true);
83
- if (Qundef != *(values + 1)) {
84
- *out->cur++ = ',';
85
- }
86
- }
87
- assure_size(out, size);
88
- if (out->opts->dump_opts.use) {
89
- if (0 < out->opts->dump_opts.array_size) {
90
- APPEND_CHARS(out->cur, out->opts->dump_opts.array_nl, out->opts->dump_opts.array_size);
91
- }
92
- if (0 < out->opts->dump_opts.indent_size) {
93
- int i;
94
-
95
- for (i = depth; 0 < i; i--) {
96
- APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
97
- }
98
- }
99
- } else {
100
- fill_indent(out, depth);
101
- }
102
- *out->cur++ = ']';
56
+ if (out->opts->dump_opts.use) {
57
+ size = d2 * out->opts->dump_opts.indent_size + out->opts->dump_opts.array_size + 2;
58
+ size += out->opts->dump_opts.array_size;
59
+ size += out->opts->dump_opts.indent_size;
60
+ } else {
61
+ size = d2 * out->indent + 3;
62
+ }
63
+ for (; Qundef != *values; values++) {
64
+ assure_size(out, size);
65
+ if (out->opts->dump_opts.use) {
66
+ if (0 < out->opts->dump_opts.array_size) {
67
+ APPEND_CHARS(out->cur, out->opts->dump_opts.array_nl, out->opts->dump_opts.array_size);
68
+ }
69
+ if (0 < out->opts->dump_opts.indent_size) {
70
+ int i;
71
+
72
+ for (i = d2; 0 < i; i--) {
73
+ APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
74
+ }
75
+ }
76
+ } else {
77
+ fill_indent(out, d2);
78
+ }
79
+ oj_dump_compat_val(*values, d2, out, true);
80
+ if (Qundef != *(values + 1)) {
81
+ *out->cur++ = ',';
82
+ }
83
+ }
84
+ assure_size(out, size);
85
+ if (out->opts->dump_opts.use) {
86
+ if (0 < out->opts->dump_opts.array_size) {
87
+ APPEND_CHARS(out->cur, out->opts->dump_opts.array_nl, out->opts->dump_opts.array_size);
88
+ }
89
+ if (0 < out->opts->dump_opts.indent_size) {
90
+ int i;
91
+
92
+ for (i = depth; 0 < i; i--) {
93
+ APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
94
+ }
95
+ }
96
+ } else {
97
+ fill_indent(out, depth);
98
+ }
99
+ *out->cur++ = ']';
103
100
  }
104
101
  }
105
102
 
106
- static void
107
- dump_to_json(VALUE obj, Out out) {
108
- volatile VALUE rs;
109
- const char *s;
110
- int len;
103
+ static void dump_to_json(VALUE obj, Out out) {
104
+ volatile VALUE rs;
105
+ const char *s;
106
+ int len;
111
107
 
112
- if (RB_UNLIKELY(Yes == out->opts->trace)) {
113
- oj_trace("to_json", obj, __FILE__, __LINE__, 0, TraceRubyIn);
114
- }
108
+ TRACE(out->opts->trace, "to_json", obj, 0, TraceRubyIn);
115
109
  if (0 == rb_obj_method_arity(obj, oj_to_json_id)) {
116
- rs = rb_funcall(obj, oj_to_json_id, 0);
110
+ rs = rb_funcall(obj, oj_to_json_id, 0);
117
111
  } else {
118
- rs = rb_funcall2(obj, oj_to_json_id, out->argc, out->argv);
119
- }
120
- if (RB_UNLIKELY(Yes == out->opts->trace)) {
121
- oj_trace("to_json", obj, __FILE__, __LINE__, 0, TraceRubyOut);
112
+ rs = rb_funcall2(obj, oj_to_json_id, out->argc, out->argv);
122
113
  }
114
+ TRACE(out->opts->trace, "to_json", obj, 0, TraceRubyOut);
123
115
 
124
- s = RSTRING_PTR(rs);
116
+ StringValue(rs);
117
+ s = RSTRING_PTR(rs);
125
118
  len = (int)RSTRING_LEN(rs);
126
119
 
127
120
  assure_size(out, len + 1);
@@ -129,107 +122,104 @@ dump_to_json(VALUE obj, Out out) {
129
122
  *out->cur = '\0';
130
123
  }
131
124
 
132
- static void
133
- dump_array(VALUE a, int depth, Out out, bool as_ok) {
134
- size_t size;
135
- int i, cnt;
136
- int d2 = depth + 1;
137
- long id = oj_check_circular(a, out);
125
+ static void dump_array(VALUE a, int depth, Out out, bool as_ok) {
126
+ size_t size;
127
+ int i, cnt;
128
+ int d2 = depth + 1;
129
+ long id = oj_check_circular(a, out);
138
130
 
139
131
  if (0 > id) {
140
- raise_json_err("Too deeply nested", "NestingError");
141
- return;
132
+ raise_json_err("Too deeply nested", "NestingError");
133
+ return;
142
134
  }
143
- if (as_ok && !oj_use_hash_alt && rb_obj_class(a) != rb_cArray && rb_respond_to(a, oj_to_json_id)) {
144
- dump_to_json(a, out);
145
- return;
135
+ if (as_ok && !oj_use_array_alt && rb_obj_class(a) != rb_cArray && rb_respond_to(a, oj_to_json_id)) {
136
+ dump_to_json(a, out);
137
+ return;
146
138
  }
147
- cnt = (int)RARRAY_LEN(a);
139
+ cnt = (int)RARRAY_LEN(a);
148
140
  *out->cur++ = '[';
149
141
  assure_size(out, 2);
150
142
  if (0 == cnt) {
151
- *out->cur++ = ']';
143
+ *out->cur++ = ']';
152
144
  } else {
153
- if (out->opts->dump_opts.use) {
154
- size = d2 * out->opts->dump_opts.indent_size + out->opts->dump_opts.array_size + 1;
155
- } else {
156
- size = d2 * out->indent + 2;
157
- }
158
- assure_size(out, size * cnt);
159
- cnt--;
160
- for (i = 0; i <= cnt; i++) {
161
- if (out->opts->dump_opts.use) {
162
- if (0 < out->opts->dump_opts.array_size) {
163
- APPEND_CHARS(out->cur, out->opts->dump_opts.array_nl, out->opts->dump_opts.array_size);
164
- }
165
- if (0 < out->opts->dump_opts.indent_size) {
166
- int i;
167
- for (i = d2; 0 < i; i--) {
168
- APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
169
- }
170
- }
171
- } else {
172
- fill_indent(out, d2);
173
- }
174
- oj_dump_compat_val(RARRAY_AREF(a, i), d2, out, true);
175
- if (i < cnt) {
176
- *out->cur++ = ',';
177
- }
178
- }
179
- if (out->opts->dump_opts.use) {
180
- size = out->opts->dump_opts.array_size + out->opts->dump_opts.indent_size * depth + 1;
181
- assure_size(out, size);
182
- if (0 < out->opts->dump_opts.array_size) {
183
- APPEND_CHARS(out->cur, out->opts->dump_opts.array_nl, out->opts->dump_opts.array_size);
184
- }
185
- if (0 < out->opts->dump_opts.indent_size) {
186
- int i;
187
-
188
- for (i = depth; 0 < i; i--) {
189
- APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
190
- }
191
- }
192
- } else {
193
- size = depth * out->indent + 1;
194
- assure_size(out, size);
195
- fill_indent(out, depth);
196
- }
197
- *out->cur++ = ']';
145
+ if (out->opts->dump_opts.use) {
146
+ size = d2 * out->opts->dump_opts.indent_size + out->opts->dump_opts.array_size + 1;
147
+ } else {
148
+ size = d2 * out->indent + 2;
149
+ }
150
+ assure_size(out, size * cnt);
151
+ cnt--;
152
+ for (i = 0; i <= cnt; i++) {
153
+ if (out->opts->dump_opts.use) {
154
+ if (0 < out->opts->dump_opts.array_size) {
155
+ APPEND_CHARS(out->cur, out->opts->dump_opts.array_nl, out->opts->dump_opts.array_size);
156
+ }
157
+ if (0 < out->opts->dump_opts.indent_size) {
158
+ int i;
159
+ for (i = d2; 0 < i; i--) {
160
+ APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
161
+ }
162
+ }
163
+ } else {
164
+ fill_indent(out, d2);
165
+ }
166
+ oj_dump_compat_val(RARRAY_AREF(a, i), d2, out, true);
167
+ if (i < cnt) {
168
+ *out->cur++ = ',';
169
+ }
170
+ }
171
+ if (out->opts->dump_opts.use) {
172
+ size = out->opts->dump_opts.array_size + out->opts->dump_opts.indent_size * depth + 1;
173
+ assure_size(out, size);
174
+ if (0 < out->opts->dump_opts.array_size) {
175
+ APPEND_CHARS(out->cur, out->opts->dump_opts.array_nl, out->opts->dump_opts.array_size);
176
+ }
177
+ if (0 < out->opts->dump_opts.indent_size) {
178
+ int i;
179
+
180
+ for (i = depth; 0 < i; i--) {
181
+ APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
182
+ }
183
+ }
184
+ } else {
185
+ size = depth * out->indent + 1;
186
+ assure_size(out, size);
187
+ fill_indent(out, depth);
188
+ }
189
+ *out->cur++ = ']';
198
190
  }
199
191
  *out->cur = '\0';
200
192
  }
201
193
 
202
- static ID _dump_id = 0;
194
+ static ID _dump_id = 0;
203
195
 
204
- static void
205
- bigdecimal_alt(VALUE obj, int depth, Out out) {
206
- struct _attr attrs[] = {
207
- { "b", 1, Qnil },
208
- { NULL, 0, Qnil },
196
+ static void bigdecimal_alt(VALUE obj, int depth, Out out) {
197
+ struct _attr attrs[] = {
198
+ {"b", 1, Qnil},
199
+ {NULL, 0, Qnil},
209
200
  };
210
201
 
211
202
  if (0 == _dump_id) {
212
- _dump_id = rb_intern("_dump");
203
+ _dump_id = rb_intern("_dump");
213
204
  }
214
205
  attrs[0].value = rb_funcall(obj, _dump_id, 0);
215
206
 
216
207
  oj_code_attrs(obj, attrs, depth, out, true);
217
208
  }
218
209
 
219
- static ID real_id = 0;
220
- static ID imag_id = 0;
210
+ static ID real_id = 0;
211
+ static ID imag_id = 0;
221
212
 
222
- static void
223
- complex_alt(VALUE obj, int depth, Out out) {
224
- struct _attr attrs[] = {
225
- { "r", 1, Qnil },
226
- { "i", 1, Qnil },
227
- { NULL, 0, Qnil },
213
+ static void complex_alt(VALUE obj, int depth, Out out) {
214
+ struct _attr attrs[] = {
215
+ {"r", 1, Qnil},
216
+ {"i", 1, Qnil},
217
+ {NULL, 0, Qnil},
228
218
  };
229
219
 
230
220
  if (0 == real_id) {
231
- real_id = rb_intern("real");
232
- imag_id = rb_intern("imag");
221
+ real_id = rb_intern("real");
222
+ imag_id = rb_intern("imag");
233
223
  }
234
224
  attrs[0].value = rb_funcall(obj, real_id, 0);
235
225
  attrs[1].value = rb_funcall(obj, imag_id, 0);
@@ -237,25 +227,24 @@ complex_alt(VALUE obj, int depth, Out out) {
237
227
  oj_code_attrs(obj, attrs, depth, out, true);
238
228
  }
239
229
 
240
- static ID year_id = 0;
241
- static ID month_id = 0;
242
- static ID day_id = 0;
243
- static ID start_id = 0;
244
-
245
- static void
246
- date_alt(VALUE obj, int depth, Out out) {
247
- struct _attr attrs[] = {
248
- { "y", 1, Qnil },
249
- { "m", 1, Qnil },
250
- { "d", 1, Qnil },
251
- { "sg", 2, Qnil },
252
- { NULL, 0, Qnil },
230
+ static ID year_id = 0;
231
+ static ID month_id = 0;
232
+ static ID day_id = 0;
233
+ static ID start_id = 0;
234
+
235
+ static void date_alt(VALUE obj, int depth, Out out) {
236
+ struct _attr attrs[] = {
237
+ {"y", 1, Qnil},
238
+ {"m", 1, Qnil},
239
+ {"d", 1, Qnil},
240
+ {"sg", 2, Qnil},
241
+ {NULL, 0, Qnil},
253
242
  };
254
243
  if (0 == year_id) {
255
- year_id = rb_intern("year");
256
- month_id = rb_intern("month");
257
- day_id = rb_intern("day");
258
- start_id = rb_intern("start");
244
+ year_id = rb_intern("year");
245
+ month_id = rb_intern("month");
246
+ day_id = rb_intern("day");
247
+ start_id = rb_intern("start");
259
248
  }
260
249
  attrs[0].value = rb_funcall(obj, year_id, 0);
261
250
  attrs[1].value = rb_funcall(obj, month_id, 0);
@@ -265,33 +254,32 @@ date_alt(VALUE obj, int depth, Out out) {
265
254
  oj_code_attrs(obj, attrs, depth, out, true);
266
255
  }
267
256
 
268
- static ID hour_id = 0;
269
- static ID min_id = 0;
270
- static ID sec_id = 0;
271
- static ID offset_id = 0;
272
-
273
- static void
274
- datetime_alt(VALUE obj, int depth, Out out) {
275
- struct _attr attrs[] = {
276
- { "y", 1, Qnil },
277
- { "m", 1, Qnil },
278
- { "d", 1, Qnil },
279
- { "H", 1, Qnil },
280
- { "M", 1, Qnil },
281
- { "S", 1, Qnil },
282
- { "of", 2, Qnil },
283
- { "sg", 2, Qnil },
284
- { NULL, 0, Qnil },
257
+ static ID hour_id = 0;
258
+ static ID min_id = 0;
259
+ static ID sec_id = 0;
260
+ static ID offset_id = 0;
261
+
262
+ static void datetime_alt(VALUE obj, int depth, Out out) {
263
+ struct _attr attrs[] = {
264
+ {"y", 1, Qnil},
265
+ {"m", 1, Qnil},
266
+ {"d", 1, Qnil},
267
+ {"H", 1, Qnil},
268
+ {"M", 1, Qnil},
269
+ {"S", 1, Qnil},
270
+ {"of", 2, Qnil},
271
+ {"sg", 2, Qnil},
272
+ {NULL, 0, Qnil},
285
273
  };
286
274
  if (0 == hour_id) {
287
- year_id = rb_intern("year");
288
- month_id = rb_intern("month");
289
- day_id = rb_intern("day");
290
- hour_id = rb_intern("hour");
291
- min_id = rb_intern("min");
292
- sec_id = rb_intern("sec");
293
- offset_id = rb_intern("offset");
294
- start_id = rb_intern("start");
275
+ year_id = rb_intern("year");
276
+ month_id = rb_intern("month");
277
+ day_id = rb_intern("day");
278
+ hour_id = rb_intern("hour");
279
+ min_id = rb_intern("min");
280
+ sec_id = rb_intern("sec");
281
+ offset_id = rb_intern("offset");
282
+ start_id = rb_intern("start");
295
283
  }
296
284
  attrs[0].value = rb_funcall(obj, year_id, 0);
297
285
  attrs[1].value = rb_funcall(obj, month_id, 0);
@@ -299,24 +287,23 @@ datetime_alt(VALUE obj, int depth, Out out) {
299
287
  attrs[3].value = rb_funcall(obj, hour_id, 0);
300
288
  attrs[4].value = rb_funcall(obj, min_id, 0);
301
289
  attrs[5].value = rb_funcall(obj, sec_id, 0);
302
- attrs[6].value = rb_funcall(rb_funcall(obj, offset_id, 0), oj_to_s_id, 0);
290
+ attrs[6].value = oj_safe_string_convert(rb_funcall(obj, offset_id, 0));
303
291
  attrs[7].value = rb_funcall(obj, start_id, 0);
304
292
 
305
293
  oj_code_attrs(obj, attrs, depth, out, true);
306
294
  }
307
295
 
308
- static ID message_id = 0;
309
- static ID backtrace_id = 0;
296
+ static ID message_id = 0;
297
+ static ID backtrace_id = 0;
310
298
 
311
- static void
312
- exception_alt(VALUE obj, int depth, Out out) {
313
- int d3 = depth + 2;
314
- size_t size = d3 * out->indent + 2;
315
- size_t sep_len = out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2;
299
+ static void exception_alt(VALUE obj, int depth, Out out) {
300
+ int d3 = depth + 2;
301
+ size_t size = d3 * out->indent + 2;
302
+ size_t sep_len = out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2;
316
303
 
317
304
  if (0 == message_id) {
318
- message_id = rb_intern("message");
319
- backtrace_id = rb_intern("backtrace");
305
+ message_id = rb_intern("message");
306
+ backtrace_id = rb_intern("backtrace");
320
307
  }
321
308
  dump_obj_classname(rb_class2name(rb_obj_class(obj)), depth, out);
322
309
 
@@ -325,11 +312,11 @@ exception_alt(VALUE obj, int depth, Out out) {
325
312
  fill_indent(out, d3);
326
313
  APPEND_CHARS(out->cur, "\"m\"", 3);
327
314
  if (0 < out->opts->dump_opts.before_size) {
328
- APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
315
+ APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
329
316
  }
330
317
  *out->cur++ = ':';
331
318
  if (0 < out->opts->dump_opts.after_size) {
332
- APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
319
+ APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
333
320
  }
334
321
  oj_dump_str(rb_funcall(obj, message_id, 0), 0, out, false);
335
322
  assure_size(out, size + sep_len + 6);
@@ -337,40 +324,38 @@ exception_alt(VALUE obj, int depth, Out out) {
337
324
  fill_indent(out, d3);
338
325
  APPEND_CHARS(out->cur, "\"b\"", 3);
339
326
  if (0 < out->opts->dump_opts.before_size) {
340
- APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
327
+ APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
341
328
  }
342
329
  *out->cur++ = ':';
343
330
  if (0 < out->opts->dump_opts.after_size) {
344
- APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
331
+ APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
345
332
  }
346
333
  dump_array(rb_funcall(obj, backtrace_id, 0), depth, out, false);
347
334
  fill_indent(out, depth);
348
335
  *out->cur++ = '}';
349
- *out->cur = '\0';
336
+ *out->cur = '\0';
350
337
  }
351
338
 
352
- static ID table_id = 0;
339
+ static ID table_id = 0;
353
340
 
354
- static void
355
- openstruct_alt(VALUE obj, int depth, Out out) {
356
- struct _attr attrs[] = {
357
- { "t", 1, Qnil },
358
- { NULL, 0, Qnil },
341
+ static void openstruct_alt(VALUE obj, int depth, Out out) {
342
+ struct _attr attrs[] = {
343
+ {"t", 1, Qnil},
344
+ {NULL, 0, Qnil},
359
345
  };
360
346
  if (0 == table_id) {
361
- table_id = rb_intern("table");
347
+ table_id = rb_intern("table");
362
348
  }
363
349
  attrs[0].value = rb_funcall(obj, table_id, 0);
364
350
 
365
351
  oj_code_attrs(obj, attrs, depth, out, true);
366
352
  }
367
353
 
368
- static void
369
- range_alt(VALUE obj, int depth, Out out) {
370
- int d3 = depth + 2;
371
- size_t size = d3 * out->indent + 2;
372
- size_t sep_len = out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2;
373
- VALUE args[] = { Qundef, Qundef, Qundef, Qundef };
354
+ static void range_alt(VALUE obj, int depth, Out out) {
355
+ int d3 = depth + 2;
356
+ size_t size = d3 * out->indent + 2;
357
+ size_t sep_len = out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2;
358
+ VALUE args[] = {Qundef, Qundef, Qundef, Qundef};
374
359
 
375
360
  dump_obj_classname(rb_class2name(rb_obj_class(obj)), depth, out);
376
361
 
@@ -379,11 +364,11 @@ range_alt(VALUE obj, int depth, Out out) {
379
364
  fill_indent(out, d3);
380
365
  APPEND_CHARS(out->cur, "\"a\"", 3);
381
366
  if (0 < out->opts->dump_opts.before_size) {
382
- APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
367
+ APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
383
368
  }
384
369
  *out->cur++ = ':';
385
370
  if (0 < out->opts->dump_opts.after_size) {
386
- APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
371
+ APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
387
372
  }
388
373
  args[0] = rb_funcall(obj, oj_begin_id, 0);
389
374
  args[1] = rb_funcall(obj, oj_end_id, 0);
@@ -391,22 +376,21 @@ range_alt(VALUE obj, int depth, Out out) {
391
376
  dump_values_array(args, depth, out);
392
377
  fill_indent(out, depth);
393
378
  *out->cur++ = '}';
394
- *out->cur = '\0';
379
+ *out->cur = '\0';
395
380
  }
396
381
 
397
- static ID numerator_id = 0;
398
- static ID denominator_id = 0;
382
+ static ID numerator_id = 0;
383
+ static ID denominator_id = 0;
399
384
 
400
- static void
401
- rational_alt(VALUE obj, int depth, Out out) {
402
- struct _attr attrs[] = {
403
- { "n", 1, Qnil },
404
- { "d", 1, Qnil },
405
- { NULL, 0, Qnil },
385
+ static void rational_alt(VALUE obj, int depth, Out out) {
386
+ struct _attr attrs[] = {
387
+ {"n", 1, Qnil},
388
+ {"d", 1, Qnil},
389
+ {NULL, 0, Qnil},
406
390
  };
407
391
  if (0 == numerator_id) {
408
- numerator_id = rb_intern("numerator");
409
- denominator_id = rb_intern("denominator");
392
+ numerator_id = rb_intern("numerator");
393
+ denominator_id = rb_intern("denominator");
410
394
  }
411
395
  attrs[0].value = rb_funcall(obj, numerator_id, 0);
412
396
  attrs[1].value = rb_funcall(obj, denominator_id, 0);
@@ -414,19 +398,18 @@ rational_alt(VALUE obj, int depth, Out out) {
414
398
  oj_code_attrs(obj, attrs, depth, out, true);
415
399
  }
416
400
 
417
- static ID options_id = 0;
418
- static ID source_id = 0;
401
+ static ID options_id = 0;
402
+ static ID source_id = 0;
419
403
 
420
- static void
421
- regexp_alt(VALUE obj, int depth, Out out) {
422
- struct _attr attrs[] = {
423
- { "o", 1, Qnil },
424
- { "s", 1, Qnil },
425
- { NULL, 0, Qnil },
404
+ static void regexp_alt(VALUE obj, int depth, Out out) {
405
+ struct _attr attrs[] = {
406
+ {"o", 1, Qnil},
407
+ {"s", 1, Qnil},
408
+ {NULL, 0, Qnil},
426
409
  };
427
410
  if (0 == options_id) {
428
- options_id = rb_intern("options");
429
- source_id = rb_intern("source");
411
+ options_id = rb_intern("options");
412
+ source_id = rb_intern("source");
430
413
  }
431
414
  attrs[0].value = rb_funcall(obj, options_id, 0);
432
415
  attrs[1].value = rb_funcall(obj, source_id, 0);
@@ -434,24 +417,23 @@ regexp_alt(VALUE obj, int depth, Out out) {
434
417
  oj_code_attrs(obj, attrs, depth, out, true);
435
418
  }
436
419
 
437
- static void
438
- time_alt(VALUE obj, int depth, Out out) {
439
- struct _attr attrs[] = {
440
- { "s", 1, Qundef, 0, Qundef },
441
- { "n", 1, Qundef, 0, Qundef },
442
- { NULL, 0, Qnil },
420
+ static void time_alt(VALUE obj, int depth, Out out) {
421
+ struct _attr attrs[] = {
422
+ {"s", 1, Qundef, 0, Qundef},
423
+ {"n", 1, Qundef, 0, Qundef},
424
+ {NULL, 0, Qnil},
443
425
  };
444
- time_t sec;
445
- long long nsec;
426
+ time_t sec;
427
+ long long nsec;
446
428
 
447
429
  if (16 <= sizeof(struct timespec)) {
448
- struct timespec ts = rb_time_timespec(obj);
430
+ struct timespec ts = rb_time_timespec(obj);
449
431
 
450
- sec = (long long)ts.tv_sec;
451
- nsec = ts.tv_nsec;
432
+ sec = (long long)ts.tv_sec;
433
+ nsec = ts.tv_nsec;
452
434
  } else {
453
- sec = NUM2LL(rb_funcall2(obj, oj_tv_sec_id, 0, 0));
454
- nsec = NUM2LL(rb_funcall2(obj, oj_tv_nsec_id, 0, 0));
435
+ sec = NUM2LL(rb_funcall2(obj, oj_tv_sec_id, 0, 0));
436
+ nsec = NUM2LL(rb_funcall2(obj, oj_tv_nsec_id, 0, 0));
455
437
  }
456
438
 
457
439
  attrs[0].num = sec;
@@ -460,68 +442,68 @@ time_alt(VALUE obj, int depth, Out out) {
460
442
  oj_code_attrs(obj, attrs, depth, out, true);
461
443
  }
462
444
 
463
- struct _code oj_compat_codes[] = {
464
- { "BigDecimal", Qnil, bigdecimal_alt, NULL, false },
465
- { "Complex", Qnil, complex_alt, NULL, false },
466
- { "Date", Qnil, date_alt, false },
467
- { "DateTime", Qnil, datetime_alt, NULL, false },
468
- { "OpenStruct", Qnil, openstruct_alt, NULL, false },
469
- { "Range", Qnil, range_alt, NULL, false },
470
- { "Rational", Qnil, rational_alt, NULL, false },
471
- { "Regexp", Qnil, regexp_alt, NULL, false },
472
- { "Time", Qnil, time_alt, NULL, false },
445
+ struct _code oj_compat_codes[] = {
446
+ {"BigDecimal", Qnil, bigdecimal_alt, NULL, false},
447
+ {"Complex", Qnil, complex_alt, NULL, false},
448
+ {"Date", Qnil, date_alt, false},
449
+ {"DateTime", Qnil, datetime_alt, NULL, false},
450
+ {"OpenStruct", Qnil, openstruct_alt, NULL, false},
451
+ {"Range", Qnil, range_alt, NULL, false},
452
+ {"Rational", Qnil, rational_alt, NULL, false},
453
+ {"Regexp", Qnil, regexp_alt, NULL, false},
454
+ {"Time", Qnil, time_alt, NULL, false},
473
455
  // TBD the rest of the library classes
474
- { NULL, Qundef, NULL, NULL, false },
456
+ {NULL, Qundef, NULL, NULL, false},
475
457
  };
476
458
 
477
459
  VALUE
478
460
  oj_add_to_json(int argc, VALUE *argv, VALUE self) {
479
- Code a;
461
+ Code a;
480
462
 
481
463
  if (0 == argc) {
482
- for (a = oj_compat_codes; NULL != a->name; a++) {
483
- if (Qnil == a->clas || Qundef == a->clas) {
484
- a->clas = rb_const_get_at(rb_cObject, rb_intern(a->name));
485
- }
486
- a->active = true;
487
- }
488
- use_struct_alt = true;
489
- use_exception_alt = true;
490
- use_bignum_alt = true;
491
- oj_use_hash_alt = true;
492
- oj_use_array_alt = true;
464
+ for (a = oj_compat_codes; NULL != a->name; a++) {
465
+ if (Qnil == a->clas || Qundef == a->clas) {
466
+ a->clas = rb_const_get_at(rb_cObject, rb_intern(a->name));
467
+ }
468
+ a->active = true;
469
+ }
470
+ use_struct_alt = true;
471
+ use_exception_alt = true;
472
+ use_bignum_alt = true;
473
+ oj_use_hash_alt = true;
474
+ oj_use_array_alt = true;
493
475
  } else {
494
- for (; 0 < argc; argc--, argv++) {
495
- if (rb_cStruct == *argv) {
496
- use_struct_alt = true;
497
- continue;
498
- }
499
- if (rb_eException == *argv) {
500
- use_exception_alt = true;
501
- continue;
502
- }
503
- if (rb_cInteger == *argv) {
504
- use_bignum_alt = true;
505
- continue;
506
- }
507
- if (rb_cHash == *argv) {
508
- oj_use_hash_alt = true;
509
- continue;
510
- }
511
- if (rb_cArray == *argv) {
512
- oj_use_array_alt = true;
513
- continue;
514
- }
515
- for (a = oj_compat_codes; NULL != a->name; a++) {
516
- if (Qnil == a->clas || Qundef == a->clas) {
517
- a->clas = rb_const_get_at(rb_cObject, rb_intern(a->name));
518
- }
519
- if (*argv == a->clas) {
520
- a->active = true;
521
- break;
522
- }
523
- }
524
- }
476
+ for (; 0 < argc; argc--, argv++) {
477
+ if (rb_cStruct == *argv) {
478
+ use_struct_alt = true;
479
+ continue;
480
+ }
481
+ if (rb_eException == *argv) {
482
+ use_exception_alt = true;
483
+ continue;
484
+ }
485
+ if (rb_cInteger == *argv) {
486
+ use_bignum_alt = true;
487
+ continue;
488
+ }
489
+ if (rb_cHash == *argv) {
490
+ oj_use_hash_alt = true;
491
+ continue;
492
+ }
493
+ if (rb_cArray == *argv) {
494
+ oj_use_array_alt = true;
495
+ continue;
496
+ }
497
+ for (a = oj_compat_codes; NULL != a->name; a++) {
498
+ if (Qnil == a->clas || Qundef == a->clas) {
499
+ a->clas = rb_const_get_at(rb_cObject, rb_intern(a->name));
500
+ }
501
+ if (*argv == a->clas) {
502
+ a->active = true;
503
+ break;
504
+ }
505
+ }
506
+ }
525
507
  }
526
508
  return Qnil;
527
509
  }
@@ -529,36 +511,36 @@ oj_add_to_json(int argc, VALUE *argv, VALUE self) {
529
511
  VALUE
530
512
  oj_remove_to_json(int argc, VALUE *argv, VALUE self) {
531
513
  if (0 == argc) {
532
- oj_code_set_active(oj_compat_codes, Qnil, false);
533
- use_struct_alt = false;
534
- use_exception_alt = false;
535
- use_bignum_alt = false;
536
- oj_use_hash_alt = false;
537
- oj_use_array_alt = false;
514
+ oj_code_set_active(oj_compat_codes, Qnil, false);
515
+ use_struct_alt = false;
516
+ use_exception_alt = false;
517
+ use_bignum_alt = false;
518
+ oj_use_hash_alt = false;
519
+ oj_use_array_alt = false;
538
520
  } else {
539
- for (; 0 < argc; argc--, argv++) {
540
- if (rb_cStruct == *argv) {
541
- use_struct_alt = false;
542
- continue;
543
- }
544
- if (rb_eException == *argv) {
545
- use_exception_alt = false;
546
- continue;
547
- }
548
- if (rb_cInteger == *argv) {
549
- use_bignum_alt = false;
550
- continue;
551
- }
552
- if (rb_cHash == *argv) {
553
- oj_use_hash_alt = false;
554
- continue;
555
- }
556
- if (rb_cArray == *argv) {
557
- oj_use_array_alt = false;
558
- continue;
559
- }
560
- oj_code_set_active(oj_compat_codes, *argv, false);
561
- }
521
+ for (; 0 < argc; argc--, argv++) {
522
+ if (rb_cStruct == *argv) {
523
+ use_struct_alt = false;
524
+ continue;
525
+ }
526
+ if (rb_eException == *argv) {
527
+ use_exception_alt = false;
528
+ continue;
529
+ }
530
+ if (rb_cInteger == *argv) {
531
+ use_bignum_alt = false;
532
+ continue;
533
+ }
534
+ if (rb_cHash == *argv) {
535
+ oj_use_hash_alt = false;
536
+ continue;
537
+ }
538
+ if (rb_cArray == *argv) {
539
+ oj_use_array_alt = false;
540
+ continue;
541
+ }
542
+ oj_code_set_active(oj_compat_codes, *argv, false);
543
+ }
562
544
  }
563
545
  return Qnil;
564
546
  }
@@ -566,366 +548,349 @@ oj_remove_to_json(int argc, VALUE *argv, VALUE self) {
566
548
  // The JSON gem is inconsistent with handling of infinity. Using
567
549
  // JSON.dump(0.1/0) returns the string Infinity but (0.1/0).to_json raise and
568
550
  // exception. Worse, for BigDecimals a quoted "Infinity" is returned.
569
- static void
570
- dump_float(VALUE obj, int depth, Out out, bool as_ok) {
571
- char buf[64];
572
- char *b;
573
- double d = rb_num2dbl(obj);
574
- int cnt = 0;
551
+ static void dump_float(VALUE obj, int depth, Out out, bool as_ok) {
552
+ char buf[64];
553
+ char *b;
554
+ double d = rb_num2dbl(obj);
555
+ int cnt = 0;
575
556
 
576
557
  if (0.0 == d) {
577
- b = buf;
578
- *b++ = '0';
579
- *b++ = '.';
580
- *b++ = '0';
581
- *b++ = '\0';
582
- cnt = 3;
558
+ b = buf;
559
+ *b++ = '0';
560
+ *b++ = '.';
561
+ *b++ = '0';
562
+ *b++ = '\0';
563
+ cnt = 3;
583
564
  } else if (OJ_INFINITY == d) {
584
- if (WordNan == out->opts->dump_opts.nan_dump) {
585
- strcpy(buf, "Infinity");
586
- cnt = 8;
587
- } else {
588
- raise_json_err("Infinity not allowed in JSON.", "GeneratorError");
589
- }
565
+ if (WordNan == out->opts->dump_opts.nan_dump) {
566
+ strcpy(buf, "Infinity");
567
+ cnt = 8;
568
+ } else {
569
+ raise_json_err("Infinity not allowed in JSON.", "GeneratorError");
570
+ }
590
571
  } else if (-OJ_INFINITY == d) {
591
- if (WordNan == out->opts->dump_opts.nan_dump) {
592
- strcpy(buf, "-Infinity");
593
- cnt = 9;
594
- } else {
595
- raise_json_err("-Infinity not allowed in JSON.", "GeneratorError");
596
- }
572
+ if (WordNan == out->opts->dump_opts.nan_dump) {
573
+ strcpy(buf, "-Infinity");
574
+ cnt = 9;
575
+ } else {
576
+ raise_json_err("-Infinity not allowed in JSON.", "GeneratorError");
577
+ }
597
578
  } else if (isnan(d)) {
598
- if (WordNan == out->opts->dump_opts.nan_dump) {
599
- strcpy(buf, "NaN");
600
- cnt = 3;
601
- } else {
602
- raise_json_err("NaN not allowed in JSON.", "GeneratorError");
603
- }
579
+ if (WordNan == out->opts->dump_opts.nan_dump) {
580
+ strcpy(buf, "NaN");
581
+ cnt = 3;
582
+ } else {
583
+ raise_json_err("NaN not allowed in JSON.", "GeneratorError");
584
+ }
604
585
  } else if (d == (double)(long long int)d) {
605
- cnt = snprintf(buf, sizeof(buf), "%.1f", d);
586
+ cnt = snprintf(buf, sizeof(buf), "%.1f", d);
606
587
  } else if (oj_rails_float_opt) {
607
- cnt = oj_dump_float_printf(buf, sizeof(buf), obj, d, "%0.16g");
588
+ cnt = oj_dump_float_printf(buf, sizeof(buf), obj, d, "%0.16g");
608
589
  } else {
609
- volatile VALUE rstr = rb_funcall(obj, oj_to_s_id, 0);
590
+ volatile VALUE rstr = oj_safe_string_convert(obj);
610
591
 
611
- strcpy(buf, RSTRING_PTR(rstr));
612
- cnt = (int)RSTRING_LEN(rstr);
592
+ strcpy(buf, RSTRING_PTR(rstr));
593
+ cnt = (int)RSTRING_LEN(rstr);
613
594
  }
614
595
  assure_size(out, cnt);
615
596
  APPEND_CHARS(out->cur, buf, cnt);
616
597
  *out->cur = '\0';
617
598
  }
618
599
 
619
- static int
620
- hash_cb(VALUE key, VALUE value, VALUE ov) {
621
- Out out = (Out)ov;
622
- int depth = out->depth;
600
+ static int hash_cb(VALUE key, VALUE value, VALUE ov) {
601
+ Out out = (Out)ov;
602
+ int depth = out->depth;
623
603
 
624
604
  if (out->omit_nil && Qnil == value) {
625
- return ST_CONTINUE;
605
+ return ST_CONTINUE;
626
606
  }
627
607
  if (!out->opts->dump_opts.use) {
628
- assure_size(out, depth * out->indent + 1);
629
- fill_indent(out, depth);
608
+ assure_size(out, depth * out->indent + 1);
609
+ fill_indent(out, depth);
630
610
  } else {
631
- assure_size(out, depth * out->opts->dump_opts.indent_size + out->opts->dump_opts.hash_size + 1);
632
- if (0 < out->opts->dump_opts.hash_size) {
633
- APPEND_CHARS(out->cur, out->opts->dump_opts.hash_nl, out->opts->dump_opts.hash_size);
634
- }
635
- if (0 < out->opts->dump_opts.indent_size) {
636
- int i;
637
- for (i = depth; 0 < i; i--) {
638
- APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
639
- }
640
- }
611
+ assure_size(out, depth * out->opts->dump_opts.indent_size + out->opts->dump_opts.hash_size + 1);
612
+ if (0 < out->opts->dump_opts.hash_size) {
613
+ APPEND_CHARS(out->cur, out->opts->dump_opts.hash_nl, out->opts->dump_opts.hash_size);
614
+ }
615
+ if (0 < out->opts->dump_opts.indent_size) {
616
+ int i;
617
+ for (i = depth; 0 < i; i--) {
618
+ APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
619
+ }
620
+ }
641
621
  }
642
622
  switch (rb_type(key)) {
643
- case T_STRING:
644
- oj_dump_str(key, 0, out, false);
645
- break;
646
- case T_SYMBOL:
647
- oj_dump_sym(key, 0, out, false);
648
- break;
623
+ case T_STRING: oj_dump_str(key, 0, out, false); break;
624
+ case T_SYMBOL: oj_dump_sym(key, 0, out, false); break;
649
625
  default:
650
- /*rb_raise(rb_eTypeError, "In :compat mode all Hash keys must be Strings or Symbols, not %s.\n", rb_class2name(rb_obj_class(key)));*/
651
- oj_dump_str(rb_funcall(key, oj_to_s_id, 0), 0, out, false);
652
- break;
626
+ /*rb_raise(rb_eTypeError, "In :compat mode all Hash keys must be Strings or Symbols, not %s.\n",
627
+ * rb_class2name(rb_obj_class(key)));*/
628
+ oj_dump_str(oj_safe_string_convert(key), 0, out, false);
629
+ break;
653
630
  }
654
631
  if (!out->opts->dump_opts.use) {
655
- *out->cur++ = ':';
632
+ *out->cur++ = ':';
656
633
  } else {
657
- assure_size(out, out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2);
658
- if (0 < out->opts->dump_opts.before_size) {
659
- APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
660
- }
661
- *out->cur++ = ':';
662
- if (0 < out->opts->dump_opts.after_size) {
663
- APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
664
- }
634
+ assure_size(out, out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2);
635
+ if (0 < out->opts->dump_opts.before_size) {
636
+ APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
637
+ }
638
+ *out->cur++ = ':';
639
+ if (0 < out->opts->dump_opts.after_size) {
640
+ APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
641
+ }
665
642
  }
666
643
  oj_dump_compat_val(value, depth, out, true);
667
- out->depth = depth;
644
+ out->depth = depth;
668
645
  *out->cur++ = ',';
669
646
 
670
647
  return ST_CONTINUE;
671
648
  }
672
649
 
673
- static void
674
- dump_hash(VALUE obj, int depth, Out out, bool as_ok) {
675
- int cnt;
676
- long id = oj_check_circular(obj, out);
650
+ static void dump_hash(VALUE obj, int depth, Out out, bool as_ok) {
651
+ int cnt;
652
+ long id = oj_check_circular(obj, out);
677
653
 
678
654
  if (0 > id) {
679
- raise_json_err("Too deeply nested", "NestingError");
680
- return;
655
+ raise_json_err("Too deeply nested", "NestingError");
656
+ return;
681
657
  }
682
658
  if (as_ok && !oj_use_hash_alt && rb_obj_class(obj) != rb_cHash && rb_respond_to(obj, oj_to_json_id)) {
683
- dump_to_json(obj, out);
684
- return;
659
+ dump_to_json(obj, out);
660
+ return;
685
661
  }
686
662
  cnt = (int)RHASH_SIZE(obj);
687
663
  assure_size(out, 2);
688
664
  if (0 == cnt) {
689
- APPEND_CHARS(out->cur, "{}", 2);
665
+ APPEND_CHARS(out->cur, "{}", 2);
690
666
  } else {
691
- *out->cur++ = '{';
692
- out->depth = depth + 1;
693
- rb_hash_foreach(obj, hash_cb, (VALUE)out);
694
- if (',' == *(out->cur - 1)) {
695
- out->cur--; // backup to overwrite last comma
696
- }
697
- if (!out->opts->dump_opts.use) {
698
- assure_size(out, depth * out->indent + 2);
699
- fill_indent(out, depth);
700
- } else {
701
- assure_size(out, depth * out->opts->dump_opts.indent_size + out->opts->dump_opts.hash_size + 1);
702
- if (0 < out->opts->dump_opts.hash_size) {
703
- APPEND_CHARS(out->cur, out->opts->dump_opts.hash_nl, out->opts->dump_opts.hash_size);
704
- }
705
- if (0 < out->opts->dump_opts.indent_size) {
706
- int i;
707
-
708
- for (i = depth; 0 < i; i--) {
709
- APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
710
- }
711
- }
712
- }
713
- *out->cur++ = '}';
667
+ *out->cur++ = '{';
668
+ out->depth = depth + 1;
669
+ rb_hash_foreach(obj, hash_cb, (VALUE)out);
670
+ if (',' == *(out->cur - 1)) {
671
+ out->cur--; // backup to overwrite last comma
672
+ }
673
+ if (!out->opts->dump_opts.use) {
674
+ assure_size(out, depth * out->indent + 2);
675
+ fill_indent(out, depth);
676
+ } else {
677
+ assure_size(out, depth * out->opts->dump_opts.indent_size + out->opts->dump_opts.hash_size + 1);
678
+ if (0 < out->opts->dump_opts.hash_size) {
679
+ APPEND_CHARS(out->cur, out->opts->dump_opts.hash_nl, out->opts->dump_opts.hash_size);
680
+ }
681
+ if (0 < out->opts->dump_opts.indent_size) {
682
+ int i;
683
+
684
+ for (i = depth; 0 < i; i--) {
685
+ APPEND_CHARS(out->cur, out->opts->dump_opts.indent_str, out->opts->dump_opts.indent_size);
686
+ }
687
+ }
688
+ }
689
+ *out->cur++ = '}';
714
690
  }
715
691
  *out->cur = '\0';
716
692
  }
717
693
 
718
694
  // In compat mode only the first call check for to_json. After that to_s is
719
695
  // called.
720
- static void
721
- dump_obj(VALUE obj, int depth, Out out, bool as_ok) {
696
+ static void dump_obj(VALUE obj, int depth, Out out, bool as_ok) {
722
697
  if (oj_code_dump(oj_compat_codes, obj, depth, out)) {
723
- return;
698
+ return;
724
699
  }
725
700
  if (use_exception_alt && rb_obj_is_kind_of(obj, rb_eException)) {
726
- exception_alt(obj, depth, out);
727
- return;
701
+ exception_alt(obj, depth, out);
702
+ return;
728
703
  }
729
704
  if (Yes == out->opts->raw_json && rb_respond_to(obj, oj_raw_json_id)) {
730
- oj_dump_raw_json(obj, depth, out);
731
- return;
705
+ oj_dump_raw_json(obj, depth, out);
706
+ return;
732
707
  }
733
708
  if (as_ok && rb_respond_to(obj, oj_to_json_id)) {
734
- dump_to_json(obj, out);
735
- return;
709
+ dump_to_json(obj, out);
710
+ return;
736
711
  }
737
712
  // Nothing else matched so encode as a JSON object with Ruby obj members
738
713
  // as JSON object members.
739
714
  oj_dump_obj_to_s(obj, out);
740
715
  }
741
716
 
742
- static void
743
- dump_struct(VALUE obj, int depth, Out out, bool as_ok) {
744
- VALUE clas = rb_obj_class(obj);
717
+ static void dump_struct(VALUE obj, int depth, Out out, bool as_ok) {
718
+ VALUE clas = rb_obj_class(obj);
745
719
 
746
720
  if (oj_code_dump(oj_compat_codes, obj, depth, out)) {
747
- return;
721
+ return;
748
722
  }
749
723
  if (rb_cRange == clas) {
750
- *out->cur++ = '"';
751
- oj_dump_compat_val(rb_funcall(obj, oj_begin_id, 0), 0, out, false);
752
- assure_size(out, 3);
753
- APPEND_CHARS(out->cur, "..", 2);
754
- if (Qtrue == rb_funcall(obj, oj_exclude_end_id, 0)) {
755
- *out->cur++ = '.';
756
- }
757
- oj_dump_compat_val(rb_funcall(obj, oj_end_id, 0), 0, out, false);
758
- *out->cur++ = '"';
759
-
760
- return;
724
+ *out->cur++ = '"';
725
+ oj_dump_compat_val(rb_funcall(obj, oj_begin_id, 0), 0, out, false);
726
+ assure_size(out, 3);
727
+ APPEND_CHARS(out->cur, "..", 2);
728
+ if (Qtrue == rb_funcall(obj, oj_exclude_end_id, 0)) {
729
+ *out->cur++ = '.';
730
+ }
731
+ oj_dump_compat_val(rb_funcall(obj, oj_end_id, 0), 0, out, false);
732
+ *out->cur++ = '"';
733
+
734
+ return;
761
735
  }
762
736
  if (as_ok && rb_respond_to(obj, oj_to_json_id)) {
763
- dump_to_json(obj, out);
737
+ dump_to_json(obj, out);
764
738
 
765
- return;
739
+ return;
766
740
  }
767
741
  if (use_struct_alt) {
768
- int d3 = depth + 2;
769
- size_t size = d3 * out->indent + 2;
770
- size_t sep_len = out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2;
771
- const char *classname = rb_class2name(rb_obj_class(obj));
772
- VALUE args[100];
773
- int cnt;
774
- int i;
775
-
776
- if (NULL == classname || '#' == *classname) {
777
- raise_json_err("Only named structs are supported.", "JSONError");
778
- }
742
+ int d3 = depth + 2;
743
+ size_t size = d3 * out->indent + 2;
744
+ size_t sep_len = out->opts->dump_opts.before_size + out->opts->dump_opts.after_size + 2;
745
+ const char *classname = rb_class2name(rb_obj_class(obj));
746
+ VALUE args[100];
747
+ int cnt;
748
+ int i;
749
+
750
+ if (NULL == classname || '#' == *classname) {
751
+ raise_json_err("Only named structs are supported.", "JSONError");
752
+ }
779
753
  #ifdef RSTRUCT_LEN
780
754
  #if RSTRUCT_LEN_RETURNS_INTEGER_OBJECT
781
- cnt = (int)NUM2LONG(RSTRUCT_LEN(obj));
782
- #else // RSTRUCT_LEN_RETURNS_INTEGER_OBJECT
783
- cnt = (int)RSTRUCT_LEN(obj);
784
- #endif // RSTRUCT_LEN_RETURNS_INTEGER_OBJECT
755
+ cnt = (int)NUM2LONG(RSTRUCT_LEN(obj));
756
+ #else // RSTRUCT_LEN_RETURNS_INTEGER_OBJECT
757
+ cnt = (int)RSTRUCT_LEN(obj);
758
+ #endif // RSTRUCT_LEN_RETURNS_INTEGER_OBJECT
785
759
  #else
786
- // This is a bit risky as a struct in C ruby is not the same as a Struct
787
- // class in interpreted Ruby so length() may not be defined.
788
- cnt = FIX2INT(rb_funcall2(obj, oj_length_id, 0, 0));
760
+ // This is a bit risky as a struct in C ruby is not the same as a Struct
761
+ // class in interpreted Ruby so length() may not be defined.
762
+ cnt = FIX2INT(rb_funcall2(obj, oj_length_id, 0, 0));
789
763
  #endif
790
- if (sizeof(args) / sizeof(*args) <= (size_t)cnt) {
791
- // TBD allocate and try again
792
- cnt = 99;
793
- }
794
- dump_obj_classname(rb_class2name(rb_obj_class(obj)), depth, out);
795
-
796
- assure_size(out, size + sep_len + 6);
797
- *out->cur++ = ',';
798
- fill_indent(out, d3);
799
- APPEND_CHARS(out->cur, "\"v\"", 3);
800
- if (0 < out->opts->dump_opts.before_size) {
801
- APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
802
- }
803
- *out->cur++ = ':';
804
- if (0 < out->opts->dump_opts.after_size) {
805
- APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
806
- }
807
- for (i = 0; i < cnt; i++) {
764
+ if (sizeof(args) / sizeof(*args) <= (size_t)cnt) {
765
+ // TBD allocate and try again
766
+ cnt = 99;
767
+ }
768
+ dump_obj_classname(rb_class2name(rb_obj_class(obj)), depth, out);
769
+
770
+ assure_size(out, size + sep_len + 6);
771
+ *out->cur++ = ',';
772
+ fill_indent(out, d3);
773
+ APPEND_CHARS(out->cur, "\"v\"", 3);
774
+ if (0 < out->opts->dump_opts.before_size) {
775
+ APPEND_CHARS(out->cur, out->opts->dump_opts.before_sep, out->opts->dump_opts.before_size);
776
+ }
777
+ *out->cur++ = ':';
778
+ if (0 < out->opts->dump_opts.after_size) {
779
+ APPEND_CHARS(out->cur, out->opts->dump_opts.after_sep, out->opts->dump_opts.after_size);
780
+ }
781
+ for (i = 0; i < cnt; i++) {
808
782
  #ifdef RSTRUCT_LEN
809
- args[i] = RSTRUCT_GET(obj, i);
783
+ args[i] = RSTRUCT_GET(obj, i);
810
784
  #else
811
- args[i] = rb_struct_aref(obj, INT2FIX(i));
785
+ args[i] = rb_struct_aref(obj, INT2FIX(i));
812
786
  #endif
813
- }
814
- args[cnt] = Qundef;
815
- dump_values_array(args, depth, out);
816
- fill_indent(out, depth);
817
- *out->cur++ = '}';
818
- *out->cur = '\0';
787
+ }
788
+ args[cnt] = Qundef;
789
+ dump_values_array(args, depth, out);
790
+ fill_indent(out, depth);
791
+ *out->cur++ = '}';
792
+ *out->cur = '\0';
819
793
  } else {
820
- oj_dump_obj_to_s(obj, out);
794
+ oj_dump_obj_to_s(obj, out);
821
795
  }
822
796
  }
823
797
 
824
- static void
825
- dump_bignum(VALUE obj, int depth, Out out, bool as_ok) {
798
+ static void dump_bignum(VALUE obj, int depth, Out out, bool as_ok) {
826
799
  // The json gem uses to_s explicitly. to_s can be overridden while
827
800
  // rb_big2str can not so unless overridden by using add_to_json(Integer)
828
801
  // this must use to_s to pass the json gem unit tests.
829
- volatile VALUE rs;
830
- int cnt;
831
- bool dump_as_string = false;
802
+ volatile VALUE rs;
803
+ int cnt;
804
+ bool dump_as_string = false;
832
805
 
833
806
  if (use_bignum_alt) {
834
- rs = rb_big2str(obj, 10);
807
+ rs = rb_big2str(obj, 10);
835
808
  } else {
836
- rs = rb_funcall(obj, oj_to_s_id, 0);
809
+ rs = oj_safe_string_convert(obj);
837
810
  }
838
811
  rb_check_type(rs, T_STRING);
839
812
  cnt = (int)RSTRING_LEN(rs);
840
813
 
841
814
  if (out->opts->int_range_min != 0 || out->opts->int_range_max != 0) {
842
- dump_as_string = true; // Bignum cannot be inside of Fixnum range
843
- assure_size(out, cnt + 2);
844
- *out->cur++ = '"';
815
+ dump_as_string = true; // Bignum cannot be inside of Fixnum range
816
+ assure_size(out, cnt + 2);
817
+ *out->cur++ = '"';
845
818
  } else {
846
- assure_size(out, cnt);
819
+ assure_size(out, cnt);
847
820
  }
848
821
  APPEND_CHARS(out->cur, RSTRING_PTR(rs), cnt);
849
822
  if (dump_as_string) {
850
- *out->cur++ = '"';
823
+ *out->cur++ = '"';
851
824
  }
852
825
  *out->cur = '\0';
853
826
  }
854
827
 
855
- static DumpFunc compat_funcs[] = {
856
- NULL, // RUBY_T_NONE = 0x00,
857
- dump_obj, // RUBY_T_OBJECT = 0x01,
858
- oj_dump_class, // RUBY_T_CLASS = 0x02,
859
- oj_dump_class, // RUBY_T_MODULE = 0x03,
860
- dump_float, // RUBY_T_FLOAT = 0x04,
861
- oj_dump_str, // RUBY_T_STRING = 0x05,
862
- dump_obj, // RUBY_T_REGEXP = 0x06,
863
- dump_array, // RUBY_T_ARRAY = 0x07,
864
- dump_hash, // RUBY_T_HASH = 0x08,
865
- dump_struct, // RUBY_T_STRUCT = 0x09,
866
- dump_bignum, // RUBY_T_BIGNUM = 0x0a,
867
- NULL, // RUBY_T_FILE = 0x0b,
868
- dump_obj, // RUBY_T_DATA = 0x0c,
869
- NULL, // RUBY_T_MATCH = 0x0d,
870
- dump_obj, // RUBY_T_COMPLEX = 0x0e,
871
- dump_obj, // RUBY_T_RATIONAL = 0x0f,
872
- NULL, // 0x10
873
- oj_dump_nil, // RUBY_T_NIL = 0x11,
874
- oj_dump_true, // RUBY_T_TRUE = 0x12,
875
- oj_dump_false, // RUBY_T_FALSE = 0x13,
876
- oj_dump_sym, // RUBY_T_SYMBOL = 0x14,
877
- oj_dump_fixnum, // RUBY_T_FIXNUM = 0x15,
828
+ static DumpFunc compat_funcs[] = {
829
+ NULL, // RUBY_T_NONE = 0x00,
830
+ dump_obj, // RUBY_T_OBJECT = 0x01,
831
+ oj_dump_class, // RUBY_T_CLASS = 0x02,
832
+ oj_dump_class, // RUBY_T_MODULE = 0x03,
833
+ dump_float, // RUBY_T_FLOAT = 0x04,
834
+ oj_dump_str, // RUBY_T_STRING = 0x05,
835
+ dump_obj, // RUBY_T_REGEXP = 0x06,
836
+ dump_array, // RUBY_T_ARRAY = 0x07,
837
+ dump_hash, // RUBY_T_HASH = 0x08,
838
+ dump_struct, // RUBY_T_STRUCT = 0x09,
839
+ dump_bignum, // RUBY_T_BIGNUM = 0x0a,
840
+ NULL, // RUBY_T_FILE = 0x0b,
841
+ dump_obj, // RUBY_T_DATA = 0x0c,
842
+ NULL, // RUBY_T_MATCH = 0x0d,
843
+ dump_obj, // RUBY_T_COMPLEX = 0x0e,
844
+ dump_obj, // RUBY_T_RATIONAL = 0x0f,
845
+ NULL, // 0x10
846
+ oj_dump_nil, // RUBY_T_NIL = 0x11,
847
+ oj_dump_true, // RUBY_T_TRUE = 0x12,
848
+ oj_dump_false, // RUBY_T_FALSE = 0x13,
849
+ oj_dump_sym, // RUBY_T_SYMBOL = 0x14,
850
+ oj_dump_fixnum, // RUBY_T_FIXNUM = 0x15,
878
851
  };
879
852
 
880
- static void
881
- set_state_depth(VALUE state, int depth) {
882
- VALUE json_module = rb_const_get_at(rb_cObject, rb_intern("JSON"));
883
- VALUE ext = rb_const_get(json_module, rb_intern("Ext"));
884
- VALUE generator = rb_const_get(ext, rb_intern("Generator"));
885
- VALUE state_class = rb_const_get(generator, rb_intern("State"));
853
+ static void set_state_depth(VALUE state, int depth) {
854
+ if (0 == rb_const_defined(rb_cObject, rb_intern("JSON"))) {
855
+ rb_require("oj/json");
856
+ }
857
+ {
858
+ VALUE json_module = rb_const_get_at(rb_cObject, rb_intern("JSON"));
859
+ VALUE ext = rb_const_get(json_module, rb_intern("Ext"));
860
+ VALUE generator = rb_const_get(ext, rb_intern("Generator"));
861
+ VALUE state_class = rb_const_get(generator, rb_intern("State"));
886
862
 
887
- if (state_class == rb_obj_class(state)) {
888
- rb_funcall(state, rb_intern("depth="), 1, INT2NUM(depth));
863
+ if (state_class == rb_obj_class(state)) {
864
+ rb_funcall(state, rb_intern("depth="), 1, INT2NUM(depth));
865
+ }
889
866
  }
890
867
  }
891
868
 
892
- void
893
- oj_dump_compat_val(VALUE obj, int depth, Out out, bool as_ok) {
894
- int type = rb_type(obj);
869
+ void oj_dump_compat_val(VALUE obj, int depth, Out out, bool as_ok) {
870
+ int type = rb_type(obj);
895
871
 
896
- if (RB_UNLIKELY(Yes == out->opts->trace)) {
897
- oj_trace("dump", obj, __FILE__, __LINE__, depth, TraceIn);
898
- }
872
+ TRACE(out->opts->trace, "dump", obj, depth, TraceIn);
873
+ // The max_nesting logic is that an empty Array or Hash is assumed to have
874
+ // content so the max_nesting should fail but a non-collection value is
875
+ // okay. That means a check for a collectable value is needed before
876
+ // raising.
899
877
  if (out->opts->dump_opts.max_depth <= depth) {
900
- // When JSON.dump is called then an ArgumentError is expected and the
901
- // limit is the depth inclusive. If JSON.generate is called then a
902
- // NestingError is expected and the limit is inclusive. Worse than
903
- // that there are unit tests for both.
904
- if (CALLER_DUMP == out->caller) {
905
- if (0 < out->argc) {
906
- set_state_depth(*out->argv, depth);
907
- }
908
- rb_raise(rb_eArgError, "Too deeply nested.");
909
- } else if (out->opts->dump_opts.max_depth < depth) {
910
- if (0 < out->argc) {
911
- set_state_depth(*out->argv, depth - 1);
912
- }
913
- raise_json_err("Too deeply nested", "NestingError");
914
- }
878
+ if (RUBY_T_ARRAY == type || RUBY_T_HASH == type) {
879
+ if (0 < out->argc) {
880
+ set_state_depth(*out->argv, depth);
881
+ }
882
+ raise_json_err("Too deeply nested", "NestingError");
883
+ }
915
884
  }
916
885
  if (0 < type && type <= RUBY_T_FIXNUM) {
917
- DumpFunc f = compat_funcs[type];
886
+ DumpFunc f = compat_funcs[type];
918
887
 
919
- if (NULL != f) {
920
- f(obj, depth, out, as_ok);
921
- if (RB_UNLIKELY(Yes == out->opts->trace)) {
922
- oj_trace("dump", obj, __FILE__, __LINE__, depth, TraceOut);
923
- }
924
- return;
925
- }
888
+ if (NULL != f) {
889
+ f(obj, depth, out, as_ok);
890
+ TRACE(out->opts->trace, "dump", obj, depth, TraceOut);
891
+ return;
892
+ }
926
893
  }
927
894
  oj_dump_nil(Qnil, depth, out, false);
928
- if (RB_UNLIKELY(Yes == out->opts->trace)) {
929
- oj_trace("dump", Qnil, __FILE__, __LINE__, depth, TraceOut);
930
- }
895
+ TRACE(out->opts->trace, "dump", Qnil, depth, TraceOut);
931
896
  }