Skip to content

Commit

Permalink
Use Rails 5.2.3 test cases (#558)
Browse files Browse the repository at this point in the history
* Use Rails 5.2.3 test cases

* Re-add code to make test pass

* Fix failing test regarding IO objects

See:
https://github.com/ohler55/oj/pull/558\#discussion_r331182415

* Add test to check if Oj is used for decoding

* Add test to check if Oj is used for encoding
  • Loading branch information
aried3r authored and ohler55 committed Oct 8, 2019
1 parent cd9aae6 commit 9ae66cd
Show file tree
Hide file tree
Showing 9 changed files with 326 additions and 208 deletions.
2 changes: 1 addition & 1 deletion ext/oj/rails.c
Original file line number Diff line number Diff line change
Expand Up @@ -1439,7 +1439,7 @@ static DumpFunc rails_funcs[] = {
dump_hash, // RUBY_T_HASH = 0x08,
dump_obj, // RUBY_T_STRUCT = 0x09,
oj_dump_bignum, // RUBY_T_BIGNUM = 0x0a,
NULL, // RUBY_T_FILE = 0x0b,
dump_as_string, // RUBY_T_FILE = 0x0b,
dump_obj, // RUBY_T_DATA = 0x0c,
NULL, // RUBY_T_MATCH = 0x0d,
// Rails raises a stack error on Complex and Rational. It also corrupts
Expand Down
2 changes: 1 addition & 1 deletion gemfiles/rails_5.gemfile
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
source "https://rubygems.org"

gem "rails", "5.0.2"
gem "rails", "5.2.3"
gem "sqlite3"

gemspec :path => "../"
7 changes: 6 additions & 1 deletion test/activesupport5/Readme.md
Original file line number Diff line number Diff line change
@@ -1 +1,6 @@
Tests copied from [rails/activesupport/test/json/](https://github.com/rails/rails/tree/v5.0.2/activesupport/test/json)
Tests copied from [rails/activesupport/test/json/],
[rails/activesupport/lib/active_support], [rails/activesupport/test].

[rails/activesupport/test/json/]: https://github.com/rails/rails/tree/v5.2.3/activesupport/test/json
[rails/activesupport/lib/active_support]: https://github.com/rails/rails/tree/v5.2.3/activesupport/lib/active_support
[rails/activesupport/test]: https://github.com/rails/rails/tree/v5.2.3/activesupport/test
45 changes: 45 additions & 0 deletions test/activesupport5/abstract_unit.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
# frozen_string_literal: true

ORIG_ARGV = ARGV.dup

require "active_support/core_ext/kernel/reporting"

silence_warnings do
Encoding.default_internal = Encoding::UTF_8
Encoding.default_external = Encoding::UTF_8
end

require "active_support/testing/autorun"
require "active_support/testing/method_call_assertions"

ENV["NO_RELOAD"] = "1"
require "active_support"

Thread.abort_on_exception = true

# Show backtraces for deprecated behavior for quicker cleanup.
ActiveSupport::Deprecation.debug = true

# Default to old to_time behavior but allow running tests with new behavior
ActiveSupport.to_time_preserves_timezone = ENV["PRESERVE_TIMEZONES"] == "1"

# Disable available locale checks to avoid warnings running the test suite.
I18n.enforce_available_locales = false

class ActiveSupport::TestCase
include ActiveSupport::Testing::MethodCallAssertions

# Skips the current run on Rubinius using Minitest::Assertions#skip
private def rubinius_skip(message = "")
skip message if RUBY_ENGINE == "rbx"
end

# Skips the current run on JRuby using Minitest::Assertions#skip
private def jruby_skip(message = "")
skip message if defined?(JRUBY_VERSION)
end

def frozen_error_class
Object.const_defined?(:FrozenError) ? FrozenError : RuntimeError
end
end
128 changes: 68 additions & 60 deletions test/activesupport5/decoding_test.rb
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
require 'activesupport5/test_helper'
require 'active_support/json'
require 'active_support/time'
require 'activesupport5/time_zone_test_helpers'
# frozen_string_literal: true

require_relative "abstract_unit"
require "active_support/json"
require "active_support/time"
require_relative "time_zone_test_helpers"

require 'oj'

Expand All @@ -10,83 +12,90 @@
class TestJSONDecoding < ActiveSupport::TestCase
include TimeZoneTestHelpers

# Added for testing if Oj is used.
test "oj is used as an encoder" do
assert_equal ActiveSupport.json_encoder, Oj::Rails::Encoder
end

class Foo
def self.json_create(object)
"Foo"
end
end

TESTS = {
%q({"returnTo":{"\/categories":"\/"}}) => {"returnTo" => {"/categories" => "/"}},
%q({"return\\"To\\":":{"\/categories":"\/"}}) => {"return\"To\":" => {"/categories" => "/"}},
%q({"returnTo":{"\/categories":1}}) => {"returnTo" => {"/categories" => 1}},
%({"returnTo":[1,"a"]}) => {"returnTo" => [1, "a"]},
%({"returnTo":[1,"\\"a\\",", "b"]}) => {"returnTo" => [1, "\"a\",", "b"]},
%({"a": "'", "b": "5,000"}) => {"a" => "'", "b" => "5,000"},
%({"a": "a's, b's and c's", "b": "5,000"}) => {"a" => "a's, b's and c's", "b" => "5,000"},
%q({"returnTo":{"\/categories":"\/"}}) => { "returnTo" => { "/categories" => "/" } },
%q({"return\\"To\\":":{"\/categories":"\/"}}) => { "return\"To\":" => { "/categories" => "/" } },
%q({"returnTo":{"\/categories":1}}) => { "returnTo" => { "/categories" => 1 } },
%({"returnTo":[1,"a"]}) => { "returnTo" => [1, "a"] },
%({"returnTo":[1,"\\"a\\",", "b"]}) => { "returnTo" => [1, "\"a\",", "b"] },
%({"a": "'", "b": "5,000"}) => { "a" => "'", "b" => "5,000" },
%({"a": "a's, b's and c's", "b": "5,000"}) => { "a" => "a's, b's and c's", "b" => "5,000" },
# multibyte
%({"matzue": "松江", "asakusa": "浅草"}) => {"matzue" => "松江", "asakusa" => "浅草"},
%({"a": "2007-01-01"}) => {'a' => Date.new(2007, 1, 1)},
%({"a": "2007-01-01 01:12:34 Z"}) => {'a' => Time.utc(2007, 1, 1, 1, 12, 34)},
%({"matzue": "松江", "asakusa": "浅草"}) => { "matzue" => "松江", "asakusa" => "浅草" },
%({"a": "2007-01-01"}) => { "a" => Date.new(2007, 1, 1) },
%({"a": "2007-01-01 01:12:34 Z"}) => { "a" => Time.utc(2007, 1, 1, 1, 12, 34) },
%(["2007-01-01 01:12:34 Z"]) => [Time.utc(2007, 1, 1, 1, 12, 34)],
%(["2007-01-01 01:12:34 Z", "2007-01-01 01:12:35 Z"]) => [Time.utc(2007, 1, 1, 1, 12, 34), Time.utc(2007, 1, 1, 1, 12, 35)],
# no time zone
%({"a": "2007-01-01 01:12:34"}) => {'a' => Time.new(2007, 1, 1, 1, 12, 34, "-05:00")},
%({"a": "2007-01-01 01:12:34"}) => { "a" => Time.new(2007, 1, 1, 1, 12, 34, "-05:00") },
# invalid date
%({"a": "1089-10-40"}) => {'a' => "1089-10-40"},
%({"a": "1089-10-40"}) => { "a" => "1089-10-40" },
# xmlschema date notation
%({"a": "2009-08-10T19:01:02"}) => {'a' => Time.new(2009, 8, 10, 19, 1, 2, "-04:00")},
%({"a": "2009-08-10T19:01:02Z"}) => {'a' => Time.utc(2009, 8, 10, 19, 1, 2)},
%({"a": "2009-08-10T19:01:02+02:00"}) => {'a' => Time.utc(2009, 8, 10, 17, 1, 2)},
%({"a": "2009-08-10T19:01:02-05:00"}) => {'a' => Time.utc(2009, 8, 11, 00, 1, 2)},
%({"a": "2009-08-10T19:01:02"}) => { "a" => Time.new(2009, 8, 10, 19, 1, 2, "-04:00") },
%({"a": "2009-08-10T19:01:02Z"}) => { "a" => Time.utc(2009, 8, 10, 19, 1, 2) },
%({"a": "2009-08-10T19:01:02+02:00"}) => { "a" => Time.utc(2009, 8, 10, 17, 1, 2) },
%({"a": "2009-08-10T19:01:02-05:00"}) => { "a" => Time.utc(2009, 8, 11, 00, 1, 2) },
# needs to be *exact*
%({"a": " 2007-01-01 01:12:34 Z "}) => {'a' => " 2007-01-01 01:12:34 Z "},
%({"a": "2007-01-01 : it's your birthday"}) => {'a' => "2007-01-01 : it's your birthday"},
%({"a": " 2007-01-01 01:12:34 Z "}) => { "a" => " 2007-01-01 01:12:34 Z " },
%({"a": "2007-01-01 : it's your birthday"}) => { "a" => "2007-01-01 : it's your birthday" },
%([]) => [],
%({}) => {},
%({"a":1}) => {"a" => 1},
%({"a": ""}) => {"a" => ""},
%({"a":"\\""}) => {"a" => "\""},
%({"a": null}) => {"a" => nil},
%({"a": true}) => {"a" => true},
%({"a": false}) => {"a" => false},
%q({"bad":"\\\\","trailing":""}) => {"bad" => "\\", "trailing" => ""},
%q({"a": "http:\/\/test.host\/posts\/1"}) => {"a" => "http://test.host/posts/1"},
%q({"a": "\u003cunicode\u0020escape\u003e"}) => {"a" => "<unicode escape>"},
%q({"a": "\\\\u0020skip double backslashes"}) => {"a" => "\\u0020skip double backslashes"},
%q({"a": "\u003cbr /\u003e"}) => {'a' => "<br />"},
%q({"b":["\u003ci\u003e","\u003cb\u003e","\u003cu\u003e"]}) => {'b' => ["<i>","<b>","<u>"]},
%({"a":1}) => { "a" => 1 },
%({"a": ""}) => { "a" => "" },
%({"a":"\\""}) => { "a" => "\"" },
%({"a": null}) => { "a" => nil },
%({"a": true}) => { "a" => true },
%({"a": false}) => { "a" => false },
'{"bad":"\\\\","trailing":""}' => { "bad" => "\\", "trailing" => "" },
%q({"a": "http:\/\/test.host\/posts\/1"}) => { "a" => "http://test.host/posts/1" },
%q({"a": "\u003cunicode\u0020escape\u003e"}) => { "a" => "<unicode escape>" },
'{"a": "\\\\u0020skip double backslashes"}' => { "a" => "\\u0020skip double backslashes" },
%q({"a": "\u003cbr /\u003e"}) => { "a" => "<br />" },
%q({"b":["\u003ci\u003e","\u003cb\u003e","\u003cu\u003e"]}) => { "b" => ["<i>", "<b>", "<u>"] },
# test combination of dates and escaped or unicode encoded data in arrays
%q([{"d":"1970-01-01", "s":"\u0020escape"},{"d":"1970-01-01", "s":"\u0020escape"}]) =>
[{'d' => Date.new(1970, 1, 1), 's' => ' escape'},{'d' => Date.new(1970, 1, 1), 's' => ' escape'}],
[{ "d" => Date.new(1970, 1, 1), "s" => " escape" }, { "d" => Date.new(1970, 1, 1), "s" => " escape" }],
%q([{"d":"1970-01-01","s":"http:\/\/example.com"},{"d":"1970-01-01","s":"http:\/\/example.com"}]) =>
[{'d' => Date.new(1970, 1, 1), 's' => 'http://example.com'},
{'d' => Date.new(1970, 1, 1), 's' => 'http://example.com'}],
[{ "d" => Date.new(1970, 1, 1), "s" => "http://example.com" },
{ "d" => Date.new(1970, 1, 1), "s" => "http://example.com" }],
# tests escaping of "\n" char with Yaml backend
%q({"a":"\n"}) => {"a"=>"\n"},
%q({"a":"\u000a"}) => {"a"=>"\n"},
%q({"a":"Line1\u000aLine2"}) => {"a"=>"Line1\nLine2"},
%q({"a":"\n"}) => { "a" => "\n" },
%q({"a":"\u000a"}) => { "a" => "\n" },
%q({"a":"Line1\u000aLine2"}) => { "a" => "Line1\nLine2" },
# prevent json unmarshalling
%q({"json_class":"TestJSONDecoding::Foo"}) => {"json_class"=>"TestJSONDecoding::Foo"},
'{"json_class":"TestJSONDecoding::Foo"}' => { "json_class" => "TestJSONDecoding::Foo" },
# json "fragments" - these are invalid JSON, but ActionPack relies on this
%q("a string") => "a string",
%q(1.1) => 1.1,
%q(1) => 1,
%q(-1) => -1,
%q(true) => true,
%q(false) => false,
%q(null) => nil
'"a string"' => "a string",
"1.1" => 1.1,
"1" => 1,
"-1" => -1,
"true" => true,
"false" => false,
"null" => nil
}

TESTS.each_with_index do |(json, expected), index|
fail_message = "JSON decoding failed for #{json}"

test "json decodes #{index}" do
with_tz_default 'Eastern Time (US & Canada)' do
with_tz_default "Eastern Time (US & Canada)" do
with_parse_json_times(true) do
silence_warnings do
if expected.nil?
assert_nil(ActiveSupport::JSON.decode(json), "JSON failed for #{json}")
assert_nil ActiveSupport::JSON.decode(json), fail_message
else
assert_equal(expected, ActiveSupport::JSON.decode(json), "JSON failed for #{json}")
assert_equal expected, ActiveSupport::JSON.decode(json), fail_message
end
end
end
Expand All @@ -96,7 +105,7 @@ def self.json_create(object)

test "json decodes time json with time parsing disabled" do
with_parse_json_times(false) do
expected = {"a" => "2007-01-01 01:12:34 Z"}
expected = { "a" => "2007-01-01 01:12:34 Z" }
assert_equal expected, ActiveSupport::JSON.decode(%({"a": "2007-01-01 01:12:34 Z"}))
end
end
Expand All @@ -114,12 +123,11 @@ def test_cannot_pass_unsupported_options

private

def with_parse_json_times(value)
old_value = ActiveSupport.parse_json_times
ActiveSupport.parse_json_times = value
yield
ensure
ActiveSupport.parse_json_times = old_value
end
def with_parse_json_times(value)
old_value = ActiveSupport.parse_json_times
ActiveSupport.parse_json_times = value
yield
ensure
ActiveSupport.parse_json_times = old_value
end
end

Loading

0 comments on commit 9ae66cd

Please sign in to comment.