compiled_templates_test.rb 7.4 KB
Newer Older
1 2 3
require "#{File.dirname(__FILE__)}/../abstract_unit"
require 'action_view/helpers/date_helper'
require 'action_view/compiled_templates'
4 5 6 7 8 9 10 11 12 13 14 15

class CompiledTemplateTests < Test::Unit::TestCase
  def setup
    @ct = ActionView::CompiledTemplates.new
    @v = Class.new
    @v.send :include, @ct
    @a = './test_compile_template_a.rhtml'
    @b = './test_compile_template_b.rhtml'
    @s = './test_compile_template_link.rhtml'
  end
  def teardown
    [@a, @b, @s].each do |f|
16
      FileUtils.rm(f) if File.exist?(f) || File.symlink?(f)
17 18 19 20 21 22 23 24
    end
  end
  attr_reader :ct, :v

  def test_name_allocation
    hi_world = ct.method_names['hi world']
    hi_sexy = ct.method_names['hi sexy']
    wish_upon_a_star = ct.method_names['I love seeing decent error messages']
25

26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70
    assert_equal hi_world, ct.method_names['hi world']
    assert_equal hi_sexy, ct.method_names['hi sexy']
    assert_equal wish_upon_a_star, ct.method_names['I love seeing decent error messages']
    assert_equal 3, [hi_world, hi_sexy, wish_upon_a_star].uniq.length
  end

  def test_wrap_source
    assert_equal(
      "def aliased_assignment(value)\nself.value = value\nend",
      @ct.wrap_source(:aliased_assignment, [:value], 'self.value = value')
    )

    assert_equal(
      "def simple()\nnil\nend",
      @ct.wrap_source(:simple, [], 'nil')
    )
  end

  def test_compile_source_single_method
    selector = ct.compile_source('doubling method', [:a], 'a + a')
    assert_equal 2, @v.new.send(selector, 1)
    assert_equal 4, @v.new.send(selector, 2)
    assert_equal -4, @v.new.send(selector, -2)
    assert_equal 0, @v.new.send(selector, 0)
    selector
  end

  def test_compile_source_two_method
    sel1 = test_compile_source_single_method # compile the method in the other test
    sel2 = ct.compile_source('doubling method', [:a, :b], 'a + b + a + b')
    assert_not_equal sel1, sel2

    assert_equal 2, @v.new.send(sel1, 1)
    assert_equal 4, @v.new.send(sel1, 2)

    assert_equal 6, @v.new.send(sel2, 1, 2)
    assert_equal 32, @v.new.send(sel2, 15, 1)
  end

  def test_mtime
    t1 = Time.now
    test_compile_source_single_method
    assert (t1..Time.now).include?(ct.mtime('doubling method', [:a]))
  end

71
  uses_mocha 'test_compile_time' do
72
  def test_compile_time
73
    t = Time.now
74

75 76 77 78 79
    File.open(@a, "w"){|f| f.puts @a}
    File.open(@b, "w"){|f| f.puts @b}
    # windows doesn't support symlinks (even under cygwin)
    windows = (RUBY_PLATFORM =~ /win32/)
    `ln -s #{@a} #{@s}` unless windows
80 81 82

    v = ActionView::Base.new
    v.base_path = '.'
83
    v.cache_template_loading = false
84

85 86 87
    # All templates were created at t+1
    File::Stat.any_instance.expects(:mtime).times(windows ? 2 : 3).returns(t + 1.second)

88 89 90 91
    # private methods template_changed_since? and compile_template?
    # should report true for all since they have not been compiled
    assert v.send(:template_changed_since?, @a, t)
    assert v.send(:template_changed_since?, @b, t)
92
    assert v.send(:template_changed_since?, @s, t) unless windows
93

94 95
    assert v.send(:compile_template?, nil, @a, {})
    assert v.send(:compile_template?, nil, @b, {})
96
    assert v.send(:compile_template?, nil, @s, {}) unless windows
97

98 99
    # All templates are rendered at t+2
    Time.expects(:now).times(windows ? 2 : 3).returns(t + 2.seconds)
100 101
    v.compile_and_render_template(:rhtml, '', @a)
    v.compile_and_render_template(:rhtml, '', @b)
102
    v.compile_and_render_template(:rhtml, '', @s) unless windows
103 104
    a_n = v.method_names[@a]
    b_n = v.method_names[@b]
105
    s_n = v.method_names[@s] unless windows
106 107 108
    # all of the files have changed since last compile
    assert v.compile_time[a_n] > t
    assert v.compile_time[b_n] > t
109
    assert v.compile_time[s_n] > t unless windows
110

111 112
    # private methods template_changed_since? and compile_template?
    # should report false for all since none have changed since compile
113
    File::Stat.any_instance.expects(:mtime).times(windows ? 6 : 12).returns(t + 1.second)
114 115
    assert !v.send(:template_changed_since?, @a, v.compile_time[a_n])
    assert !v.send(:template_changed_since?, @b, v.compile_time[b_n])
116
    assert !v.send(:template_changed_since?, @s, v.compile_time[s_n]) unless windows
117 118
    assert !v.send(:compile_template?, nil, @a, {})
    assert !v.send(:compile_template?, nil, @b, {})
119
    assert !v.send(:compile_template?, nil, @s, {}) unless windows
120 121
    v.compile_and_render_template(:rhtml, '', @a)
    v.compile_and_render_template(:rhtml, '', @b)
122
    v.compile_and_render_template(:rhtml, '', @s)  unless windows
123
    # none of the files have changed since last compile
124 125 126
    assert v.compile_time[a_n] < t + 3.seconds
    assert v.compile_time[b_n] < t + 3.seconds
    assert v.compile_time[s_n] < t + 3.seconds  unless windows
127

128
    `rm #{@s}; ln -s #{@b} #{@s}` unless windows
129 130
    # private methods template_changed_since? and compile_template?
    # should report true for symlink since it has changed since compile
131 132 133 134 135
    
    # t + 3.seconds is for the symlink
    File::Stat.any_instance.expects(:mtime).times(windows ? 6 : 9).returns(
      *(windows ? [ t + 1.second, t + 1.second ] :
        [ t + 1.second, t + 1.second, t + 3.second ]) * 3)
136 137
    assert !v.send(:template_changed_since?, @a, v.compile_time[a_n])
    assert !v.send(:template_changed_since?, @b, v.compile_time[b_n])
138
    assert v.send(:template_changed_since?, @s, v.compile_time[s_n]) unless windows
139 140
    assert !v.send(:compile_template?, nil, @a, {})
    assert !v.send(:compile_template?, nil, @b, {})
141
    assert v.send(:compile_template?, nil, @s, {}) unless windows
142 143 144

    # Only the symlink template gets rendered at t+3
    Time.stubs(:now).returns(t + 3.seconds) unless windows
145 146
    v.compile_and_render_template(:rhtml, '', @a)
    v.compile_and_render_template(:rhtml, '', @b)
147
    v.compile_and_render_template(:rhtml, '', @s) unless windows
148
    # the symlink has changed since last compile
149 150 151
    assert v.compile_time[a_n] < t + 3.seconds
    assert v.compile_time[b_n] < t + 3.seconds
    assert_equal v.compile_time[s_n], t + 3.seconds unless windows
152

153
    FileUtils.touch @b
154 155 156
    # private methods template_changed_since? and compile_template?
    # should report true for symlink and file at end of symlink
    # since it has changed since last compile
157 158 159 160 161
    #
    # t+4 is for @b and also for the file that @s points to, which is @b
    File::Stat.any_instance.expects(:mtime).times(windows ? 6 : 12).returns(
      *(windows ? [ t + 1.second, t + 4.seconds ] :
        [ t + 1.second, t + 4.seconds, t + 3.second, t + 4.seconds ]) * 3)
162 163
    assert !v.send(:template_changed_since?, @a, v.compile_time[a_n])
    assert v.send(:template_changed_since?, @b, v.compile_time[b_n])
164
    assert v.send(:template_changed_since?, @s, v.compile_time[s_n]) unless windows
165 166
    assert !v.send(:compile_template?, nil, @a, {})
    assert v.send(:compile_template?, nil, @b, {})
167 168
    assert v.send(:compile_template?, nil, @s, {}) unless windows

169
    Time.expects(:now).times(windows ? 1 : 2).returns(t + 5.seconds)
170 171
    v.compile_and_render_template(:rhtml, '', @a)
    v.compile_and_render_template(:rhtml, '', @b)
172
    v.compile_and_render_template(:rhtml, '', @s) unless windows
173 174
    # the file at the end of the symlink has changed since last compile
    # both the symlink and the file at the end of it should be recompiled
175 176 177 178
    assert v.compile_time[a_n] < t + 5.seconds
    assert_equal v.compile_time[b_n], t + 5.seconds
    assert_equal v.compile_time[s_n], t + 5.seconds unless windows
  end
179 180 181 182 183 184 185 186 187 188 189 190 191
  end
end

module ActionView
  class Base
    def compile_time
      @@compile_time
    end
    def method_names
      @@method_names
    end
  end
end