# frozen_string_literal: true require "helper" class TestFilters < JekyllUnitTest class JekyllFilter include Jekyll::Filters attr_accessor :site, :context def initialize(opts = {}) @site = Jekyll::Site.new(opts.merge("skip_config_files" => true)) @context = Liquid::Context.new(@site.site_payload, {}, :site => @site) end end class Value def initialize(value) @value = value end def to_s @value.respond_to?(:call) ? @value.call : @value.to_s end end def make_filter_mock(opts = {}) JekyllFilter.new(site_configuration(opts)).tap do |f| tz = f.site.config["timezone"] Jekyll.set_timezone(tz) if tz end end class SelectDummy def select; end end class KeyValue def initialize(key:, value:) @key = key @val = value end def inspect "{#{@key.inspect}=>#{@val.inspect}}" end end context "filters" do setup do @sample_time = Time.utc(2013, 3, 27, 11, 22, 33) @timezone_before_test = ENV["TZ"] @filter = make_filter_mock( "timezone" => "UTC", "url" => "http://example.com", "baseurl" => "/base", "dont_show_posts_before" => @sample_time ) @sample_date = Date.parse("2013-03-02") @time_as_string = "September 11, 2001 12:46:30 -0000" @time_as_numeric = 1_399_680_607 @integer_as_string = "142857" @array_of_objects = [ { "color" => "teal", "size" => "large" }, { "color" => "red", "size" => "large" }, { "color" => "red", "size" => "medium" }, { "color" => "blue", "size" => "medium" }, ] end teardown do ENV["TZ"] = @timezone_before_test end should "markdownify with simple string" do assert_equal( "
something really simple
\n", @filter.markdownify("something **really** simple") ) end should "markdownify with a number" do assert_equal( "404
\n", @filter.markdownify(404) ) end context "smartify filter" do should "convert quotes and typographic characters" do assert_equal( "SmartyPants is *not* Markdown", @filter.smartify("SmartyPants is *not* Markdown") ) assert_equal( "“This filter’s test…”", @filter.smartify(%q{"This filter's test..."}) ) end should "convert not convert markdown to block HTML elements" do assert_equal( "#hashtag", # NOT "command <filename>
")
)
end
should "not error when xml escaping nil" do
assert_equal "", @filter.xml_escape(nil)
end
should "escape space as plus" do
assert_equal "my+things", @filter.cgi_escape("my things")
end
should "escape special characters" do
assert_equal "hey%21", @filter.cgi_escape("hey!")
end
should "escape space as %20" do
assert_equal "my%20things", @filter.uri_escape("my things")
end
should "allow reserver characters in URI" do
assert_equal(
"foo!*'();:@&=+$,/?#[]bar",
@filter.uri_escape("foo!*'();:@&=+$,/?#[]bar")
)
assert_equal(
"foo%20bar!*'();:@&=+$,/?#[]baz",
@filter.uri_escape("foo bar!*'();:@&=+$,/?#[]baz")
)
end
context "absolute_url filter" do
should "produce an absolute URL from a page URL" do
page_url = "/about/my_favorite_page/"
assert_equal "http://example.com/base#{page_url}", @filter.absolute_url(page_url)
end
should "ensure the leading slash" do
page_url = "about/my_favorite_page/"
assert_equal "http://example.com/base/#{page_url}", @filter.absolute_url(page_url)
end
should "ensure the leading slash for the baseurl" do
page_url = "about/my_favorite_page/"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => "base"
)
assert_equal "http://example.com/base/#{page_url}", filter.absolute_url(page_url)
end
should "be ok with a blank but present 'url'" do
page_url = "about/my_favorite_page/"
filter = make_filter_mock(
"url" => "",
"baseurl" => "base"
)
assert_equal "/base/#{page_url}", filter.absolute_url(page_url)
end
should "be ok with a nil 'url'" do
page_url = "about/my_favorite_page/"
filter = make_filter_mock(
"url" => nil,
"baseurl" => "base"
)
assert_equal "/base/#{page_url}", filter.absolute_url(page_url)
end
should "be ok with a nil 'baseurl'" do
page_url = "about/my_favorite_page/"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => nil
)
assert_equal "http://example.com/#{page_url}", filter.absolute_url(page_url)
end
should "not prepend a forward slash if input is empty" do
page_url = ""
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => "/base"
)
assert_equal "http://example.com/base", filter.absolute_url(page_url)
end
should "not append a forward slash if input is '/'" do
page_url = "/"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => "/base"
)
assert_equal "http://example.com/base/", filter.absolute_url(page_url)
end
should "not append a forward slash if input is '/' and nil 'baseurl'" do
page_url = "/"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => nil
)
assert_equal "http://example.com/", filter.absolute_url(page_url)
end
should "not append a forward slash if both input and baseurl are simply '/'" do
page_url = "/"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => "/"
)
assert_equal "http://example.com/", filter.absolute_url(page_url)
end
should "normalize international URLs" do
page_url = ""
filter = make_filter_mock(
"url" => "http://ümlaut.example.org/",
"baseurl" => nil
)
assert_equal "http://xn--mlaut-jva.example.org/", filter.absolute_url(page_url)
end
should "not modify an absolute URL" do
page_url = "http://example.com/"
assert_equal "http://example.com/", @filter.absolute_url(page_url)
end
should "transform the input URL to a string" do
page_url = "/my-page.html"
filter = make_filter_mock("url" => Value.new(proc { "http://example.org" }))
assert_equal "http://example.org#{page_url}", filter.absolute_url(page_url)
end
should "not raise a TypeError when passed a hash" do
assert @filter.absolute_url("foo" => "bar")
end
context "with a document" do
setup do
@site = fixture_site(
"collections" => ["methods"]
)
@site.process
@document = @site.collections["methods"].docs.detect do |d|
d.relative_path == "_methods/configuration.md"
end
end
should "make a url" do
expected = "http://example.com/base/methods/configuration.html"
assert_equal expected, @filter.absolute_url(@document)
end
end
end
context "relative_url filter" do
should "produce a relative URL from a page URL" do
page_url = "/about/my_favorite_page/"
assert_equal "/base#{page_url}", @filter.relative_url(page_url)
end
should "ensure the leading slash between baseurl and input" do
page_url = "about/my_favorite_page/"
assert_equal "/base/#{page_url}", @filter.relative_url(page_url)
end
should "ensure the leading slash for the baseurl" do
page_url = "about/my_favorite_page/"
filter = make_filter_mock("baseurl" => "base")
assert_equal "/base/#{page_url}", filter.relative_url(page_url)
end
should "normalize international URLs" do
page_url = "错误.html"
assert_equal "/base/%E9%94%99%E8%AF%AF.html", @filter.relative_url(page_url)
end
should "be ok with a nil 'baseurl'" do
page_url = "about/my_favorite_page/"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => nil
)
assert_equal "/#{page_url}", filter.relative_url(page_url)
end
should "not prepend a forward slash if input is empty" do
page_url = ""
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => "/base"
)
assert_equal "/base", filter.relative_url(page_url)
end
should "not prepend a forward slash if baseurl ends with a single '/'" do
page_url = "/css/main.css"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => "/base/"
)
assert_equal "/base/css/main.css", filter.relative_url(page_url)
end
should "not return valid URI if baseurl ends with multiple '/'" do
page_url = "/css/main.css"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => "/base//"
)
refute_equal "/base/css/main.css", filter.relative_url(page_url)
end
should "not prepend a forward slash if both input and baseurl are simply '/'" do
page_url = "/"
filter = make_filter_mock(
"url" => "http://example.com",
"baseurl" => "/"
)
assert_equal "/", filter.relative_url(page_url)
end
should "not return the url by reference" do
filter = make_filter_mock(:baseurl => nil)
page = Page.new(filter.site, test_dir("fixtures"), "", "front_matter.erb")
assert_equal "/front_matter.erb", page.url
url = filter.relative_url(page.url)
url << "foo"
assert_equal "/front_matter.erb", filter.relative_url(page.url)
assert_equal "/front_matter.erb", page.url
end
should "transform the input baseurl to a string" do
page_url = "/my-page.html"
filter = make_filter_mock("baseurl" => Value.new(proc { "/baseurl/" }))
assert_equal "/baseurl#{page_url}", filter.relative_url(page_url)
end
should "transform protocol-relative url" do
url = "//example.com/"
assert_equal "/base//example.com/", @filter.relative_url(url)
end
should "not modify an absolute url with scheme" do
url = "file:///file.html"
assert_equal url, @filter.relative_url(url)
end
should "not normalize absolute international URLs" do
url = "https://example.com/错误"
assert_equal "https://example.com/错误", @filter.relative_url(url)
end
end
context "strip_index filter" do
should "strip trailing /index.html" do
assert_equal "/foo/", @filter.strip_index("/foo/index.html")
end
should "strip trailing /index.htm" do
assert_equal "/foo/", @filter.strip_index("/foo/index.htm")
end
should "not strip HTML in the middle of URLs" do
assert_equal "/index.html/foo", @filter.strip_index("/index.html/foo")
end
should "not raise an error on nil strings" do
assert_nil @filter.strip_index(nil)
end
should "not mangle other URLs" do
assert_equal "/foo/", @filter.strip_index("/foo/")
end
end
context "jsonify filter" do
should "convert hash to json" do
assert_equal "{\"age\":18}", @filter.jsonify(:age => 18)
end
should "convert array to json" do
assert_equal "[1,2]", @filter.jsonify([1, 2])
assert_equal(
"[{\"name\":\"Jack\"},{\"name\":\"Smith\"}]",
@filter.jsonify([{ :name => "Jack" }, { :name => "Smith" }])
)
end
should "convert drop to json" do
@filter.site.read
expected = {
"name" => "2008-02-02-published.markdown",
"path" => "_posts/2008-02-02-published.markdown",
"previous" => nil,
"output" => nil,
"content" => "This should be published.\n",
"id" => "/publish_test/2008/02/02/published",
"url" => "/publish_test/2008/02/02/published.html",
"relative_path" => "_posts/2008-02-02-published.markdown",
"collection" => "posts",
"excerpt" => "This should be published.
\n", "draft" => false, "categories" => [ "publish_test", ], "layout" => "default", "title" => "Publish", "category" => "publish_test", "date" => "2008-02-02 00:00:00 +0000", "slug" => "published", "ext" => ".markdown", "tags" => [], } actual = JSON.parse(@filter.jsonify(@filter.site.docs_to_write.first.to_liquid)) next_doc = actual.delete("next") refute_nil next_doc assert_kind_of Hash, next_doc, "doc.next should be an object" assert_equal expected, actual end should "convert drop with drops to json" do @filter.site.read actual = @filter.jsonify(@filter.site.to_liquid) expected = { "environment" => "development", "version" => Jekyll::VERSION, } assert_equal expected, JSON.parse(actual)["jekyll"] end # rubocop:disable Style/StructInheritance class M < Struct.new(:message) def to_liquid [message] end end class T < Struct.new(:name) def to_liquid { "name" => name, :v => 1, :thing => M.new({:kay => "jewelers"}), :stuff => true, } end end should "call #to_liquid " do expected = [ { "name" => "Jeremiah", "v" => 1, "thing" => [ { "kay" => "jewelers", }, ], "stuff" => true, }, { "name" => "Smathers", "v" => 1, "thing" => [ { "kay" => "jewelers", }, ], "stuff" => true, }, ] result = @filter.jsonify([T.new("Jeremiah"), T.new("Smathers")]) assert_equal expected, JSON.parse(result) end # rubocop:enable Style/StructInheritance should "handle hashes with all sorts of weird keys and values" do my_hash = { "posts" => Array.new(3) { |i| T.new(i) } } expected = { "posts" => [ { "name" => 0, "v" => 1, "thing" => [ { "kay" => "jewelers", }, ], "stuff" => true, }, { "name" => 1, "v" => 1, "thing" => [ { "kay" => "jewelers", }, ], "stuff" => true, }, { "name" => 2, "v" => 1, "thing" => [ { "kay" => "jewelers", }, ], "stuff" => true, }, ], } result = @filter.jsonify(my_hash) assert_equal expected, JSON.parse(result) end end context "group_by filter" do should "successfully group array of Jekyll::Page's" do @filter.site.process grouping = @filter.group_by(@filter.site.pages, "layout") names = ["default", "nil", ""] grouping.each do |g| assert_includes names, g["name"], "#{g["name"]} isn't a valid grouping." case g["name"] when "default" assert_kind_of( Array, g["items"], "The list of grouped items for 'default' is not an Array." ) # adjust array.size to ignore symlinked page in Windows qty = Utils::Platforms.really_windows? ? 4 : 5 assert_equal qty, g["items"].size when "nil" assert_kind_of( Array, g["items"], "The list of grouped items for 'nil' is not an Array." ) assert_equal 2, g["items"].size when "" assert_kind_of( Array, g["items"], "The list of grouped items for '' is not an Array." ) # adjust array.size to ignore symlinked page in Windows qty = Utils::Platforms.really_windows? ? 19 : 21 assert_equal qty, g["items"].size end end end should "include the size of each grouping" do grouping = @filter.group_by(@filter.site.pages, "layout") grouping.each do |g| assert_equal( g["items"].size, g["size"], "The size property for '#{g["name"]}' doesn't match the size of the Array." ) end end should "should pass integers as is" do grouping = @filter.group_by([ { "name" => "Allison", "year" => 2016 }, { "name" => "Amy", "year" => 2016 }, { "name" => "George", "year" => 2019 }, ], "year") assert_equal "2016", grouping[0]["name"] assert_equal "2019", grouping[1]["name"] end end context "where filter" do should "return any input that is not an array" do assert_equal "some string", @filter.where("some string", "la", "le") end should "filter objects in a hash appropriately" do hash = { "a" => { "color"=>"red" }, "b" => { "color"=>"blue" } } assert_equal 1, @filter.where(hash, "color", "red").length assert_equal [{ "color"=>"red" }], @filter.where(hash, "color", "red") end should "filter objects appropriately" do assert_equal 2, @filter.where(@array_of_objects, "color", "red").length end should "filter objects with null properties appropriately" do array = [{}, { "color" => nil }, { "color" => "" }, { "color" => "text" }] assert_equal 2, @filter.where(array, "color", nil).length end should "filter objects with numerical properties appropriately" do array = [ { "value" => "555" }, { "value" => 555 }, { "value" => 24.625 }, { "value" => "24.625" }, ] assert_equal 2, @filter.where(array, "value", 24.625).length assert_equal 2, @filter.where(array, "value", 555).length end should "filter array properties appropriately" do hash = { "a" => { "tags"=>%w(x y) }, "b" => { "tags"=>["x"] }, "c" => { "tags"=>%w(y z) }, } assert_equal 2, @filter.where(hash, "tags", "x").length end should "filter array properties alongside string properties" do hash = { "a" => { "tags"=>%w(x y) }, "b" => { "tags"=>"x" }, "c" => { "tags"=>%w(y z) }, } assert_equal 2, @filter.where(hash, "tags", "x").length end should "filter hash properties with null and empty values" do hash = { "a" => { "tags" => {} }, "b" => { "tags" => "" }, "c" => { "tags" => nil }, "d" => { "tags" => ["x", nil] }, "e" => { "tags" => [] }, "f" => { "tags" => "xtra" }, } assert_equal [{ "tags" => nil }], @filter.where(hash, "tags", nil) assert_equal( [{ "tags" => "" }, { "tags" => ["x", nil] }], @filter.where(hash, "tags", "") ) # `{{ hash | where: 'tags', empty }}` assert_equal( [{ "tags" => {} }, { "tags" => "" }, { "tags" => nil }, { "tags" => [] }], @filter.where(hash, "tags", Liquid::Expression::LITERALS["empty"]) ) # `{{ `hash | where: 'tags', blank }}` assert_equal( [{ "tags" => {} }, { "tags" => "" }, { "tags" => nil }, { "tags" => [] }], @filter.where(hash, "tags", Liquid::Expression::LITERALS["blank"]) ) end should "not match substrings" do hash = { "a" => { "category"=>"bear" }, "b" => { "category"=>"wolf" }, "c" => { "category"=>%w(bear lion) }, } assert_equal 0, @filter.where(hash, "category", "ear").length end should "stringify during comparison for compatibility with liquid parsing" do hash = { "The Words" => { "rating" => 1.2, "featured" => false }, "Limitless" => { "rating" => 9.2, "featured" => true }, "Hustle" => { "rating" => 4.7, "featured" => true }, } results = @filter.where(hash, "featured", "true") assert_equal 2, results.length assert_in_delta(9.2, results[0]["rating"]) assert_in_delta(4.7, results[1]["rating"]) results = @filter.where(hash, "rating", 4.7) assert_equal 1, results.length assert_in_delta(4.7, results[0]["rating"]) end should "always return an array if the object responds to 'select'" do results = @filter.where(SelectDummy.new, "obj", "1 == 1") assert_equal [], results end should "gracefully handle invalid property type" do hash = { "members" => { "name" => %w(John Jane Jimmy) }, "roles" => %w(Admin Recruiter Manager), } err = assert_raises(TypeError) { @filter.where(hash, "name", "Jimmy") } truncatd_arr_str = hash["roles"].to_liquid.to_s[0...20] msg = "Error accessing object (#{truncatd_arr_str}) with given key. Expected an integer " \ 'but got "name" instead.' assert_equal msg, err.message end end context "where_exp filter" do should "return any input that is not an array" do assert_equal "some string", @filter.where_exp("some string", "la", "le") end should "filter objects in a hash appropriately" do hash = { "a" => { "color"=>"red" }, "b" => { "color"=>"blue" } } assert_equal 1, @filter.where_exp(hash, "item", "item.color == 'red'").length assert_equal( [{ "color"=>"red" }], @filter.where_exp(hash, "item", "item.color == 'red'") ) end should "filter objects appropriately" do assert_equal( 2, @filter.where_exp(@array_of_objects, "item", "item.color == 'red'").length ) end should "filter objects appropriately with 'or', 'and' operators" do assert_equal( [ { "color" => "teal", "size" => "large" }, { "color" => "red", "size" => "large" }, { "color" => "red", "size" => "medium" }, ], @filter.where_exp( @array_of_objects, "item", "item.color == 'red' or item.size == 'large'" ) ) assert_equal( [ { "color" => "red", "size" => "large" }, ], @filter.where_exp( @array_of_objects, "item", "item.color == 'red' and item.size == 'large'" ) ) end should "filter objects across multiple conditions" do sample = [ { "color" => "teal", "size" => "large", "type" => "variable" }, { "color" => "red", "size" => "large", "type" => "fixed" }, { "color" => "red", "size" => "medium", "type" => "variable" }, { "color" => "blue", "size" => "medium", "type" => "fixed" }, ] assert_equal( [ { "color" => "red", "size" => "large", "type" => "fixed" }, ], @filter.where_exp( sample, "item", "item.type == 'fixed' and item.color == 'red' or item.color == 'teal'" ) ) end should "stringify during comparison for compatibility with liquid parsing" do hash = { "The Words" => { "rating" => 1.2, "featured" => false }, "Limitless" => { "rating" => 9.2, "featured" => true }, "Hustle" => { "rating" => 4.7, "featured" => true }, } results = @filter.where_exp(hash, "item", "item.featured == true") assert_equal 2, results.length assert_in_delta(9.2, results[0]["rating"]) assert_in_delta(4.7, results[1]["rating"]) results = @filter.where_exp(hash, "item", "item.rating == 4.7") assert_equal 1, results.length assert_in_delta(4.7, results[0]["rating"]) end should "filter with other operators" do assert_equal [3, 4, 5], @filter.where_exp([1, 2, 3, 4, 5], "n", "n >= 3") end objects = [ { "id" => "a", "groups" => [1, 2] }, { "id" => "b", "groups" => [2, 3] }, { "id" => "c" }, { "id" => "d", "groups" => [1, 3] }, ] should "filter with the contains operator over arrays" do results = @filter.where_exp(objects, "obj", "obj.groups contains 1") assert_equal 2, results.length assert_equal "a", results[0]["id"] assert_equal "d", results[1]["id"] end should "filter with the contains operator over hash keys" do results = @filter.where_exp(objects, "obj", "obj contains 'groups'") assert_equal 3, results.length assert_equal "a", results[0]["id"] assert_equal "b", results[1]["id"] assert_equal "d", results[2]["id"] end should "filter posts" do site = fixture_site.tap(&:read) posts = site.site_payload["site"]["posts"] results = @filter.where_exp(posts, "obj", "obj.title == 'Foo Bar'") assert_equal 1, results.length assert_equal site.posts.find { |p| p.title == "Foo Bar" }, results.first end should "always return an array if the object responds to 'select'" do results = @filter.where_exp(SelectDummy.new, "obj", "1 == 1") assert_equal [], results end should "filter by variable values" do @filter.site.tap(&:read) posts = @filter.site.site_payload["site"]["posts"] results = @filter.where_exp(posts, "post", "post.date > site.dont_show_posts_before") assert_equal posts.count { |p| p.date > @sample_time }, results.length end end context "find filter" do should "return any input that is not an array" do assert_equal "some string", @filter.find("some string", "la", "le") end should "filter objects in a hash appropriately" do hash = { "a" => { "color" => "red" }, "b" => { "color" => "blue" } } assert_equal({ "color" => "red" }, @filter.find(hash, "color", "red")) end should "filter objects appropriately" do assert_equal( { "color" => "red", "size" => "large" }, @filter.find(@array_of_objects, "color", "red") ) end should "filter objects with null properties appropriately" do array = [{}, { "color" => nil }, { "color" => "" }, { "color" => "text" }] assert_equal({}, @filter.find(array, "color", nil)) end should "filter objects with numerical properties appropriately" do array = [ { "value" => "555" }, { "value" => 555 }, { "value" => 24.625 }, { "value" => "24.625" }, ] assert_equal({ "value" => 24.625 }, @filter.find(array, "value", 24.625)) assert_equal({ "value" => "555" }, @filter.find(array, "value", 555)) end should "filter array properties appropriately" do hash = { "a" => { "tags" => %w(x y) }, "b" => { "tags" => ["x"] }, "c" => { "tags" => %w(y z) }, } assert_equal({ "tags" => %w(x y) }, @filter.find(hash, "tags", "x")) end should "filter array properties alongside string properties" do hash = { "a" => { "tags" => %w(x y) }, "b" => { "tags" => "x" }, "c" => { "tags" => %w(y z) }, } assert_equal({ "tags" => %w(x y) }, @filter.find(hash, "tags", "x")) end should "filter hash properties with null and empty values" do hash = { "a" => { "tags" => {} }, "b" => { "tags" => "" }, "c" => { "tags" => nil }, "d" => { "tags" => ["x", nil] }, "e" => { "tags" => [] }, "f" => { "tags" => "xtra" }, } assert_equal({ "tags" => nil }, @filter.find(hash, "tags", nil)) assert_equal({ "tags" => "" }, @filter.find(hash, "tags", "")) # `{{ hash | find: 'tags', empty }}` assert_equal( { "tags" => {} }, @filter.find(hash, "tags", Liquid::Expression::LITERALS["empty"]) ) # `{{ `hash | find: 'tags', blank }}` assert_equal( { "tags" => {} }, @filter.find(hash, "tags", Liquid::Expression::LITERALS["blank"]) ) end should "not match substrings" do hash = { "a" => { "category" => "bear" }, "b" => { "category" => "wolf" }, "c" => { "category" => %w(bear lion) }, } assert_nil @filter.find(hash, "category", "ear") end should "stringify during comparison for compatibility with liquid parsing" do hash = { "The Words" => { "rating" => 1.2, "featured" => false }, "Limitless" => { "rating" => 9.2, "featured" => true }, "Hustle" => { "rating" => 4.7, "featured" => true }, } result = @filter.find(hash, "featured", "true") assert_in_delta(9.2, result["rating"]) result = @filter.find(hash, "rating", 4.7) assert_in_delta(4.7, result["rating"]) end end context "find_exp filter" do should "return any input that is not an array" do assert_equal "some string", @filter.find_exp("some string", "la", "le") end should "filter objects in a hash appropriately" do hash = { "a" => { "color"=>"red" }, "b" => { "color"=>"blue" } } assert_equal( { "color" => "red" }, @filter.find_exp(hash, "item", "item.color == 'red'") ) end should "filter objects appropriately" do assert_equal( { "color" => "red", "size" => "large" }, @filter.find_exp(@array_of_objects, "item", "item.color == 'red'") ) end should "filter objects appropriately with 'or', 'and' operators" do assert_equal( { "color" => "teal", "size" => "large" }, @filter.find_exp( @array_of_objects, "item", "item.color == 'red' or item.size == 'large'" ) ) assert_equal( { "color" => "red", "size" => "large" }, @filter.find_exp( @array_of_objects, "item", "item.color == 'red' and item.size == 'large'" ) ) end should "filter objects across multiple conditions" do sample = [ { "color" => "teal", "size" => "large", "type" => "variable" }, { "color" => "red", "size" => "large", "type" => "fixed" }, { "color" => "red", "size" => "medium", "type" => "variable" }, { "color" => "blue", "size" => "medium", "type" => "fixed" }, ] assert_equal( { "color" => "red", "size" => "large", "type" => "fixed" }, @filter.find_exp( sample, "item", "item.type == 'fixed' and item.color == 'red' or item.color == 'teal'" ) ) end should "stringify during comparison for compatibility with liquid parsing" do hash = { "The Words" => { "rating" => 1.2, "featured" => false }, "Limitless" => { "rating" => 9.2, "featured" => true }, "Hustle" => { "rating" => 4.7, "featured" => true }, } result = @filter.find_exp(hash, "item", "item.featured == true") assert_in_delta(9.2, result["rating"]) result = @filter.find_exp(hash, "item", "item.rating == 4.7") assert_in_delta(4.7, result["rating"]) end should "filter with other operators" do assert_equal 3, @filter.find_exp([1, 2, 3, 4, 5], "n", "n >= 3") end objects = [ { "id" => "a", "groups" => [1, 2] }, { "id" => "b", "groups" => [2, 3] }, { "id" => "c" }, { "id" => "d", "groups" => [1, 3] }, ] should "filter with the contains operator over arrays" do result = @filter.find_exp(objects, "obj", "obj.groups contains 1") assert_equal "a", result["id"] end should "filter with the contains operator over hash keys" do result = @filter.find_exp(objects, "obj", "obj contains 'groups'") assert_equal "a", result["id"] end should "filter posts" do site = fixture_site.tap(&:read) posts = site.site_payload["site"]["posts"] result = @filter.find_exp(posts, "obj", "obj.title == 'Foo Bar'") assert_equal(result, site.posts.find { |p| p.title == "Foo Bar" }) end should "filter by variable values" do @filter.site.tap(&:read) posts = @filter.site.site_payload["site"]["posts"] result = @filter.find_exp(posts, "post", "post.date > site.dont_show_posts_before") assert result.date > @sample_time end end context "group_by_exp filter" do should "successfully group array of Jekyll::Page's" do @filter.site.process groups = @filter.group_by_exp(@filter.site.pages, "page", "page.layout | upcase") names = ["DEFAULT", "NIL", ""] groups.each do |g| assert_includes names, g["name"], "#{g["name"]} isn't a valid grouping." case g["name"] when "DEFAULT" assert_kind_of( Array, g["items"], "The list of grouped items for 'default' is not an Array." ) # adjust array.size to ignore symlinked page in Windows qty = Utils::Platforms.really_windows? ? 4 : 5 assert_equal qty, g["items"].size when "nil" assert_kind_of( Array, g["items"], "The list of grouped items for 'nil' is not an Array." ) assert_equal 2, g["items"].size when "" assert_kind_of( Array, g["items"], "The list of grouped items for '' is not an Array." ) # adjust array.size to ignore symlinked page in Windows qty = Utils::Platforms.really_windows? ? 19 : 21 assert_equal qty, g["items"].size end end end should "include the size of each grouping" do groups = @filter.group_by_exp(@filter.site.pages, "page", "page.layout") groups.each do |g| assert_equal( g["items"].size, g["size"], "The size property for '#{g["name"]}' doesn't match the size of the Array." ) end end should "allow more complex filters" do items = [ { "version" => "1.0", "result" => "slow" }, { "version" => "1.1.5", "result" => "medium" }, { "version" => "2.7.3", "result" => "fast" }, ] result = @filter.group_by_exp(items, "item", "item.version | split: '.' | first") assert_equal 2, result.size end should "be equivalent of group_by" do actual = @filter.group_by_exp(@filter.site.pages, "page", "page.layout") expected = @filter.group_by(@filter.site.pages, "layout") assert_equal expected, actual end should "return any input that is not an array" do assert_equal "some string", @filter.group_by_exp("some string", "la", "le") end should "group by full element (as opposed to a field of the element)" do items = %w(a b c d) result = @filter.group_by_exp(items, "item", "item") assert_equal 4, result.length assert_equal ["a"], result.first["items"] end should "accept hashes" do hash = { 1 => "a", 2 => "b", 3 => "c", 4 => "d" } result = @filter.group_by_exp(hash, "item", "item") assert_equal 4, result.length end end context "sort filter" do should "raise Exception when input is nil" do err = assert_raises ArgumentError do @filter.sort(nil) end assert_equal "Cannot sort a null object.", err.message end should "return sorted numbers" do assert_equal [1, 2, 2.2, 3], @filter.sort([3, 2.2, 2, 1]) end should "return sorted strings" do assert_equal %w(10 2), @filter.sort(%w(10 2)) assert_equal %w(FOO Foo foo), @filter.sort(%w(foo Foo FOO)) assert_equal %w(_foo foo foo_), @filter.sort(%w(foo_ _foo foo)) # Cyrillic assert_equal %w(ВУЗ Вуз вуз), @filter.sort(%w(Вуз вуз ВУЗ)) assert_equal %w(_вуз вуз вуз_), @filter.sort(%w(вуз_ _вуз вуз)) # Hebrew assert_equal %w(אלף בית), @filter.sort(%w(בית אלף)) end should "return sorted by property array" do assert_equal [{ "a" => 1 }, { "a" => 2 }, { "a" => 3 }, { "a" => 4 }], @filter.sort([{ "a" => 4 }, { "a" => 3 }, { "a" => 1 }, { "a" => 2 }], "a") end should "return sorted by property array with numeric strings sorted as numbers" do assert_equal([{ "a" => ".5" }, { "a" => "0.65" }, { "a" => "10" }], @filter.sort([{ "a" => "10" }, { "a" => ".5" }, { "a" => "0.65" }], "a")) end should "return sorted by property array with numeric strings first" do assert_equal([{ "a" => ".5" }, { "a" => "0.6" }, { "a" => "twelve" }], @filter.sort([{ "a" => "twelve" }, { "a" => ".5" }, { "a" => "0.6" }], "a")) end should "return sorted by property array with numbers and strings " do assert_equal([{ "a" => "1" }, { "a" => "1abc" }, { "a" => "20" }], @filter.sort([{ "a" => "20" }, { "a" => "1" }, { "a" => "1abc" }], "a")) end should "return sorted by property array with nils first" do ary = [{ "a" => 2 }, { "b" => 1 }, { "a" => 1 }] assert_equal [{ "b" => 1 }, { "a" => 1 }, { "a" => 2 }], @filter.sort(ary, "a") assert_equal @filter.sort(ary, "a"), @filter.sort(ary, "a", "first") end should "return sorted by property array with nils last" do assert_equal [{ "a" => 1 }, { "a" => 2 }, { "b" => 1 }], @filter.sort([{ "a" => 2 }, { "b" => 1 }, { "a" => 1 }], "a", "last") end should "return sorted by subproperty array" do assert_equal [{ "a" => { "b" => 1 } }, { "a" => { "b" => 2 } }, { "a" => { "b" => 3 } },], @filter.sort([{ "a" => { "b" => 2 } }, { "a" => { "b" => 1 } }, { "a" => { "b" => 3 } },], "a.b") end end context "to_integer filter" do should "raise Exception when input is not integer or string" do assert_raises NoMethodError do @filter.to_integer([1, 2]) end end should "return 0 when input is nil" do assert_equal 0, @filter.to_integer(nil) end should "return integer when input is boolean" do assert_equal 0, @filter.to_integer(false) assert_equal 1, @filter.to_integer(true) end should "return integers" do assert_equal 0, @filter.to_integer(0) assert_equal 1, @filter.to_integer(1) assert_equal 1, @filter.to_integer(1.42857) assert_equal(-1, @filter.to_integer(-1)) assert_equal(-1, @filter.to_integer(-1.42857)) end end context "inspect filter" do should "return a HTML-escaped string representation of an object" do hash_like_object = KeyValue.new(:key => "", :value => 1) assert_equal '{""=>1}', hash_like_object.inspect assert_equal "{"<a>"=>1}", @filter.inspect(hash_like_object) end should "quote strings" do assert_equal ""string"", @filter.inspect("string") end end context "slugify filter" do should "return a slugified string" do assert_equal "q-bert-says", @filter.slugify(" Q*bert says @!#?@!") end should "return a slugified string with mode" do assert_equal "q-bert-says-@!-@!", @filter.slugify(" Q*bert says @!#?@!", "pretty") end end context "push filter" do should "return a new array with the element pushed to the end" do assert_equal %w(hi there bernie), @filter.push(%w(hi there), "bernie") end end context "pop filter" do should "return a new array with the last element popped" do assert_equal %w(hi there), @filter.pop(%w(hi there bernie)) end should "allow multiple els to be popped" do assert_equal %w(hi there bert), @filter.pop(%w(hi there bert and ernie), 2) end should "cast string inputs for # into nums" do assert_equal %w(hi there bert), @filter.pop(%w(hi there bert and ernie), "2") end end context "shift filter" do should "return a new array with the element removed from the front" do assert_equal %w(a friendly greeting), @filter.shift(%w(just a friendly greeting)) end should "allow multiple els to be shifted" do assert_equal %w(bert and ernie), @filter.shift(%w(hi there bert and ernie), 2) end should "cast string inputs for # into nums" do assert_equal %w(bert and ernie), @filter.shift(%w(hi there bert and ernie), "2") end end context "unshift filter" do should "return a new array with the element put at the front" do assert_equal %w(aloha there bernie), @filter.unshift(%w(there bernie), "aloha") end end context "sample filter" do should "return a random item from the array" do input = %w(hey there bernie) assert_includes input, @filter.sample(input) end should "allow sampling of multiple values (n > 1)" do input = %w(hey there bernie) @filter.sample(input, 2).each do |val| assert_includes input, val end end end context "number_of_words filter" do should "return the number of words for Latin-only text" do assert_equal 5, @filter.number_of_words("hello world and taoky strong!", "auto") assert_equal 5, @filter.number_of_words("hello world and taoky strong!", "cjk") end should "return the number of characters for CJK-only text" do assert_equal 17, @filter.number_of_words("こんにちは、世界!안녕하세요 세상!", "auto") assert_equal 17, @filter.number_of_words("こんにちは、世界!안녕하세요 세상!", "cjk") end should "process Latin and CJK independently" do # Intentional: No space between Latin and CJK assert_equal 6, @filter.number_of_words("你好hello世界world", "auto") assert_equal 6, @filter.number_of_words("你好hello世界world", "cjk") end should "maintain original behavior unless specified" do assert_equal 1, @filter.number_of_words("你好hello世界world") end end end end