Revision 912:5e80956cc792 .svn/pristine/07

View differences:

.svn/pristine/07/07402976102eaabe37c42cc6aab7aeb00d221891.svn-base
1
# Redmine - project management software
2
# Copyright (C) 2006-2011  Jean-Philippe Lang
3
#
4
# This program is free software; you can redistribute it and/or
5
# modify it under the terms of the GNU General Public License
6
# as published by the Free Software Foundation; either version 2
7
# of the License, or (at your option) any later version.
8
#
9
# This program is distributed in the hope that it will be useful,
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
# GNU General Public License for more details.
13
#
14
# You should have received a copy of the GNU General Public License
15
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
17

  
18
require File.expand_path('../../test_helper', __FILE__)
19
require 'repositories_controller'
20

  
21
# Re-raise errors caught by the controller.
22
class RepositoriesController; def rescue_action(e) raise e end; end
23

  
24
class RepositoriesDarcsControllerTest < ActionController::TestCase
25
  fixtures :projects, :users, :roles, :members, :member_roles,
26
           :repositories, :enabled_modules
27

  
28
  REPOSITORY_PATH = Rails.root.join('tmp/test/darcs_repository').to_s
29
  PRJ_ID = 3
30
  NUM_REV = 6
31

  
32
  def setup
33
    @controller = RepositoriesController.new
34
    @request    = ActionController::TestRequest.new
35
    @response   = ActionController::TestResponse.new
36
    User.current = nil
37
    @project = Project.find(PRJ_ID)
38
    @repository = Repository::Darcs.create(
39
                        :project      => @project,
40
                        :url          => REPOSITORY_PATH,
41
                        :log_encoding => 'UTF-8'
42
                        )
43
    assert @repository
44
  end
45

  
46
  if File.directory?(REPOSITORY_PATH)
47
    def test_browse_root
48
      assert_equal 0, @repository.changesets.count
49
      @repository.fetch_changesets
50
      @project.reload
51
      assert_equal NUM_REV, @repository.changesets.count
52
      get :show, :id => PRJ_ID
53
      assert_response :success
54
      assert_template 'show'
55
      assert_not_nil assigns(:entries)
56
      assert_equal 3, assigns(:entries).size
57
      assert assigns(:entries).detect {|e| e.name == 'images' && e.kind == 'dir'}
58
      assert assigns(:entries).detect {|e| e.name == 'sources' && e.kind == 'dir'}
59
      assert assigns(:entries).detect {|e| e.name == 'README' && e.kind == 'file'}
60
    end
61

  
62
    def test_browse_directory
63
      assert_equal 0, @repository.changesets.count
64
      @repository.fetch_changesets
65
      @project.reload
66
      assert_equal NUM_REV, @repository.changesets.count
67
      get :show, :id => PRJ_ID, :path => ['images']
68
      assert_response :success
69
      assert_template 'show'
70
      assert_not_nil assigns(:entries)
71
      assert_equal ['delete.png', 'edit.png'], assigns(:entries).collect(&:name)
72
      entry = assigns(:entries).detect {|e| e.name == 'edit.png'}
73
      assert_not_nil entry
74
      assert_equal 'file', entry.kind
75
      assert_equal 'images/edit.png', entry.path
76
    end
77

  
78
    def test_browse_at_given_revision
79
      assert_equal 0, @repository.changesets.count
80
      @repository.fetch_changesets
81
      @project.reload
82
      assert_equal NUM_REV, @repository.changesets.count
83
      get :show, :id => PRJ_ID, :path => ['images'], :rev => 1
84
      assert_response :success
85
      assert_template 'show'
86
      assert_not_nil assigns(:entries)
87
      assert_equal ['delete.png'], assigns(:entries).collect(&:name)
88
    end
89

  
90
    def test_changes
91
      assert_equal 0, @repository.changesets.count
92
      @repository.fetch_changesets
93
      @project.reload
94
      assert_equal NUM_REV, @repository.changesets.count
95
      get :changes, :id => PRJ_ID, :path => ['images', 'edit.png']
96
      assert_response :success
97
      assert_template 'changes'
98
      assert_tag :tag => 'h2', :content => 'edit.png'
99
    end
100

  
101
    def test_diff
102
      assert_equal 0, @repository.changesets.count
103
      @repository.fetch_changesets
104
      @project.reload
105
      assert_equal NUM_REV, @repository.changesets.count
106
      # Full diff of changeset 5
107
      ['inline', 'sbs'].each do |dt|
108
        get :diff, :id => PRJ_ID, :rev => 5, :type => dt
109
        assert_response :success
110
        assert_template 'diff'
111
        # Line 22 removed
112
        assert_tag :tag => 'th',
113
                   :content => '22',
114
                   :sibling => { :tag => 'td',
115
                                 :attributes => { :class => /diff_out/ },
116
                                 :content => /def remove/ }
117
      end
118
    end
119

  
120
    def test_destroy_valid_repository
121
      @request.session[:user_id] = 1 # admin
122
      assert_equal 0, @repository.changesets.count
123
      @repository.fetch_changesets
124
      @project.reload
125
      assert_equal NUM_REV, @repository.changesets.count
126

  
127
      get :destroy, :id => PRJ_ID
128
      assert_response 302
129
      @project.reload
130
      assert_nil @project.repository
131
    end
132

  
133
    def test_destroy_invalid_repository
134
      @request.session[:user_id] = 1 # admin
135
      assert_equal 0, @repository.changesets.count
136
      @repository.fetch_changesets
137
      @project.reload
138
      assert_equal NUM_REV, @repository.changesets.count
139

  
140
      get :destroy, :id => PRJ_ID
141
      assert_response 302
142
      @project.reload
143
      assert_nil @project.repository
144

  
145
      @repository = Repository::Darcs.create(
146
                        :project      => @project,
147
                        :url          => "/invalid",
148
                        :log_encoding => 'UTF-8'
149
                        )
150
      assert @repository
151
      @repository.fetch_changesets
152
      @project.reload
153
      assert_equal 0, @repository.changesets.count
154

  
155
      get :destroy, :id => PRJ_ID
156
      assert_response 302
157
      @project.reload
158
      assert_nil @project.repository
159
    end
160
  else
161
    puts "Darcs test repository NOT FOUND. Skipping functional tests !!!"
162
    def test_fake; assert true end
163
  end
164
end
.svn/pristine/07/074588d42b3ac157ab70417170443c5531685d7e.svn-base
1
module ActiveRecord
2
  module Acts #:nodoc:
3
    module List #:nodoc:
4
      def self.included(base)
5
        base.extend(ClassMethods)
6
      end
7

  
8
      # This +acts_as+ extension provides the capabilities for sorting and reordering a number of objects in a list.
9
      # The class that has this specified needs to have a +position+ column defined as an integer on
10
      # the mapped database table.
11
      #
12
      # Todo list example:
13
      #
14
      #   class TodoList < ActiveRecord::Base
15
      #     has_many :todo_items, :order => "position"
16
      #   end
17
      #
18
      #   class TodoItem < ActiveRecord::Base
19
      #     belongs_to :todo_list
20
      #     acts_as_list :scope => :todo_list
21
      #   end
22
      #
23
      #   todo_list.first.move_to_bottom
24
      #   todo_list.last.move_higher
25
      module ClassMethods
26
        # Configuration options are:
27
        #
28
        # * +column+ - specifies the column name to use for keeping the position integer (default: +position+)
29
        # * +scope+ - restricts what is to be considered a list. Given a symbol, it'll attach <tt>_id</tt> 
30
        #   (if it hasn't already been added) and use that as the foreign key restriction. It's also possible 
31
        #   to give it an entire string that is interpolated if you need a tighter scope than just a foreign key.
32
        #   Example: <tt>acts_as_list :scope => 'todo_list_id = #{todo_list_id} AND completed = 0'</tt>
33
        def acts_as_list(options = {})
34
          configuration = { :column => "position", :scope => "1 = 1" }
35
          configuration.update(options) if options.is_a?(Hash)
36

  
37
          configuration[:scope] = "#{configuration[:scope]}_id".intern if configuration[:scope].is_a?(Symbol) && configuration[:scope].to_s !~ /_id$/
38

  
39
          if configuration[:scope].is_a?(Symbol)
40
            scope_condition_method = %(
41
              def scope_condition
42
                if #{configuration[:scope].to_s}.nil?
43
                  "#{configuration[:scope].to_s} IS NULL"
44
                else
45
                  "#{configuration[:scope].to_s} = \#{#{configuration[:scope].to_s}}"
46
                end
47
              end
48
            )
49
          else
50
            scope_condition_method = "def scope_condition() \"#{configuration[:scope]}\" end"
51
          end
52

  
53
          class_eval <<-EOV
54
            include ActiveRecord::Acts::List::InstanceMethods
55

  
56
            def acts_as_list_class
57
              ::#{self.name}
58
            end
59

  
60
            def position_column
61
              '#{configuration[:column]}'
62
            end
63

  
64
            #{scope_condition_method}
65

  
66
            before_destroy :remove_from_list
67
            before_create  :add_to_list_bottom
68
          EOV
69
        end
70
      end
71

  
72
      # All the methods available to a record that has had <tt>acts_as_list</tt> specified. Each method works
73
      # by assuming the object to be the item in the list, so <tt>chapter.move_lower</tt> would move that chapter
74
      # lower in the list of all chapters. Likewise, <tt>chapter.first?</tt> would return +true+ if that chapter is
75
      # the first in the list of all chapters.
76
      module InstanceMethods
77
        # Insert the item at the given position (defaults to the top position of 1).
78
        def insert_at(position = 1)
79
          insert_at_position(position)
80
        end
81

  
82
        # Swap positions with the next lower item, if one exists.
83
        def move_lower
84
          return unless lower_item
85

  
86
          acts_as_list_class.transaction do
87
            lower_item.decrement_position
88
            increment_position
89
          end
90
        end
91

  
92
        # Swap positions with the next higher item, if one exists.
93
        def move_higher
94
          return unless higher_item
95

  
96
          acts_as_list_class.transaction do
97
            higher_item.increment_position
98
            decrement_position
99
          end
100
        end
101

  
102
        # Move to the bottom of the list. If the item is already in the list, the items below it have their
103
        # position adjusted accordingly.
104
        def move_to_bottom
105
          return unless in_list?
106
          acts_as_list_class.transaction do
107
            decrement_positions_on_lower_items
108
            assume_bottom_position
109
          end
110
        end
111

  
112
        # Move to the top of the list. If the item is already in the list, the items above it have their
113
        # position adjusted accordingly.
114
        def move_to_top
115
          return unless in_list?
116
          acts_as_list_class.transaction do
117
            increment_positions_on_higher_items
118
            assume_top_position
119
          end
120
        end
121
        
122
        # Move to the given position
123
        def move_to=(pos)
124
          case pos.to_s
125
          when 'highest'
126
            move_to_top
127
          when 'higher'
128
            move_higher
129
          when 'lower'
130
            move_lower
131
          when 'lowest'
132
            move_to_bottom
133
          end
134
        end
135

  
136
        # Removes the item from the list.
137
        def remove_from_list
138
          if in_list?
139
            decrement_positions_on_lower_items
140
            update_attribute position_column, nil
141
          end
142
        end
143

  
144
        # Increase the position of this item without adjusting the rest of the list.
145
        def increment_position
146
          return unless in_list?
147
          update_attribute position_column, self.send(position_column).to_i + 1
148
        end
149

  
150
        # Decrease the position of this item without adjusting the rest of the list.
151
        def decrement_position
152
          return unless in_list?
153
          update_attribute position_column, self.send(position_column).to_i - 1
154
        end
155

  
156
        # Return +true+ if this object is the first in the list.
157
        def first?
158
          return false unless in_list?
159
          self.send(position_column) == 1
160
        end
161

  
162
        # Return +true+ if this object is the last in the list.
163
        def last?
164
          return false unless in_list?
165
          self.send(position_column) == bottom_position_in_list
166
        end
167

  
168
        # Return the next higher item in the list.
169
        def higher_item
170
          return nil unless in_list?
171
          acts_as_list_class.find(:first, :conditions =>
172
            "#{scope_condition} AND #{position_column} = #{(send(position_column).to_i - 1).to_s}"
173
          )
174
        end
175

  
176
        # Return the next lower item in the list.
177
        def lower_item
178
          return nil unless in_list?
179
          acts_as_list_class.find(:first, :conditions =>
180
            "#{scope_condition} AND #{position_column} = #{(send(position_column).to_i + 1).to_s}"
181
          )
182
        end
183

  
184
        # Test if this record is in a list
185
        def in_list?
186
          !send(position_column).nil?
187
        end
188

  
189
        private
190
          def add_to_list_top
191
            increment_positions_on_all_items
192
          end
193

  
194
          def add_to_list_bottom
195
            self[position_column] = bottom_position_in_list.to_i + 1
196
          end
197

  
198
          # Overwrite this method to define the scope of the list changes
199
          def scope_condition() "1" end
200

  
201
          # Returns the bottom position number in the list.
202
          #   bottom_position_in_list    # => 2
203
          def bottom_position_in_list(except = nil)
204
            item = bottom_item(except)
205
            item ? item.send(position_column) : 0
206
          end
207

  
208
          # Returns the bottom item
209
          def bottom_item(except = nil)
210
            conditions = scope_condition
211
            conditions = "#{conditions} AND #{self.class.primary_key} != #{except.id}" if except
212
            acts_as_list_class.find(:first, :conditions => conditions, :order => "#{position_column} DESC")
213
          end
214

  
215
          # Forces item to assume the bottom position in the list.
216
          def assume_bottom_position
217
            update_attribute(position_column, bottom_position_in_list(self).to_i + 1)
218
          end
219

  
220
          # Forces item to assume the top position in the list.
221
          def assume_top_position
222
            update_attribute(position_column, 1)
223
          end
224

  
225
          # This has the effect of moving all the higher items up one.
226
          def decrement_positions_on_higher_items(position)
227
            acts_as_list_class.update_all(
228
              "#{position_column} = (#{position_column} - 1)", "#{scope_condition} AND #{position_column} <= #{position}"
229
            )
230
          end
231

  
232
          # This has the effect of moving all the lower items up one.
233
          def decrement_positions_on_lower_items
234
            return unless in_list?
235
            acts_as_list_class.update_all(
236
              "#{position_column} = (#{position_column} - 1)", "#{scope_condition} AND #{position_column} > #{send(position_column).to_i}"
237
            )
238
          end
239

  
240
          # This has the effect of moving all the higher items down one.
241
          def increment_positions_on_higher_items
242
            return unless in_list?
243
            acts_as_list_class.update_all(
244
              "#{position_column} = (#{position_column} + 1)", "#{scope_condition} AND #{position_column} < #{send(position_column).to_i}"
245
            )
246
          end
247

  
248
          # This has the effect of moving all the lower items down one.
249
          def increment_positions_on_lower_items(position)
250
            acts_as_list_class.update_all(
251
              "#{position_column} = (#{position_column} + 1)", "#{scope_condition} AND #{position_column} >= #{position}"
252
           )
253
          end
254

  
255
          # Increments position (<tt>position_column</tt>) of all items in the list.
256
          def increment_positions_on_all_items
257
            acts_as_list_class.update_all(
258
              "#{position_column} = (#{position_column} + 1)",  "#{scope_condition}"
259
            )
260
          end
261

  
262
          def insert_at_position(position)
263
            remove_from_list
264
            increment_positions_on_lower_items(position)
265
            self.update_attribute(position_column, position)
266
          end
267
      end 
268
    end
269
  end
270
end

Also available in: Unified diff