require File.expand_path(File.dirname(__FILE__) + '/test_helper')
class RadiusParserTest < Minitest::Test
include RadiusTestHelper
def setup
@context = new_context
@parser = Radius::Parser.new(@context, :tag_prefix => 'r')
end
def test_initialize
@parser = Radius::Parser.new
assert_kind_of Radius::Context, @parser.context
end
def test_initialize_with_params
@parser = Radius::Parser.new(TestContext.new)
assert_kind_of TestContext, @parser.context
@parser = Radius::Parser.new(:context => TestContext.new)
assert_kind_of TestContext, @parser.context
@parser = Radius::Parser.new('context' => TestContext.new)
assert_kind_of TestContext, @parser.context
@parser = Radius::Parser.new(:tag_prefix => 'r')
assert_kind_of Radius::Context, @parser.context
assert_equal 'r', @parser.tag_prefix
@parser = Radius::Parser.new(TestContext.new, :tag_prefix => 'r')
assert_kind_of TestContext, @parser.context
assert_equal 'r', @parser.tag_prefix
end
def test_parse_tag_with_dashes
define_tag 'some-tag' do |tag|
'ok'
end
assert_parse_output 'ok', 'nope'
end
def test_parse_tag_with_dashed_attributes
define_tag 'tag-with-dashed-attributes' do |tag|
"dashed: #{tag.attr['data-dashed']} regular: #{tag.attr['regular']}"
end
assert_parse_output 'dashed: dashed-value regular: value', ''
end
def test_parse_individual_tags_and_parameters
define_tag "add" do |tag|
tag.attr["param1"].to_i + tag.attr["param2"].to_i
end
assert_parse_output "<3>", %{<>}
end
def test_parse_attributes
attributes = %{{"a"=>"1", "b"=>"2", "c"=>"3", "d"=>"'"}}
assert_parse_output attributes, %{}
assert_parse_output attributes, %{}
end
def test_parse_attributes_with_slashes_or_angle_brackets
slash = %{{"slash"=>"/"}}
angle = %{{"angle"=>">"}}
assert_parse_output slash, %{}
assert_parse_output slash, %{}
assert_parse_output angle, %{}
end
def test_parse_quotes
assert_parse_output "test []", %{ }
end
def test_things_that_should_be_left_alone
[
%{ test="2"="4" },
%{="2" }
].each do |middle|
assert_parsed_is_unchanged ""
assert_parsed_is_unchanged ""
end
end
def test_tags_inside_html_tags
assert_parse_output %{tags in yo tags
},
%{tags in yo tags
}
end
def test_parse_result_is_always_a_string
define_tag("twelve") { 12 }
assert_parse_output "12", ""
end
def test_parse_double_tags
assert_parse_output "test".reverse, "test"
assert_parse_output "tset TEST", "test test"
end
def test_parse_tag_nesting
define_tag("parent", :for => '')
define_tag("parent:child", :for => '')
define_tag("extra", :for => '')
define_tag("nesting") { |tag| tag.nesting }
define_tag("extra:nesting") { |tag| tag.nesting.gsub(':', ' > ') }
define_tag("parent:child:nesting") { |tag| tag.nesting.gsub(':', ' * ') }
assert_parse_output "nesting", ""
assert_parse_output "parent:nesting", ""
assert_parse_output "extra > nesting", ""
assert_parse_output "parent * child * nesting", ""
assert_parse_output "parent > extra > nesting", ""
assert_parse_output "parent > child > extra > nesting", ""
assert_parse_output "parent * extra * child * nesting", ""
assert_parse_output "parent > extra > child > extra > nesting", ""
assert_parse_output "parent > extra > child > extra > nesting", ""
assert_parse_output "extra * parent * child * nesting", ""
assert_parse_output "extra > parent > nesting", ""
assert_parse_output "extra * parent * child * nesting", ""
assert_raises(Radius::UndefinedTagError) { @parser.parse("") }
end
def test_parse_tag_nesting_2
define_tag("parent", :for => '')
define_tag("parent:child", :for => '')
define_tag("content") { |tag| tag.nesting }
assert_parse_output 'parent:child:content', ''
end
def test_parse_tag__binding_do_missing
define_tag 'test' do |tag|
tag.missing!
end
e = assert_raises(Radius::UndefinedTagError) { @parser.parse("") }
assert_equal "undefined tag `test'", e.message
end
def test_parse_chirpy_bird
# :> chirp chirp
assert_parse_output "<:", "<:"
end
def test_parse_tag__binding_render_tag
define_tag('test') { |tag| "Hello #{tag.attr['name']}!" }
define_tag('hello') { |tag| tag.render('test', tag.attr) }
assert_parse_output 'Hello John!', ''
end
def test_accessing_tag_attributes_through_tag_indexer
define_tag('test') { |tag| "Hello #{tag['name']}!" }
assert_parse_output 'Hello John!', ''
end
def test_parse_tag__binding_render_tag_with_block
define_tag('test') { |tag| "Hello #{tag.expand}!" }
define_tag('hello') { |tag| tag.render('test') { tag.expand } }
assert_parse_output 'Hello John!', 'John'
end
def test_tag_locals
define_tag "outer" do |tag|
tag.locals.var = 'outer'
tag.expand
end
define_tag "outer:inner" do |tag|
tag.locals.var = 'inner'
tag.expand
end
define_tag "outer:var" do |tag|
tag.locals.var
end
assert_parse_output 'outer', ""
assert_parse_output 'outer:inner:outer', "::"
assert_parse_output 'outer:inner:outer:inner:outer', "::::"
assert_parse_output 'outer', ""
end
def test_tag_globals
define_tag "set" do |tag|
tag.globals.var = tag.attr['value']
''
end
define_tag "var" do |tag|
tag.globals.var
end
assert_parse_output " true false", %{ }
end
def test_parse_loops
@item = nil
define_tag "each" do |tag|
result = []
["Larry", "Moe", "Curly"].each do |item|
tag.locals.item = item
result << tag.expand
end
result.join(tag.attr["between"] || "")
end
define_tag "each:item" do |tag|
tag.locals.item
end
assert_parse_output %{Three Stooges: "Larry", "Moe", "Curly"}, %{Three Stooges: ""}
end
def test_parse_speed
define_tag "set" do |tag|
tag.globals.var = tag.attr['value']
''
end
define_tag "var" do |tag|
tag.globals.var
end
parts = %w{decima nobis augue at facer processus commodo legentis odio lectorum dolore nulla esse lius qui nonummy ullamcorper erat ii notare}
multiplier = parts.map{|p| "#{p}=\"#{rand}\""}.join(' ')
assert ->{
Timeout.timeout(10) do
assert_parse_output " false", %{ }
end
}.call
end
def test_tag_option_for
define_tag 'fun', :for => 'just for kicks'
assert_parse_output 'just for kicks', ''
end
def test_tag_expose_option
define_tag 'user', :for => users.first, :expose => ['name', :age]
assert_parse_output 'John', ''
assert_parse_output '25', ''
e = assert_raises(Radius::UndefinedTagError) { @parser.parse "" }
assert_equal "undefined tag `email'", e.message
end
def test_tag_expose_attributes_option_on_by_default
define_tag 'user', :for => user_with_attributes
assert_parse_output 'John', ''
end
def test_tag_expose_attributes_set_to_false
define_tag 'user_without_attributes', :for => user_with_attributes, :attributes => false
assert_raises(Radius::UndefinedTagError) { @parser.parse "" }
end
def test_tag_options_must_contain_a_for_option_if_methods_are_exposed
e = assert_raises(ArgumentError) { define_tag('fun', :expose => :today) { 'test' } }
assert_equal "tag definition must contain a :for option when used with the :expose option", e.message
end
def test_parse_fail_on_missing_end_tag
assert_raises(Radius::MissingEndTagError) { @parser.parse("") }
end
def test_parse_fail_on_wrong_end_tag
assert_raises(Radius::WrongEndTagError) { @parser.parse("") }
end
def test_parse_with_default_tag_prefix
@parser = Radius::Parser.new(@context)
define_tag("hello") { |tag| "Hello world!" }
assert_equal "Hello world!
", @parser.parse('
')
end
def test_parse_with_other_radius_like_tags
@parser = Radius::Parser.new(@context, :tag_prefix => "ralph")
define_tag('hello') { "hello" }
assert_equal "", @parser.parse("")
end
def test_copyin_global_values
@context.globals.foo = 'bar'
assert_equal 'bar', Radius::Parser.new(@context).context.globals.foo
end
def test_does_not_pollute_copied_globals
@context.globals.foo = 'bar'
parser = Radius::Parser.new(@context)
parser.context.globals.foo = '[baz]'
assert_equal 'bar', @context.globals.foo
end
def test_parse_with_other_namespaces
@parser = Radius::Parser.new(@context, :tag_prefix => 'r')
assert_equal "hello world", @parser.parse("hello world")
end
protected
def assert_parse_output(output, input, message = nil)
r = @parser.parse(input)
assert_equal(output, r, message)
end
def assert_parsed_is_unchanged(something)
assert_parse_output something, something
end
class User
attr_accessor :name, :age, :email, :friend
def initialize(name, age, email)
@name, @age, @email = name, age, email
end
def <=>(other)
name <=> other.name
end
end
class UserWithAttributes < User
def attributes
{ :name => name, :age => age, :email => email }
end
end
def users
[
User.new('John', 25, 'test@johnwlong.com'),
User.new('James', 27, 'test@jameslong.com')
]
end
def user_with_attributes
UserWithAttributes.new('John', 25, 'test@johnwlong.com')
end
end