#!/usr/bin/env ruby require 'test/unit' require File.dirname(__FILE__) + '/../../lib/haml' require 'haml/engine' class EngineTest < Test::Unit::TestCase def render(text, options = {}) Haml::Engine.new(text, options).to_html end def test_empty_render_should_remain_empty assert_equal('', render('')) end # This is ugly because Hashes are unordered; we don't always know the order # in which attributes will be returned. # There is probably a better way to do this. def test_attributes_should_render_correctly assert_equal("
\n
", render(".atlantis{:style => 'ugly'}").chomp) rescue assert_equal("
\n
", render(".atlantis{:style => 'ugly'}").chomp) end def test_ruby_code_should_work_inside_attributes author = 'hcatlin' assert_equal("

foo

", render("%p{:class => 1+2} foo").chomp) end def test_nil_should_render_empty_tag assert_equal("
\n
", render(".no_attributes{:nil => nil}").chomp) end def test_strings_should_get_stripped_inside_tags assert_equal("
This should have no spaces in front of it
", render(".stripped This should have no spaces in front of it").chomp) end def test_one_liner_should_be_one_line assert_equal("

Hello

", render('%p Hello').chomp) end def test_long_liner_should_not_print_on_one_line assert_equal("
\n #{'x' * 51}\n
", render("%div #{'x' * 51}").chomp) end def test_non_prerendered_one_liner assert_equal("

One line

\n", render("%p{:class => c} One line", :locals => {:c => 'awesome'})) end def test_non_prerendered_script_one_liner assert_equal("

One line

\n", render("%p{:class => c}= 'One line'", :locals => {:c => 'awesome'})) end def test_non_prerendered_long_script_one_liner assert_equal("

\n #{'x' * 60}\n

\n", render("%p{:class => c}= 'x' * 60", :locals => {:c => 'awesome'})) end def test_multi_render engine = Haml::Engine.new("%strong Hi there!") assert_equal("Hi there!\n", engine.to_html) assert_equal("Hi there!\n", engine.to_html) assert_equal("Hi there!\n", engine.to_html) end def test_double_equals assert_equal("

Hello World

\n", render('%p== Hello #{who}', :locals => {:who => 'World'})) assert_equal("

\n Hello World\n

\n", render("%p\n == Hello \#{who}", :locals => {:who => 'World'})) end def test_double_equals_in_the_middle_of_a_string assert_equal("\"title 'Title'. \"\n", render("== \"title '\#{\"Title\"}'. \"")) end # Options tests def test_stop_eval assert_equal("", render("= 'Hello'", :suppress_eval => true)) assert_equal("", render("- puts 'foo'", :suppress_eval => true)) assert_equal("
\n", render("#foo{:yes => 'no'}/", :suppress_eval => true)) assert_equal("
\n", render("#foo{:yes => 'no', :call => a_function() }/", :suppress_eval => true)) assert_equal("
\n", render("%div[1]/", :suppress_eval => true)) begin assert_equal("", render(":ruby\n puts 'hello'", :suppress_eval => true)) rescue Haml::HamlError => err caught = true assert_equal('Filter "ruby" is not defined!', err.message) end assert(caught, "Rendering a ruby filter without evaluating didn't throw an error!") end def test_attr_wrapper assert_equal("

\n

\n", render("%p{ :strange => 'attrs'}", :attr_wrapper => '*')) assert_equal("

\n

\n", render("%p{ :escaped => 'quo\"te'}", :attr_wrapper => '"')) assert_equal("

\n

\n", render("%p{ :escaped => 'q\\'uo\"te'}", :attr_wrapper => '"')) assert_equal("\n", render("!!! XML", :attr_wrapper => '"')) end def test_attrs_parsed_correctly assert_equal("

biddly='bar => baz'>\n

\n", render("%p{'boom=>biddly' => 'bar => baz'}")) assert_equal("

\n

\n", render("%p{'foo,bar' => 'baz, qux'}")) assert_equal("

\n

\n", render("%p{ :escaped => \"quo\\nte\"}")) assert_equal("

\n

\n", render("%p{ :escaped => \"quo\#{2 + 2}te\"}")) end def test_empty_attrs assert_equal("

empty

\n", render("%p{ :attr => '' } empty")) assert_equal("

empty

\n", render("%p{ :attr => x } empty", :locals => {:x => ''})) end def test_nil_attrs assert_equal("

nil

\n", render("%p{ :attr => nil } nil")) assert_equal("

nil

\n", render("%p{ :attr => x } nil", :locals => {:x => nil})) end def test_locals assert_equal("

Paragraph!

\n", render("%p= text", :locals => { :text => "Paragraph!" })) end def test_recompile_with_new_locals template = "%p= (text == 'first time') ? text : new_text" assert_equal("

first time

\n", render(template, :locals => { :text => "first time" })) assert_equal("

second time

\n", render(template, :locals => { :text => "recompile", :new_text => "second time" })) # Make sure the method called will return junk unless recompiled method_name = Haml::Engine.send(:class_variable_get, '@@method_names')[template] Haml::Engine::CompiledTemplates.module_eval "def #{method_name}(stuff); @haml_stack[-1].push_text(\"NOT RECOMPILED\n\"); end" assert_equal("NOT RECOMPILED\n", render(template, :locals => { :text => "first time" })) assert_equal("

first time

\n", render(template, :locals => { :text => "first time", :foo => 'bar' })) end def test_comps assert_equal(-1, "foo" <=> nil) assert_equal(1, nil <=> "foo") end def test_rec_merge hash1 = {1=>2, 3=>{5=>7, 8=>9}} hash2 = {4=>5, 3=>{5=>2, 16=>12}} hash3 = {1=>2, 4=>5, 3=>{5=>2, 8=>9, 16=>12}} hash1.rec_merge!(hash2) assert_equal(hash3, hash1) end def test_exception_type begin render("%p hi\n= undefined") rescue Exception => e assert(e.is_a?(Haml::Error)) assert_equal(2, e.haml_line) assert_equal(nil, e.haml_filename) assert_equal('(haml):2', e.backtrace[0]) else # Test failed... should have raised an exception assert(false) end end def test_syntax_errors errs = [ "!!!\n a", "a\n b", "a\n:foo\nb", "/ a\n b", "% a", "%p a\n b", "a\n%p=\nb", "%p=\n a", "a\n%p~\nb", "a\n~\nb", "a\n~\n b", "%p~\n b", "%p/\n a", "%p\n \t%a b", "%a\n b\nc", "%a\n b\nc", ":notafilter\n This isn't\n a filter!", ".{} a", "\#{} a", ".= 'foo'", "%a/ b" ] errs.each do |err| begin render(err) rescue Exception => e assert(e.is_a?(Haml::Error), "#{err.dump} doesn't produce Haml::SyntaxError!") else assert(false, "#{err.dump} doesn't produce an exception!") end end end def test_compile_error begin render("a\nb\n- fee do\nc") rescue Exception => e assert_equal(3, e.haml_line) else assert(false, '"a\nb\n- fee do\nc" doesn\'t produce an exception!') end end def test_no_bluecloth Kernel.module_eval do def gem_original_require_with_bluecloth(file) raise LoadError if file == 'bluecloth' gem_original_require_without_bluecloth(file) end alias_method :gem_original_require_without_bluecloth, :gem_original_require alias_method :gem_original_require, :gem_original_require_with_bluecloth end begin assert_equal("

Foo

\t

- a\n- b

\n", Haml::Engine.new(":markdown\n Foo\n ===\n - a\n - b").to_html) rescue Haml::HamlError => e if e.message == "Can't run Markdown filter; required 'bluecloth' or 'redcloth', but none were found" puts "\nCouldn't require 'bluecloth' or 'redcloth'; skipping a test." else raise e end end Kernel.module_eval do alias_method :gem_original_require, :gem_original_require_without_bluecloth end end def test_no_redcloth Kernel.module_eval do def gem_original_require_with_redcloth(file) raise LoadError if file == 'redcloth' gem_original_require_without_redcloth(file) end alias_method :gem_original_require_without_redcloth, :gem_original_require alias_method :gem_original_require, :gem_original_require_with_redcloth end begin Haml::Engine.new(":redcloth\n _foo_").to_html rescue Haml::HamlError else assert(false, "No exception raised!") end Kernel.module_eval do alias_method :gem_original_require, :gem_original_require_without_redcloth end end def test_no_redcloth_or_bluecloth Kernel.module_eval do def gem_original_require_with_redcloth_and_bluecloth(file) raise LoadError if file == 'redcloth' || file == 'bluecloth' gem_original_require_without_redcloth_and_bluecloth(file) end alias_method :gem_original_require_without_redcloth_and_bluecloth, :gem_original_require alias_method :gem_original_require, :gem_original_require_with_redcloth_and_bluecloth end begin Haml::Engine.new(":markdown\n _foo_").to_html rescue Haml::HamlError else assert(false, "No exception raised!") end Kernel.module_eval do alias_method :gem_original_require, :gem_original_require_without_redcloth_and_bluecloth end end def test_local_assigns_dont_modify_class assert_equal("bar\n", render("= foo", :locals => {:foo => 'bar'})) assert_equal(nil, defined?(foo)) end def test_object_ref_with_nil_id user = Struct.new('User', :id).new assert_equal("

New User

\n", render("%p[user] New User", :locals => {:user => user})) end end