ffi-yajl 2.2.2-universal-java → 2.3.4-universal-java
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.
- checksums.yaml +5 -5
- data/README.md +37 -74
- data/bin/ffi-yajl-bench +7 -7
- data/ext/ffi_yajl/ext/dlopen/extconf.rb +5 -5
- data/ext/ffi_yajl/ext/encoder/encoder.c +4 -0
- data/ext/ffi_yajl/ext/encoder/extconf.rb +14 -14
- data/ext/ffi_yajl/ext/parser/extconf.rb +14 -14
- data/lib/ffi_yajl.rb +7 -7
- data/lib/ffi_yajl/benchmark.rb +5 -5
- data/lib/ffi_yajl/benchmark/encode.rb +8 -8
- data/lib/ffi_yajl/benchmark/encode_json_and_marshal.rb +9 -9
- data/lib/ffi_yajl/benchmark/encode_json_and_yaml.rb +11 -11
- data/lib/ffi_yajl/benchmark/encode_profile.rb +5 -5
- data/lib/ffi_yajl/benchmark/http.rb +12 -12
- data/lib/ffi_yajl/benchmark/parse.rb +8 -8
- data/lib/ffi_yajl/benchmark/parse_json_and_marshal.rb +10 -10
- data/lib/ffi_yajl/benchmark/parse_json_and_yaml.rb +11 -11
- data/lib/ffi_yajl/benchmark/parse_profile.rb +5 -5
- data/lib/ffi_yajl/benchmark/parse_profile_ruby_prof.rb +4 -4
- data/lib/ffi_yajl/benchmark/parse_stream.rb +11 -11
- data/lib/ffi_yajl/encoder.rb +15 -2
- data/lib/ffi_yajl/ext.rb +9 -9
- data/lib/ffi_yajl/ffi.rb +36 -36
- data/lib/ffi_yajl/ffi/encoder.rb +10 -26
- data/lib/ffi_yajl/ffi/parser.rb +14 -13
- data/lib/ffi_yajl/map_library_name.rb +6 -6
- data/lib/ffi_yajl/parser.rb +2 -2
- data/lib/ffi_yajl/version.rb +2 -2
- metadata +15 -21
- data/Rakefile +0 -173
- data/spec/ffi_yajl/encoder_spec.rb +0 -208
- data/spec/ffi_yajl/map_library_name_spec.rb +0 -115
- data/spec/ffi_yajl/parser_spec.rb +0 -569
- data/spec/spec_helper.rb +0 -45
@@ -1,208 +0,0 @@
|
|
1
|
-
# encoding: UTF-8
|
2
|
-
# Copyright (c) 2015 Lamont Granquist
|
3
|
-
# Copyright (c) 2015 Chef Software, Inc.
|
4
|
-
#
|
5
|
-
# Permission is hereby granted, free of charge, to any person obtaining
|
6
|
-
# a copy of this software and associated documentation files (the
|
7
|
-
# "Software"), to deal in the Software without restriction, including
|
8
|
-
# without limitation the rights to use, copy, modify, merge, publish,
|
9
|
-
# distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
-
# permit persons to whom the Software is furnished to do so, subject to
|
11
|
-
# the following conditions:
|
12
|
-
#
|
13
|
-
# The above copyright notice and this permission notice shall be
|
14
|
-
# included in all copies or substantial portions of the Software.
|
15
|
-
#
|
16
|
-
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
-
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
-
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
-
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
-
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
-
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
-
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
23
|
-
|
24
|
-
require 'spec_helper'
|
25
|
-
require 'date'
|
26
|
-
|
27
|
-
describe "FFI_Yajl::Encoder" do
|
28
|
-
let(:options) { {} }
|
29
|
-
|
30
|
-
let(:encoder) { FFI_Yajl::Encoder.new(options) }
|
31
|
-
|
32
|
-
it "encodes hashes in keys as strings", ruby_gte_193: true do
|
33
|
-
ruby = { { 'a' => 'b' } => 2 }
|
34
|
-
expect(encoder.encode(ruby)).to eq('{"{\"a\"=>\"b\"}":2}')
|
35
|
-
end
|
36
|
-
|
37
|
-
it "encodes arrays in keys as strings", ruby_gte_193: true do
|
38
|
-
ruby = { [0, 1] => 2 }
|
39
|
-
expect(encoder.encode(ruby)).to eq('{"[0, 1]":2}')
|
40
|
-
end
|
41
|
-
|
42
|
-
it "encodes nil in keys as strings" do
|
43
|
-
ruby = { nil => 2 }
|
44
|
-
expect(encoder.encode(ruby)).to eq('{"":2}')
|
45
|
-
end
|
46
|
-
|
47
|
-
it "encodes true in keys as strings" do
|
48
|
-
ruby = { true => 2 }
|
49
|
-
expect(encoder.encode(ruby)).to eq('{"true":2}')
|
50
|
-
end
|
51
|
-
|
52
|
-
it "encodes false in keys as strings" do
|
53
|
-
ruby = { false => 2 }
|
54
|
-
expect(encoder.encode(ruby)).to eq('{"false":2}')
|
55
|
-
end
|
56
|
-
|
57
|
-
it "encodes fixnums in keys as strings" do
|
58
|
-
ruby = { 1 => 2 }
|
59
|
-
expect(encoder.encode(ruby)).to eq('{"1":2}')
|
60
|
-
end
|
61
|
-
|
62
|
-
it "encodes floats in keys as strings" do
|
63
|
-
ruby = { 1.1 => 2 }
|
64
|
-
expect(encoder.encode(ruby)).to eq('{"1.1":2}')
|
65
|
-
end
|
66
|
-
|
67
|
-
it "encodes bignums in keys as strings" do
|
68
|
-
ruby = { 12_345_678_901_234_567_890 => 2 }
|
69
|
-
expect(encoder.encode(ruby)).to eq('{"12345678901234567890":2}')
|
70
|
-
end
|
71
|
-
|
72
|
-
it "encodes objects in keys as strings" do
|
73
|
-
o = Object.new
|
74
|
-
ruby = { o => 2 }
|
75
|
-
expect(encoder.encode(ruby)).to eq(%{{"#{o}":2}})
|
76
|
-
end
|
77
|
-
|
78
|
-
it "encodes an object in a key which has a #to_json method as strings" do
|
79
|
-
class Thing
|
80
|
-
def to_json(*a)
|
81
|
-
"{}"
|
82
|
-
end
|
83
|
-
end
|
84
|
-
o = Thing.new
|
85
|
-
ruby = { o => 2 }
|
86
|
-
expect(encoder.encode(ruby)).to eq(%{{"#{o}":2}})
|
87
|
-
end
|
88
|
-
|
89
|
-
# XXX: 127 == YAJL_MAX_DEPTH hardcodedness, zero control for us, it isn't even a twiddleable #define
|
90
|
-
it "raises an exception for deeply nested arrays" do
|
91
|
-
root = []
|
92
|
-
a = root
|
93
|
-
127.times { |_| a << []; a = a[0] }
|
94
|
-
expect { encoder.encode(root) }.to raise_error(FFI_Yajl::EncodeError)
|
95
|
-
end
|
96
|
-
|
97
|
-
it "raises an exception for deeply nested hashes" do
|
98
|
-
root = {}
|
99
|
-
a = root
|
100
|
-
127.times { |_| a["a"] = {}; a = a["a"] }
|
101
|
-
expect { encoder.encode(root) }.to raise_error(FFI_Yajl::EncodeError)
|
102
|
-
end
|
103
|
-
|
104
|
-
it "encodes symbols in keys as strings" do
|
105
|
-
ruby = { thing: 1 }
|
106
|
-
expect(encoder.encode(ruby)).to eq('{"thing":1}')
|
107
|
-
end
|
108
|
-
|
109
|
-
it "encodes symbols in values as strings" do
|
110
|
-
ruby = { "thing" => :one }
|
111
|
-
expect(encoder.encode(ruby)).to eq('{"thing":"one"}')
|
112
|
-
end
|
113
|
-
|
114
|
-
it "can encode 32-bit unsigned ints" do
|
115
|
-
ruby = { "gid" => 4_294_967_294 }
|
116
|
-
expect(encoder.encode(ruby)).to eq('{"gid":4294967294}')
|
117
|
-
end
|
118
|
-
|
119
|
-
context "when the encoder has nil passed in for options" do
|
120
|
-
let(:encoder) { FFI_Yajl::Encoder.new(nil) }
|
121
|
-
|
122
|
-
it "does not throw an exception" do
|
123
|
-
ruby = { "foo" => "bar" }
|
124
|
-
expect(encoder.encode(ruby)).to eq("{\"foo\":\"bar\"}")
|
125
|
-
end
|
126
|
-
end
|
127
|
-
|
128
|
-
it "can encode Date objects" do
|
129
|
-
ruby = Date.parse('2001-02-03')
|
130
|
-
expect(encoder.encode(ruby)).to eq( '"2001-02-03"' )
|
131
|
-
end
|
132
|
-
|
133
|
-
it "can encode StringIOs" do
|
134
|
-
ruby = { "foo" => StringIO.new('THING') }
|
135
|
-
expect(encoder.encode(ruby)).to eq("{\"foo\":\"THING\"}")
|
136
|
-
end
|
137
|
-
|
138
|
-
context "when encoding Time objects in UTC timezone" do
|
139
|
-
before do
|
140
|
-
@saved_tz = ENV['TZ']
|
141
|
-
ENV['TZ'] = 'UTC'
|
142
|
-
end
|
143
|
-
|
144
|
-
after do
|
145
|
-
ENV['TZ'] = @saved_tz
|
146
|
-
end
|
147
|
-
|
148
|
-
it "encodes them correctly" do
|
149
|
-
ruby = Time.local(2001, 02, 02, 21, 05, 06)
|
150
|
-
expect(encoder.encode(ruby)).to eq( '"2001-02-02 21:05:06 +0000"' )
|
151
|
-
end
|
152
|
-
end
|
153
|
-
|
154
|
-
it "can encode DateTime objects" do
|
155
|
-
ruby = DateTime.parse('2001-02-03T04:05:06.1+07:00')
|
156
|
-
expect(encoder.encode(ruby)).to eq( '"2001-02-03T04:05:06+07:00"' )
|
157
|
-
end
|
158
|
-
|
159
|
-
describe "testing .to_json for Objects" do
|
160
|
-
class NoToJson; end
|
161
|
-
class HasToJson
|
162
|
-
def to_json(*args)
|
163
|
-
"{}"
|
164
|
-
end
|
165
|
-
end
|
166
|
-
|
167
|
-
it "calls .to_s for objects without .to_json" do
|
168
|
-
expect(encoder.encode(NoToJson.new)).to match(/^"#<NoToJson:\w+>"$/)
|
169
|
-
end
|
170
|
-
|
171
|
-
it "calls .to_json for objects wit .to_json" do
|
172
|
-
expect(encoder.encode(HasToJson.new)).to eq("{}")
|
173
|
-
end
|
174
|
-
end
|
175
|
-
|
176
|
-
context "when encoding invalid utf-8" do
|
177
|
-
ruby = {
|
178
|
-
"automatic" => {
|
179
|
-
"etc" => {
|
180
|
-
"passwd" => {
|
181
|
-
"root" => { "dir" => "/root", "gid" => 0, "uid" => 0, "shell" => "/bin/sh", "gecos" => "Elan Ruusam\xc3\xa4e" },
|
182
|
-
"glen" => { "dir" => "/home/glen", "gid" => 500, "uid" => 500, "shell" => "/bin/bash", "gecos" => "Elan Ruusam\xE4e" },
|
183
|
-
},
|
184
|
-
},
|
185
|
-
},
|
186
|
-
}
|
187
|
-
|
188
|
-
it "raises an error on invalid json" do
|
189
|
-
expect { encoder.encode(ruby) }.to raise_error(FFI_Yajl::EncodeError, /Invalid UTF-8 string 'Elan Ruusam.e': cannot encode to UTF-8/)
|
190
|
-
end
|
191
|
-
|
192
|
-
context "when validate_utf8 is off" do
|
193
|
-
let(:options) { { validate_utf8: false } }
|
194
|
-
|
195
|
-
it "does not raise an error" do
|
196
|
-
expect { encoder.encode(ruby) }.not_to raise_error
|
197
|
-
end
|
198
|
-
|
199
|
-
it "returns utf8" do
|
200
|
-
expect( encoder.encode(ruby).encoding ).to eq(Encoding::UTF_8)
|
201
|
-
end
|
202
|
-
|
203
|
-
it "returns valid utf8" do
|
204
|
-
expect( encoder.encode(ruby).valid_encoding? ).to be true
|
205
|
-
end
|
206
|
-
end
|
207
|
-
end
|
208
|
-
end
|
@@ -1,115 +0,0 @@
|
|
1
|
-
# Copyright (c) 2015 Lamont Granquist
|
2
|
-
# Copyright (c) 2015 Chef Software, Inc.
|
3
|
-
#
|
4
|
-
# Permission is hereby granted, free of charge, to any person obtaining
|
5
|
-
# a copy of this software and associated documentation files (the
|
6
|
-
# "Software"), to deal in the Software without restriction, including
|
7
|
-
# without limitation the rights to use, copy, modify, merge, publish,
|
8
|
-
# distribute, sublicense, and/or sell copies of the Software, and to
|
9
|
-
# permit persons to whom the Software is furnished to do so, subject to
|
10
|
-
# the following conditions:
|
11
|
-
#
|
12
|
-
# The above copyright notice and this permission notice shall be
|
13
|
-
# included in all copies or substantial portions of the Software.
|
14
|
-
#
|
15
|
-
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
16
|
-
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
17
|
-
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
18
|
-
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
19
|
-
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
20
|
-
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
21
|
-
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
22
|
-
|
23
|
-
require 'spec_helper'
|
24
|
-
|
25
|
-
class Test
|
26
|
-
extend FFI_Yajl::MapLibraryName
|
27
|
-
end
|
28
|
-
|
29
|
-
host_os_library_name_mapping = {
|
30
|
-
"mingw" => [ "libyajl.so", "yajl.dll" ],
|
31
|
-
"mswin" => [ "libyajl.so", "yajl.dll" ],
|
32
|
-
"cygwin" => [ "libyajl.so", "cygyajl.dll" ],
|
33
|
-
"darwin" => [ "libyajl.bundle", "libyajl.dylib" ],
|
34
|
-
"solaris2" => [ "libyajl.so" ],
|
35
|
-
"linux" => [ "libyajl.so" ],
|
36
|
-
"aix" => [ "libyajl.so" ],
|
37
|
-
"hpux" => [ "libyajl.so" ],
|
38
|
-
"netbsd" => [ "libyajl.so" ],
|
39
|
-
"openbsd" => [ "libyajl.so" ],
|
40
|
-
"freebsd" => [ "libyajl.so" ],
|
41
|
-
}
|
42
|
-
|
43
|
-
describe "FFI_Yajl::MapLibraryName" do
|
44
|
-
let(:libyajl2_opt_path) { "/libyajl2/lib" }
|
45
|
-
before do
|
46
|
-
allow(Libyajl2).to receive(:opt_path).and_return(libyajl2_opt_path)
|
47
|
-
end
|
48
|
-
|
49
|
-
host_os_library_name_mapping.each do |host_os, library_names|
|
50
|
-
context "#library_names" do
|
51
|
-
it "maps #{host_os} correctly" do
|
52
|
-
allow(Test).to receive(:host_os).and_return(host_os)
|
53
|
-
expect(Test.send(:library_names)).to eq(library_names)
|
54
|
-
end
|
55
|
-
end
|
56
|
-
|
57
|
-
context "#expanded_library_names" do
|
58
|
-
it "maps #{host_os} correctly" do
|
59
|
-
allow(Test).to receive(:host_os).and_return(host_os)
|
60
|
-
expanded_library_names = []
|
61
|
-
library_names.each do |library_name|
|
62
|
-
path = File.expand_path(File.join(libyajl2_opt_path, library_name))
|
63
|
-
expanded_library_names.push(path)
|
64
|
-
expect(File).to receive(:file?).with(path).and_return(true)
|
65
|
-
end
|
66
|
-
expect(Test.send(:expanded_library_names)).to eq(expanded_library_names)
|
67
|
-
end
|
68
|
-
end
|
69
|
-
|
70
|
-
context "#dlopen_yajl_library" do
|
71
|
-
it "should call dlopen against an expanded library name if it finds it on #{host_os}" do
|
72
|
-
allow(Test).to receive(:host_os).and_return(host_os)
|
73
|
-
library_names.each do |library_name|
|
74
|
-
path = File.expand_path(File.join(libyajl2_opt_path, library_name))
|
75
|
-
allow(File).to receive(:file?).with(path).and_return(true)
|
76
|
-
allow(Test).to receive(:dlopen).with(path).and_return(nil)
|
77
|
-
end
|
78
|
-
Test.send(:dlopen_yajl_library)
|
79
|
-
end
|
80
|
-
it "if dlopen calls all raise it should still use the short names on #{host_os}" do
|
81
|
-
allow(Test).to receive(:host_os).and_return(host_os)
|
82
|
-
library_names.each do |library_name|
|
83
|
-
path = File.expand_path(File.join(libyajl2_opt_path, library_name))
|
84
|
-
allow(File).to receive(:file?).with(path).and_return(true)
|
85
|
-
allow(Test).to receive(:dlopen).with(path).and_raise(ArgumentError)
|
86
|
-
end
|
87
|
-
allow(Test).to receive(:dlopen).with(library_names.first).and_return(nil)
|
88
|
-
Test.send(:dlopen_yajl_library)
|
89
|
-
end
|
90
|
-
end
|
91
|
-
|
92
|
-
context "ffi_open_yajl_library" do
|
93
|
-
it "should call ffi_lib against an expanded library name if it finds it on #{host_os}" do
|
94
|
-
allow(Test).to receive(:host_os).and_return(host_os)
|
95
|
-
library_names.each do |library_name|
|
96
|
-
path = File.expand_path(File.join(libyajl2_opt_path, library_name))
|
97
|
-
allow(File).to receive(:file?).with(path).and_return(true)
|
98
|
-
allow(Test).to receive(:ffi_lib).with(path).and_return(nil)
|
99
|
-
end
|
100
|
-
Test.send(:ffi_open_yajl_library)
|
101
|
-
end
|
102
|
-
|
103
|
-
it "if dlopen calls all raise it should still use 'yajl' on #{host_os}" do
|
104
|
-
allow(Test).to receive(:host_os).and_return(host_os)
|
105
|
-
library_names.each do |library_name|
|
106
|
-
path = File.expand_path(File.join(libyajl2_opt_path, library_name))
|
107
|
-
allow(File).to receive(:file?).with(path).and_return(true)
|
108
|
-
allow(Test).to receive(:ffi_lib).with(path).and_raise(LoadError)
|
109
|
-
end
|
110
|
-
allow(Test).to receive(:ffi_lib).with('yajl').and_return(nil)
|
111
|
-
Test.send(:ffi_open_yajl_library)
|
112
|
-
end
|
113
|
-
end
|
114
|
-
end
|
115
|
-
end
|
@@ -1,569 +0,0 @@
|
|
1
|
-
# encoding: UTF-8
|
2
|
-
# Copyright (c) 2015 Lamont Granquist
|
3
|
-
# Copyright (c) 2015 Chef Software, Inc.
|
4
|
-
#
|
5
|
-
# Permission is hereby granted, free of charge, to any person obtaining
|
6
|
-
# a copy of this software and associated documentation files (the
|
7
|
-
# "Software"), to deal in the Software without restriction, including
|
8
|
-
# without limitation the rights to use, copy, modify, merge, publish,
|
9
|
-
# distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
-
# permit persons to whom the Software is furnished to do so, subject to
|
11
|
-
# the following conditions:
|
12
|
-
#
|
13
|
-
# The above copyright notice and this permission notice shall be
|
14
|
-
# included in all copies or substantial portions of the Software.
|
15
|
-
#
|
16
|
-
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
-
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
-
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
-
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
-
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
-
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
-
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
23
|
-
|
24
|
-
require 'spec_helper'
|
25
|
-
|
26
|
-
describe "FFI_Yajl::Parser" do
|
27
|
-
shared_examples_for "correct json parsing" do
|
28
|
-
context "when json has 23456789012E666" do
|
29
|
-
let(:json) { '{"key": 23456789012E666}' }
|
30
|
-
|
31
|
-
it "should return infinity" do
|
32
|
-
infinity = (1.0 / 0)
|
33
|
-
expect(parser).to eq("key" => infinity)
|
34
|
-
end
|
35
|
-
end
|
36
|
-
|
37
|
-
context "when parsing nil" do
|
38
|
-
let(:json) { nil }
|
39
|
-
it "should not coredump ruby" do
|
40
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
41
|
-
end
|
42
|
-
end
|
43
|
-
|
44
|
-
context "when parsing bare int" do
|
45
|
-
let(:json) { "1" }
|
46
|
-
it "should parse to the int value" do
|
47
|
-
expect( parser ).to eq(1)
|
48
|
-
end
|
49
|
-
end
|
50
|
-
|
51
|
-
context "when parsing bare string" do
|
52
|
-
let(:json) { '"a"' }
|
53
|
-
it "should parse to the string value" do
|
54
|
-
expect( parser ).to eq("a")
|
55
|
-
end
|
56
|
-
end
|
57
|
-
|
58
|
-
context "when parsing bare true" do
|
59
|
-
let(:json) { "true" }
|
60
|
-
it "should parse to the true value" do
|
61
|
-
expect( parser ).to eq(true)
|
62
|
-
end
|
63
|
-
end
|
64
|
-
|
65
|
-
context "when parsing bare false" do
|
66
|
-
let(:json) { "false" }
|
67
|
-
it "should parse to the false value" do
|
68
|
-
expect( parser ).to eq(false)
|
69
|
-
end
|
70
|
-
end
|
71
|
-
|
72
|
-
context "when parsing bare null" do
|
73
|
-
let(:json) { "null" }
|
74
|
-
it "should parse to the nil value" do
|
75
|
-
expect( parser ).to eq(nil)
|
76
|
-
end
|
77
|
-
end
|
78
|
-
|
79
|
-
context "when parsing bare float" do
|
80
|
-
let(:json) { "1.1" }
|
81
|
-
it "should parse to the a float" do
|
82
|
-
expect( parser ).to eq(1.1)
|
83
|
-
end
|
84
|
-
end
|
85
|
-
|
86
|
-
context "when json has comments" do
|
87
|
-
let(:json) { '{"key": /* this is a comment */ "value"}' }
|
88
|
-
|
89
|
-
context "when allow_comments is false" do
|
90
|
-
let(:options) { { allow_comments: false } }
|
91
|
-
|
92
|
-
it "should not parse" do
|
93
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
94
|
-
end
|
95
|
-
end
|
96
|
-
|
97
|
-
context "when allow_comments is true" do
|
98
|
-
let(:options) { { allow_comments: true } }
|
99
|
-
|
100
|
-
it "should parse" do
|
101
|
-
expect(parser).to eq("key" => "value")
|
102
|
-
end
|
103
|
-
end
|
104
|
-
|
105
|
-
context "by default" do
|
106
|
-
let(:options) {}
|
107
|
-
|
108
|
-
it "should parse" do
|
109
|
-
expect(parser).to eq("key" => "value")
|
110
|
-
end
|
111
|
-
end
|
112
|
-
end
|
113
|
-
|
114
|
-
context "when json has multiline comments" do
|
115
|
-
let(:json) { %{{"key": \n/*\n this is a multiline comment \n*/\n "value"}} }
|
116
|
-
|
117
|
-
context "when allow_comments is false" do
|
118
|
-
let(:options) { { allow_comments: false } }
|
119
|
-
|
120
|
-
it "should not parse" do
|
121
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
122
|
-
end
|
123
|
-
end
|
124
|
-
|
125
|
-
context "when allow_comments is true" do
|
126
|
-
let(:options) { { allow_comments: true } }
|
127
|
-
|
128
|
-
it "should parse" do
|
129
|
-
expect(parser).to eq("key" => "value")
|
130
|
-
end
|
131
|
-
end
|
132
|
-
end
|
133
|
-
|
134
|
-
context "when json has inline comments" do
|
135
|
-
let(:json) { %{{"key": \n// this is an inline comment\n "value"}} }
|
136
|
-
|
137
|
-
context "when allow_comments is false" do
|
138
|
-
let(:options) { { allow_comments: false } }
|
139
|
-
|
140
|
-
it "should not parse" do
|
141
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
142
|
-
end
|
143
|
-
end
|
144
|
-
|
145
|
-
context "when allow_comments is true" do
|
146
|
-
let(:options) { { allow_comments: true } }
|
147
|
-
|
148
|
-
it "should parse" do
|
149
|
-
expect(parser).to eq("key" => "value")
|
150
|
-
end
|
151
|
-
end
|
152
|
-
end
|
153
|
-
|
154
|
-
context "when json is invalid UTF8" do
|
155
|
-
let(:json) { "[\"\201\203\"]" }
|
156
|
-
|
157
|
-
it "should not parse by default" do
|
158
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
159
|
-
end
|
160
|
-
|
161
|
-
context "when :dont_validate_strings is set to true" do
|
162
|
-
let(:options) { { dont_validate_strings: true } }
|
163
|
-
|
164
|
-
it "should parse" do
|
165
|
-
expect(parser).to eq(["\x81\x83"])
|
166
|
-
end
|
167
|
-
end
|
168
|
-
|
169
|
-
context "when :dont_validate_strings is set to false" do
|
170
|
-
let(:options) { { dont_validate_strings: false } }
|
171
|
-
|
172
|
-
it "should not parse" do
|
173
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
174
|
-
end
|
175
|
-
end
|
176
|
-
|
177
|
-
context "when :check_utf8 is set to true" do
|
178
|
-
let(:options) { { check_utf8: true } }
|
179
|
-
|
180
|
-
it "should not parse" do
|
181
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
182
|
-
end
|
183
|
-
|
184
|
-
context "when :dont_validate_strings is set to true" do
|
185
|
-
let(:options) { { check_utf8: true, dont_validate_strings: true } }
|
186
|
-
|
187
|
-
it "should raise an ArgumentError" do
|
188
|
-
expect { parser }.to raise_error(ArgumentError)
|
189
|
-
end
|
190
|
-
end
|
191
|
-
|
192
|
-
context "when :dont_validate_strings is set to false" do
|
193
|
-
let(:options) { { check_utf8: true, dont_validate_strings: false } }
|
194
|
-
|
195
|
-
it "should not parse" do
|
196
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
197
|
-
end
|
198
|
-
end
|
199
|
-
end
|
200
|
-
|
201
|
-
context "when :check_utf8 is set to false" do
|
202
|
-
let(:options) { { check_utf8: false } }
|
203
|
-
|
204
|
-
it "should parse" do
|
205
|
-
expect(parser).to eq(["\x81\x83"])
|
206
|
-
end
|
207
|
-
|
208
|
-
context "when :dont_validate_strings is set to true" do
|
209
|
-
let(:options) { { check_utf8: false, dont_validate_strings: true } }
|
210
|
-
|
211
|
-
it "should parse" do
|
212
|
-
expect(parser).to eq(["\x81\x83"])
|
213
|
-
end
|
214
|
-
end
|
215
|
-
|
216
|
-
context "when :dont_validate_strings is set to false" do
|
217
|
-
let(:options) { { check_utf8: false, dont_validate_strings: false } }
|
218
|
-
|
219
|
-
it "should raise an ArgumentError" do
|
220
|
-
expect { parser }.to raise_error(ArgumentError)
|
221
|
-
end
|
222
|
-
end
|
223
|
-
end
|
224
|
-
end
|
225
|
-
|
226
|
-
context "when JSON is a StringIO" do
|
227
|
-
let(:json) { StringIO.new('{"key": 1234}') }
|
228
|
-
|
229
|
-
it "should parse" do
|
230
|
-
expect(parser).to eq("key" => 1234)
|
231
|
-
end
|
232
|
-
end
|
233
|
-
|
234
|
-
context "when parsing a JSON string" do
|
235
|
-
let(:json) { '{"key": 1234}' }
|
236
|
-
|
237
|
-
it "should parse correctly" do
|
238
|
-
expect(parser).to eq("key" => 1234)
|
239
|
-
end
|
240
|
-
|
241
|
-
context "when symbolize_keys is true" do
|
242
|
-
let(:options) { { symbolize_keys: true } }
|
243
|
-
|
244
|
-
it "should symbolize keys correctly" do
|
245
|
-
expect(parser).to eq(key: 1234)
|
246
|
-
end
|
247
|
-
end
|
248
|
-
|
249
|
-
context "when passing a block" do
|
250
|
-
it "should parse correctly" do
|
251
|
-
skip "handle blocks"
|
252
|
-
output = nil
|
253
|
-
parser do |obj|
|
254
|
-
output = obj
|
255
|
-
end
|
256
|
-
expect(output).to eq("key" => 1234)
|
257
|
-
end
|
258
|
-
end
|
259
|
-
end
|
260
|
-
|
261
|
-
context "when parsing a JSON hash with only strings" do
|
262
|
-
let(:json) { '{"key": "value"}' }
|
263
|
-
|
264
|
-
if RUBY_VERSION.to_f >= 1.9
|
265
|
-
context "when Encoding.default_internal is nil" do
|
266
|
-
before do
|
267
|
-
@saved_encoding = Encoding.default_internal
|
268
|
-
Encoding.default_internal = nil
|
269
|
-
end
|
270
|
-
after do
|
271
|
-
Encoding.default_internal = @saved_encoding
|
272
|
-
end
|
273
|
-
it "encodes keys to UTF-8" do
|
274
|
-
expect(parser.keys.first.encoding).to eql(Encoding.find('utf-8'))
|
275
|
-
end
|
276
|
-
it "encodes values to UTF-8" do
|
277
|
-
expect(parser.values.first.encoding).to eql(Encoding.find('utf-8'))
|
278
|
-
end
|
279
|
-
end
|
280
|
-
|
281
|
-
%w{utf-8 us-ascii}.each do |encoding|
|
282
|
-
context "when Encoding.default_internal is #{encoding}" do
|
283
|
-
before do
|
284
|
-
@saved_encoding = Encoding.default_internal
|
285
|
-
Encoding.default_internal = nil
|
286
|
-
end
|
287
|
-
after do
|
288
|
-
Encoding.default_internal = @saved_encoding
|
289
|
-
end
|
290
|
-
it "encodes keys to #{encoding}" do
|
291
|
-
skip "fix us-ascii" if encoding == "us-ascii"
|
292
|
-
expect(parser.keys.first.encoding).to eql(Encoding.find(encoding))
|
293
|
-
end
|
294
|
-
it "encodes values to #{encoding}" do
|
295
|
-
skip "fix us-ascii" if encoding == "us-ascii"
|
296
|
-
expect(parser.values.first.encoding).to eql(Encoding.find(encoding))
|
297
|
-
end
|
298
|
-
end
|
299
|
-
end
|
300
|
-
end
|
301
|
-
end
|
302
|
-
|
303
|
-
context "when a parsed key has utf-8 multibyte characters" do
|
304
|
-
let(:json) { '{"日本語": 1234}' }
|
305
|
-
|
306
|
-
it "should parse correctly" do
|
307
|
-
expect(parser).to eq("日本語" => 1234)
|
308
|
-
end
|
309
|
-
|
310
|
-
context "when symbolize_keys is true" do
|
311
|
-
let(:options) { { symbolize_keys: true } }
|
312
|
-
|
313
|
-
it "should symbolize keys correctly" do
|
314
|
-
expect(parser).to eq(:"日本語" => 1234) # rubocop:disable Style/HashSyntax
|
315
|
-
end
|
316
|
-
|
317
|
-
if RUBY_VERSION.to_f >= 1.9
|
318
|
-
it "should parse non-ascii symbols in UTF-8" do
|
319
|
-
expect(parser.keys.fetch(0).encoding).to eq(Encoding::UTF_8)
|
320
|
-
end
|
321
|
-
end
|
322
|
-
end
|
323
|
-
end
|
324
|
-
|
325
|
-
context "when parsing 2147483649" do
|
326
|
-
let(:json) { "{\"id\": 2147483649}" }
|
327
|
-
|
328
|
-
it "should parse corectly" do
|
329
|
-
expect(parser).to eql("id" => 2_147_483_649)
|
330
|
-
end
|
331
|
-
end
|
332
|
-
|
333
|
-
context "when parsing 5687389800" do
|
334
|
-
let(:json) { "{\"id\": 5687389800}" }
|
335
|
-
|
336
|
-
it "should parse corectly" do
|
337
|
-
expect(parser).to eql("id" => 5_687_389_800)
|
338
|
-
end
|
339
|
-
end
|
340
|
-
|
341
|
-
context "when parsing 1046289770033519442869495707521600000000" do
|
342
|
-
let(:json) { "{\"id\": 1046289770033519442869495707521600000000}" }
|
343
|
-
|
344
|
-
it "should parse corectly" do
|
345
|
-
expect(parser).to eql("id" => 1_046_289_770_033_519_442_869_495_707_521_600_000_000)
|
346
|
-
end
|
347
|
-
end
|
348
|
-
|
349
|
-
# NOTE: we are choosing to be compatible with yajl-ruby here vs. JSON
|
350
|
-
# gem and libyajl C behavior (which is to throw an exception in this case)
|
351
|
-
context "when the JSON is empty string" do
|
352
|
-
let(:json) { '' }
|
353
|
-
|
354
|
-
it "returns nil" do
|
355
|
-
expect(parser).to be_nil
|
356
|
-
end
|
357
|
-
end
|
358
|
-
|
359
|
-
# NOTE: this fixes yajl-ruby being too permissive
|
360
|
-
context "when dealing with too much or too little input" do
|
361
|
-
context "when trailing braces are missing" do
|
362
|
-
let(:json) { '{"foo":{"foo": 1234}' }
|
363
|
-
|
364
|
-
it "raises an exception" do
|
365
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
366
|
-
end
|
367
|
-
end
|
368
|
-
|
369
|
-
context "when trailing brackets are missing" do
|
370
|
-
let(:json) { '[["foo", "bar"]' }
|
371
|
-
|
372
|
-
it "raises an exception" do
|
373
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
374
|
-
end
|
375
|
-
end
|
376
|
-
|
377
|
-
context "when an extra brace is present" do
|
378
|
-
let(:json) { '{"foo":{"foo": 1234}}}' }
|
379
|
-
|
380
|
-
it "raises an exception" do
|
381
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
382
|
-
end
|
383
|
-
|
384
|
-
context "with allow_trailing_garbage" do
|
385
|
-
let(:options) { { allow_trailing_garbage: true } }
|
386
|
-
it "parses" do
|
387
|
-
expect(parser).to eq("foo" => { "foo" => 1234 })
|
388
|
-
end
|
389
|
-
end
|
390
|
-
end
|
391
|
-
|
392
|
-
context "when an extra bracket is present" do
|
393
|
-
let(:json) { '[["foo", "bar"]]]' }
|
394
|
-
|
395
|
-
it "raises an exception" do
|
396
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
397
|
-
end
|
398
|
-
end
|
399
|
-
end
|
400
|
-
|
401
|
-
context "when parsing heavy metal umlauts in keys" do
|
402
|
-
let(:json) { '{"München": "Bayern"}' }
|
403
|
-
|
404
|
-
it "correctly parses" do
|
405
|
-
expect(parser).to eql( "München" => "Bayern" )
|
406
|
-
end
|
407
|
-
end
|
408
|
-
|
409
|
-
context "when parsing floats" do
|
410
|
-
context "parses simple floating point values" do
|
411
|
-
let(:json) { '{"foo": 3.14159265358979}' }
|
412
|
-
|
413
|
-
it "correctly parses" do
|
414
|
-
expect(parser).to eql( "foo" => 3.14159265358979 )
|
415
|
-
end
|
416
|
-
end
|
417
|
-
|
418
|
-
context "parses simple negative floating point values" do
|
419
|
-
let(:json) { '{"foo":-2.00231930436153}' }
|
420
|
-
|
421
|
-
it "correctly parses" do
|
422
|
-
expect(parser).to eql( "foo" => -2.00231930436153 )
|
423
|
-
end
|
424
|
-
end
|
425
|
-
|
426
|
-
context "parses floats with negative exponents and a large E" do
|
427
|
-
let(:json) { '{"foo": 1.602176565E-19}' }
|
428
|
-
|
429
|
-
it "correctly parses" do
|
430
|
-
expect(parser).to eql( "foo" => 1.602176565e-19 )
|
431
|
-
end
|
432
|
-
end
|
433
|
-
|
434
|
-
context "parses floats with negative exponents and a small e" do
|
435
|
-
let(:json) { '{"foo": 6.6260689633e-34 }' }
|
436
|
-
|
437
|
-
it "correctly parses" do
|
438
|
-
expect(parser).to eql( "foo" => 6.6260689633e-34 )
|
439
|
-
end
|
440
|
-
end
|
441
|
-
|
442
|
-
context "parses floats with positive exponents and a large E" do
|
443
|
-
let(:json) { '{"foo": 6.0221413E+23}' }
|
444
|
-
|
445
|
-
it "correctly parses" do
|
446
|
-
expect(parser).to eql( "foo" => 6.0221413e+23 )
|
447
|
-
end
|
448
|
-
end
|
449
|
-
|
450
|
-
context "parses floats with positive exponents and a small e" do
|
451
|
-
let(:json) { '{"foo": 8.9875517873681764e+9 }' }
|
452
|
-
|
453
|
-
it "correctly parses" do
|
454
|
-
expect(parser).to eql( "foo" => 8.9875517873681764e+9 )
|
455
|
-
end
|
456
|
-
end
|
457
|
-
|
458
|
-
context "parses floats with an exponent without a sign and a large E" do
|
459
|
-
let(:json) { '{"foo": 2.99792458E8 }' }
|
460
|
-
|
461
|
-
it "correctly parses" do
|
462
|
-
expect(parser).to eql( "foo" => 2.99792458e+8 )
|
463
|
-
end
|
464
|
-
end
|
465
|
-
|
466
|
-
context "parses floats with an exponent without a sign and a small e" do
|
467
|
-
let(:json) { '{"foo": 1.0973731568539e7 }' }
|
468
|
-
|
469
|
-
it "correctly parses" do
|
470
|
-
expect(parser).to eql( "foo" => 1.0973731568539e+7 )
|
471
|
-
end
|
472
|
-
end
|
473
|
-
end
|
474
|
-
|
475
|
-
# NOTE: parsing floats with 8 million digits on windows has some kind of huge
|
476
|
-
# perf issues likely in ruby and/or the underlying windows libs
|
477
|
-
context "when parsing big floats", ruby_gte_193: true, unix_only: true do
|
478
|
-
let(:json) { '[0.' + '1' * 2**23 + ']' }
|
479
|
-
|
480
|
-
it "parses" do
|
481
|
-
expect { parser }.not_to raise_error
|
482
|
-
end
|
483
|
-
end
|
484
|
-
|
485
|
-
context "when parsing long hash keys with symbolize_keys option", ruby_gte_193: true do
|
486
|
-
let(:json) { '{"' + 'a' * 2**23 + '": 0}' }
|
487
|
-
let(:options) { { symbolize_keys: true } }
|
488
|
-
|
489
|
-
it "parses" do
|
490
|
-
expect { parser }.not_to raise_error
|
491
|
-
end
|
492
|
-
end
|
493
|
-
|
494
|
-
context "should ignore repeated keys by default" do
|
495
|
-
let(:json) { '{"foo":"bar","foo":"baz"}' }
|
496
|
-
it "should replace the first hash key with the second" do
|
497
|
-
expect(parser).to eql( "foo" => "baz" )
|
498
|
-
end
|
499
|
-
end
|
500
|
-
|
501
|
-
context "should raise an exception for repeated keys" do
|
502
|
-
let(:json) { '{"foo":"bar","foo":"baz"}' }
|
503
|
-
let(:options) { { unique_key_checking: true } }
|
504
|
-
it "should raise" do
|
505
|
-
expect { parser }.to raise_error(FFI_Yajl::ParseError)
|
506
|
-
end
|
507
|
-
end
|
508
|
-
end
|
509
|
-
|
510
|
-
context "when options are set to empty hash" do
|
511
|
-
let(:options) { {} }
|
512
|
-
|
513
|
-
context "when using a parsing object" do
|
514
|
-
let(:parser) { FFI_Yajl::Parser.new(options).parse(json) }
|
515
|
-
|
516
|
-
it_behaves_like "correct json parsing"
|
517
|
-
end
|
518
|
-
|
519
|
-
context "when using the class method" do
|
520
|
-
let(:parser) { FFI_Yajl::Parser.parse(json, options) }
|
521
|
-
|
522
|
-
it_behaves_like "correct json parsing"
|
523
|
-
end
|
524
|
-
end
|
525
|
-
|
526
|
-
context "when options are set to nil" do
|
527
|
-
let(:options) { nil }
|
528
|
-
|
529
|
-
context "when using a parsing object" do
|
530
|
-
let(:parser) { FFI_Yajl::Parser.new(options).parse(json) }
|
531
|
-
|
532
|
-
it_behaves_like "correct json parsing"
|
533
|
-
end
|
534
|
-
|
535
|
-
context "when using the class method" do
|
536
|
-
let(:parser) { FFI_Yajl::Parser.parse(json, options) }
|
537
|
-
|
538
|
-
it_behaves_like "correct json parsing"
|
539
|
-
end
|
540
|
-
end
|
541
|
-
|
542
|
-
context "when options default to nothing" do
|
543
|
-
let(:options) { nil }
|
544
|
-
|
545
|
-
context "when using a parsing object" do
|
546
|
-
let(:parser) do
|
547
|
-
if options.nil?
|
548
|
-
FFI_Yajl::Parser.new.parse(json)
|
549
|
-
else
|
550
|
-
FFI_Yajl::Parser.new(options).parse(json)
|
551
|
-
end
|
552
|
-
end
|
553
|
-
|
554
|
-
it_behaves_like "correct json parsing"
|
555
|
-
end
|
556
|
-
|
557
|
-
context "when using the class method" do
|
558
|
-
let(:parser) do
|
559
|
-
if options.nil?
|
560
|
-
FFI_Yajl::Parser.parse(json)
|
561
|
-
else
|
562
|
-
FFI_Yajl::Parser.parse(json, options)
|
563
|
-
end
|
564
|
-
end
|
565
|
-
|
566
|
-
it_behaves_like "correct json parsing"
|
567
|
-
end
|
568
|
-
end
|
569
|
-
end
|