##// END OF EJS Templates
Refactor: move method, ProjectsController#roadmap to VersionsController#index....
Eric Davis -
r3936:a188abbe2813
parent child
Show More
@@ -1,345 +1,313
1 1 # Redmine - project management software
2 2 # Copyright (C) 2006-2009 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 class ProjectsController < ApplicationController
19 19 menu_item :overview
20 20 menu_item :roadmap, :only => :roadmap
21 21 menu_item :files, :only => [:list_files, :add_file]
22 22 menu_item :settings, :only => :settings
23 23
24 24 before_filter :find_project, :except => [ :index, :list, :add, :copy ]
25 25 before_filter :authorize, :except => [ :index, :list, :add, :copy, :archive, :unarchive, :destroy]
26 26 before_filter :authorize_global, :only => :add
27 27 before_filter :require_admin, :only => [ :copy, :archive, :unarchive, :destroy ]
28 28 accept_key_auth :index
29 29
30 30 after_filter :only => [:add, :edit, :archive, :unarchive, :destroy] do |controller|
31 31 if controller.request.post?
32 32 controller.send :expire_action, :controller => 'welcome', :action => 'robots.txt'
33 33 end
34 34 end
35 35
36 36 helper :sort
37 37 include SortHelper
38 38 helper :custom_fields
39 39 include CustomFieldsHelper
40 40 helper :issues
41 41 helper :queries
42 42 include QueriesHelper
43 43 helper :repositories
44 44 include RepositoriesHelper
45 45 include ProjectsHelper
46 46
47 47 # Lists visible projects
48 48 def index
49 49 respond_to do |format|
50 50 format.html {
51 51 @projects = Project.visible.find(:all, :order => 'lft')
52 52 }
53 53 format.xml {
54 54 @projects = Project.visible.find(:all, :order => 'lft')
55 55 }
56 56 format.atom {
57 57 projects = Project.visible.find(:all, :order => 'created_on DESC',
58 58 :limit => Setting.feeds_limit.to_i)
59 59 render_feed(projects, :title => "#{Setting.app_title}: #{l(:label_project_latest)}")
60 60 }
61 61 end
62 62 end
63 63
64 64 # Add a new project
65 65 def add
66 66 @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position")
67 67 @trackers = Tracker.all
68 68 @project = Project.new(params[:project])
69 69 if request.get?
70 70 @project.identifier = Project.next_identifier if Setting.sequential_project_identifiers?
71 71 @project.trackers = Tracker.all
72 72 @project.is_public = Setting.default_projects_public?
73 73 @project.enabled_module_names = Setting.default_projects_modules
74 74 else
75 75 @project.enabled_module_names = params[:enabled_modules]
76 76 if validate_parent_id && @project.save
77 77 @project.set_allowed_parent!(params[:project]['parent_id']) if params[:project].has_key?('parent_id')
78 78 # Add current user as a project member if he is not admin
79 79 unless User.current.admin?
80 80 r = Role.givable.find_by_id(Setting.new_project_user_role_id.to_i) || Role.givable.first
81 81 m = Member.new(:user => User.current, :roles => [r])
82 82 @project.members << m
83 83 end
84 84 respond_to do |format|
85 85 format.html {
86 86 flash[:notice] = l(:notice_successful_create)
87 87 redirect_to :controller => 'projects', :action => 'settings', :id => @project
88 88 }
89 89 format.xml { head :created, :location => url_for(:controller => 'projects', :action => 'show', :id => @project.id) }
90 90 end
91 91 else
92 92 respond_to do |format|
93 93 format.html
94 94 format.xml { render :xml => @project.errors, :status => :unprocessable_entity }
95 95 end
96 96 end
97 97 end
98 98 end
99 99
100 100 def copy
101 101 @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position")
102 102 @trackers = Tracker.all
103 103 @root_projects = Project.find(:all,
104 104 :conditions => "parent_id IS NULL AND status = #{Project::STATUS_ACTIVE}",
105 105 :order => 'name')
106 106 @source_project = Project.find(params[:id])
107 107 if request.get?
108 108 @project = Project.copy_from(@source_project)
109 109 if @project
110 110 @project.identifier = Project.next_identifier if Setting.sequential_project_identifiers?
111 111 else
112 112 redirect_to :controller => 'admin', :action => 'projects'
113 113 end
114 114 else
115 115 Mailer.with_deliveries(params[:notifications] == '1') do
116 116 @project = Project.new(params[:project])
117 117 @project.enabled_module_names = params[:enabled_modules]
118 118 if validate_parent_id && @project.copy(@source_project, :only => params[:only])
119 119 @project.set_allowed_parent!(params[:project]['parent_id']) if params[:project].has_key?('parent_id')
120 120 flash[:notice] = l(:notice_successful_create)
121 121 redirect_to :controller => 'admin', :action => 'projects'
122 122 elsif !@project.new_record?
123 123 # Project was created
124 124 # But some objects were not copied due to validation failures
125 125 # (eg. issues from disabled trackers)
126 126 # TODO: inform about that
127 127 redirect_to :controller => 'admin', :action => 'projects'
128 128 end
129 129 end
130 130 end
131 131 rescue ActiveRecord::RecordNotFound
132 132 redirect_to :controller => 'admin', :action => 'projects'
133 133 end
134 134
135 135 # Show @project
136 136 def show
137 137 if params[:jump]
138 138 # try to redirect to the requested menu item
139 139 redirect_to_project_menu_item(@project, params[:jump]) && return
140 140 end
141 141
142 142 @users_by_role = @project.users_by_role
143 143 @subprojects = @project.children.visible
144 144 @news = @project.news.find(:all, :limit => 5, :include => [ :author, :project ], :order => "#{News.table_name}.created_on DESC")
145 145 @trackers = @project.rolled_up_trackers
146 146
147 147 cond = @project.project_condition(Setting.display_subprojects_issues?)
148 148
149 149 @open_issues_by_tracker = Issue.visible.count(:group => :tracker,
150 150 :include => [:project, :status, :tracker],
151 151 :conditions => ["(#{cond}) AND #{IssueStatus.table_name}.is_closed=?", false])
152 152 @total_issues_by_tracker = Issue.visible.count(:group => :tracker,
153 153 :include => [:project, :status, :tracker],
154 154 :conditions => cond)
155 155
156 156 TimeEntry.visible_by(User.current) do
157 157 @total_hours = TimeEntry.sum(:hours,
158 158 :include => :project,
159 159 :conditions => cond).to_f
160 160 end
161 161 @key = User.current.rss_key
162 162
163 163 respond_to do |format|
164 164 format.html
165 165 format.xml
166 166 end
167 167 end
168 168
169 169 def settings
170 170 @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position")
171 171 @issue_category ||= IssueCategory.new
172 172 @member ||= @project.members.new
173 173 @trackers = Tracker.all
174 174 @repository ||= @project.repository
175 175 @wiki ||= @project.wiki
176 176 end
177 177
178 178 # Edit @project
179 179 def edit
180 180 if request.get?
181 181 else
182 182 @project.attributes = params[:project]
183 183 if validate_parent_id && @project.save
184 184 @project.set_allowed_parent!(params[:project]['parent_id']) if params[:project].has_key?('parent_id')
185 185 respond_to do |format|
186 186 format.html {
187 187 flash[:notice] = l(:notice_successful_update)
188 188 redirect_to :action => 'settings', :id => @project
189 189 }
190 190 format.xml { head :ok }
191 191 end
192 192 else
193 193 respond_to do |format|
194 194 format.html {
195 195 settings
196 196 render :action => 'settings'
197 197 }
198 198 format.xml { render :xml => @project.errors, :status => :unprocessable_entity }
199 199 end
200 200 end
201 201 end
202 202 end
203 203
204 204 def modules
205 205 @project.enabled_module_names = params[:enabled_modules]
206 206 flash[:notice] = l(:notice_successful_update)
207 207 redirect_to :action => 'settings', :id => @project, :tab => 'modules'
208 208 end
209 209
210 210 def archive
211 211 if request.post?
212 212 unless @project.archive
213 213 flash[:error] = l(:error_can_not_archive_project)
214 214 end
215 215 end
216 216 redirect_to(url_for(:controller => 'admin', :action => 'projects', :status => params[:status]))
217 217 end
218 218
219 219 def unarchive
220 220 @project.unarchive if request.post? && !@project.active?
221 221 redirect_to(url_for(:controller => 'admin', :action => 'projects', :status => params[:status]))
222 222 end
223 223
224 224 # Delete @project
225 225 def destroy
226 226 @project_to_destroy = @project
227 227 if request.get?
228 228 # display confirmation view
229 229 else
230 230 if params[:format] == 'xml' || params[:confirm]
231 231 @project_to_destroy.destroy
232 232 respond_to do |format|
233 233 format.html { redirect_to :controller => 'admin', :action => 'projects' }
234 234 format.xml { head :ok }
235 235 end
236 236 end
237 237 end
238 238 # hide project in layout
239 239 @project = nil
240 240 end
241 241
242 242 def add_file
243 243 if request.post?
244 244 container = (params[:version_id].blank? ? @project : @project.versions.find_by_id(params[:version_id]))
245 245 attachments = Attachment.attach_files(container, params[:attachments])
246 246 render_attachment_warning_if_needed(container)
247 247
248 248 if !attachments.empty? && Setting.notified_events.include?('file_added')
249 249 Mailer.deliver_attachments_added(attachments[:files])
250 250 end
251 251 redirect_to :controller => 'projects', :action => 'list_files', :id => @project
252 252 return
253 253 end
254 254 @versions = @project.versions.sort
255 255 end
256 256
257 257 def save_activities
258 258 if request.post? && params[:enumerations]
259 259 Project.transaction do
260 260 params[:enumerations].each do |id, activity|
261 261 @project.update_or_create_time_entry_activity(id, activity)
262 262 end
263 263 end
264 264 flash[:notice] = l(:notice_successful_update)
265 265 end
266 266
267 267 redirect_to :controller => 'projects', :action => 'settings', :tab => 'activities', :id => @project
268 268 end
269 269
270 270 def reset_activities
271 271 @project.time_entry_activities.each do |time_entry_activity|
272 272 time_entry_activity.destroy(time_entry_activity.parent)
273 273 end
274 274 flash[:notice] = l(:notice_successful_update)
275 275 redirect_to :controller => 'projects', :action => 'settings', :tab => 'activities', :id => @project
276 276 end
277 277
278 278 def list_files
279 279 sort_init 'filename', 'asc'
280 280 sort_update 'filename' => "#{Attachment.table_name}.filename",
281 281 'created_on' => "#{Attachment.table_name}.created_on",
282 282 'size' => "#{Attachment.table_name}.filesize",
283 283 'downloads' => "#{Attachment.table_name}.downloads"
284 284
285 285 @containers = [ Project.find(@project.id, :include => :attachments, :order => sort_clause)]
286 286 @containers += @project.versions.find(:all, :include => :attachments, :order => sort_clause).sort.reverse
287 287 render :layout => !request.xhr?
288 288 end
289 289
290 def roadmap
291 @trackers = @project.trackers.find(:all, :order => 'position')
292 retrieve_selected_tracker_ids(@trackers, @trackers.select {|t| t.is_in_roadmap?})
293 @with_subprojects = params[:with_subprojects].nil? ? Setting.display_subprojects_issues? : (params[:with_subprojects] == '1')
294 project_ids = @with_subprojects ? @project.self_and_descendants.collect(&:id) : [@project.id]
295
296 @versions = @project.shared_versions || []
297 @versions += @project.rolled_up_versions.visible if @with_subprojects
298 @versions = @versions.uniq.sort
299 @versions.reject! {|version| version.closed? || version.completed? } unless params[:completed]
300
301 @issues_by_version = {}
302 unless @selected_tracker_ids.empty?
303 @versions.each do |version|
304 issues = version.fixed_issues.visible.find(:all,
305 :include => [:project, :status, :tracker, :priority],
306 :conditions => {:tracker_id => @selected_tracker_ids, :project_id => project_ids},
307 :order => "#{Project.table_name}.lft, #{Tracker.table_name}.position, #{Issue.table_name}.id")
308 @issues_by_version[version] = issues
309 end
310 end
311 @versions.reject! {|version| !project_ids.include?(version.project_id) && @issues_by_version[version].blank?}
312 end
313
314 290 private
315 291 def find_optional_project
316 292 return true unless params[:id]
317 293 @project = Project.find(params[:id])
318 294 authorize
319 295 rescue ActiveRecord::RecordNotFound
320 296 render_404
321 297 end
322 298
323 def retrieve_selected_tracker_ids(selectable_trackers, default_trackers=nil)
324 if ids = params[:tracker_ids]
325 @selected_tracker_ids = (ids.is_a? Array) ? ids.collect { |id| id.to_i.to_s } : ids.split('/').collect { |id| id.to_i.to_s }
326 else
327 @selected_tracker_ids = (default_trackers || selectable_trackers).collect {|t| t.id.to_s }
328 end
329 end
330
331 299 # Validates parent_id param according to user's permissions
332 300 # TODO: move it to Project model in a validation that depends on User.current
333 301 def validate_parent_id
334 302 return true if User.current.admin?
335 303 parent_id = params[:project] && params[:project][:parent_id]
336 304 if parent_id || @project.new_record?
337 305 parent = parent_id.blank? ? nil : Project.find_by_id(parent_id.to_i)
338 306 unless @project.allowed_parents.include?(parent)
339 307 @project.errors.add :parent_id, :invalid
340 308 return false
341 309 end
342 310 end
343 311 true
344 312 end
345 313 end
@@ -1,108 +1,141
1 1 # redMine - project management software
2 2 # Copyright (C) 2006 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 class VersionsController < ApplicationController
19 19 menu_item :roadmap
20 20 model_object Version
21 before_filter :find_model_object, :except => [:new, :close_completed]
22 before_filter :find_project_from_association, :except => [:new, :close_completed]
23 before_filter :find_project, :only => [:new, :close_completed]
21 before_filter :find_model_object, :except => [:index, :new, :close_completed]
22 before_filter :find_project_from_association, :except => [:index, :new, :close_completed]
23 before_filter :find_project, :only => [:index, :new, :close_completed]
24 24 before_filter :authorize
25 25
26 26 helper :custom_fields
27 27 helper :projects
28
29 def index
30 @trackers = @project.trackers.find(:all, :order => 'position')
31 retrieve_selected_tracker_ids(@trackers, @trackers.select {|t| t.is_in_roadmap?})
32 @with_subprojects = params[:with_subprojects].nil? ? Setting.display_subprojects_issues? : (params[:with_subprojects] == '1')
33 project_ids = @with_subprojects ? @project.self_and_descendants.collect(&:id) : [@project.id]
34
35 @versions = @project.shared_versions || []
36 @versions += @project.rolled_up_versions.visible if @with_subprojects
37 @versions = @versions.uniq.sort
38 @versions.reject! {|version| version.closed? || version.completed? } unless params[:completed]
39
40 @issues_by_version = {}
41 unless @selected_tracker_ids.empty?
42 @versions.each do |version|
43 issues = version.fixed_issues.visible.find(:all,
44 :include => [:project, :status, :tracker, :priority],
45 :conditions => {:tracker_id => @selected_tracker_ids, :project_id => project_ids},
46 :order => "#{Project.table_name}.lft, #{Tracker.table_name}.position, #{Issue.table_name}.id")
47 @issues_by_version[version] = issues
48 end
49 end
50 @versions.reject! {|version| !project_ids.include?(version.project_id) && @issues_by_version[version].blank?}
51 end
28 52
29 53 def show
30 54 @issues = @version.fixed_issues.visible.find(:all,
31 55 :include => [:status, :tracker, :priority],
32 56 :order => "#{Tracker.table_name}.position, #{Issue.table_name}.id")
33 57 end
34 58
35 59 def new
36 60 @version = @project.versions.build
37 61 if params[:version]
38 62 attributes = params[:version].dup
39 63 attributes.delete('sharing') unless attributes.nil? || @version.allowed_sharings.include?(attributes['sharing'])
40 64 @version.attributes = attributes
41 65 end
42 66 if request.post?
43 67 if @version.save
44 68 respond_to do |format|
45 69 format.html do
46 70 flash[:notice] = l(:notice_successful_create)
47 71 redirect_to :controller => 'projects', :action => 'settings', :tab => 'versions', :id => @project
48 72 end
49 73 format.js do
50 74 # IE doesn't support the replace_html rjs method for select box options
51 75 render(:update) {|page| page.replace "issue_fixed_version_id",
52 76 content_tag('select', '<option></option>' + version_options_for_select(@project.shared_versions.open, @version), :id => 'issue_fixed_version_id', :name => 'issue[fixed_version_id]')
53 77 }
54 78 end
55 79 end
56 80 else
57 81 respond_to do |format|
58 82 format.html
59 83 format.js do
60 84 render(:update) {|page| page.alert(@version.errors.full_messages.join('\n')) }
61 85 end
62 86 end
63 87 end
64 88 end
65 89 end
66 90
67 91 def edit
68 92 if request.post? && params[:version]
69 93 attributes = params[:version].dup
70 94 attributes.delete('sharing') unless @version.allowed_sharings.include?(attributes['sharing'])
71 95 if @version.update_attributes(attributes)
72 96 flash[:notice] = l(:notice_successful_update)
73 97 redirect_to :controller => 'projects', :action => 'settings', :tab => 'versions', :id => @project
74 98 end
75 99 end
76 100 end
77 101
78 102 def close_completed
79 103 if request.post?
80 104 @project.close_completed_versions
81 105 end
82 106 redirect_to :controller => 'projects', :action => 'settings', :tab => 'versions', :id => @project
83 107 end
84 108
85 109 def destroy
86 110 if @version.fixed_issues.empty?
87 111 @version.destroy
88 112 redirect_to :controller => 'projects', :action => 'settings', :tab => 'versions', :id => @project
89 113 else
90 114 flash[:error] = l(:notice_unable_delete_version)
91 115 redirect_to :controller => 'projects', :action => 'settings', :tab => 'versions', :id => @project
92 116 end
93 117 end
94 118
95 119 def status_by
96 120 respond_to do |format|
97 121 format.html { render :action => 'show' }
98 122 format.js { render(:update) {|page| page.replace_html 'status_by', render_issue_status_by(@version, params[:status_by])} }
99 123 end
100 124 end
101 125
102 126 private
103 127 def find_project
104 128 @project = Project.find(params[:project_id])
105 129 rescue ActiveRecord::RecordNotFound
106 130 render_404
107 131 end
132
133 def retrieve_selected_tracker_ids(selectable_trackers, default_trackers=nil)
134 if ids = params[:tracker_ids]
135 @selected_tracker_ids = (ids.is_a? Array) ? ids.collect { |id| id.to_i.to_s } : ids.split('/').collect { |id| id.to_i.to_s }
136 else
137 @selected_tracker_ids = (default_trackers || selectable_trackers).collect {|t| t.id.to_s }
138 end
139 end
140
108 141 end
1 NO CONTENT: file renamed from app/views/projects/roadmap.rhtml to app/views/versions/index.html.erb
@@ -1,282 +1,283
1 1 ActionController::Routing::Routes.draw do |map|
2 2 # Add your own custom routes here.
3 3 # The priority is based upon order of creation: first created -> highest priority.
4 4
5 5 # Here's a sample route:
6 6 # map.connect 'products/:id', :controller => 'catalog', :action => 'view'
7 7 # Keep in mind you can assign values other than :controller and :action
8 8
9 9 map.home '', :controller => 'welcome'
10 10
11 11 map.signin 'login', :controller => 'account', :action => 'login'
12 12 map.signout 'logout', :controller => 'account', :action => 'logout'
13 13
14 14 map.connect 'roles/workflow/:id/:role_id/:tracker_id', :controller => 'roles', :action => 'workflow'
15 15 map.connect 'help/:ctrl/:page', :controller => 'help'
16 16
17 17 map.connect 'time_entries/:id/edit', :action => 'edit', :controller => 'timelog'
18 18 map.connect 'projects/:project_id/time_entries/new', :action => 'edit', :controller => 'timelog'
19 19 map.connect 'projects/:project_id/issues/:issue_id/time_entries/new', :action => 'edit', :controller => 'timelog'
20 20
21 21 map.with_options :controller => 'timelog' do |timelog|
22 22 timelog.connect 'projects/:project_id/time_entries', :action => 'details'
23 23
24 24 timelog.with_options :action => 'details', :conditions => {:method => :get} do |time_details|
25 25 time_details.connect 'time_entries'
26 26 time_details.connect 'time_entries.:format'
27 27 time_details.connect 'issues/:issue_id/time_entries'
28 28 time_details.connect 'issues/:issue_id/time_entries.:format'
29 29 time_details.connect 'projects/:project_id/time_entries.:format'
30 30 time_details.connect 'projects/:project_id/issues/:issue_id/time_entries'
31 31 time_details.connect 'projects/:project_id/issues/:issue_id/time_entries.:format'
32 32 end
33 33 timelog.connect 'projects/:project_id/time_entries/report', :action => 'report'
34 34 timelog.with_options :action => 'report',:conditions => {:method => :get} do |time_report|
35 35 time_report.connect 'time_entries/report'
36 36 time_report.connect 'time_entries/report.:format'
37 37 time_report.connect 'projects/:project_id/time_entries/report.:format'
38 38 end
39 39
40 40 timelog.with_options :action => 'edit', :conditions => {:method => :get} do |time_edit|
41 41 time_edit.connect 'issues/:issue_id/time_entries/new'
42 42 end
43 43
44 44 timelog.connect 'time_entries/:id/destroy', :action => 'destroy', :conditions => {:method => :post}
45 45 end
46 46
47 47 map.connect 'projects/:id/wiki', :controller => 'wikis', :action => 'edit', :conditions => {:method => :post}
48 48 map.connect 'projects/:id/wiki/destroy', :controller => 'wikis', :action => 'destroy', :conditions => {:method => :get}
49 49 map.connect 'projects/:id/wiki/destroy', :controller => 'wikis', :action => 'destroy', :conditions => {:method => :post}
50 50 map.with_options :controller => 'wiki' do |wiki_routes|
51 51 wiki_routes.with_options :conditions => {:method => :get} do |wiki_views|
52 52 wiki_views.connect 'projects/:id/wiki/:page', :action => 'special', :page => /page_index|date_index|export/i
53 53 wiki_views.connect 'projects/:id/wiki/:page', :action => 'index', :page => nil
54 54 wiki_views.connect 'projects/:id/wiki/:page/edit', :action => 'edit'
55 55 wiki_views.connect 'projects/:id/wiki/:page/rename', :action => 'rename'
56 56 wiki_views.connect 'projects/:id/wiki/:page/history', :action => 'history'
57 57 wiki_views.connect 'projects/:id/wiki/:page/diff/:version/vs/:version_from', :action => 'diff'
58 58 wiki_views.connect 'projects/:id/wiki/:page/annotate/:version', :action => 'annotate'
59 59 end
60 60
61 61 wiki_routes.connect 'projects/:id/wiki/:page/:action',
62 62 :action => /edit|rename|destroy|preview|protect/,
63 63 :conditions => {:method => :post}
64 64 end
65 65
66 66 map.with_options :controller => 'messages' do |messages_routes|
67 67 messages_routes.with_options :conditions => {:method => :get} do |messages_views|
68 68 messages_views.connect 'boards/:board_id/topics/new', :action => 'new'
69 69 messages_views.connect 'boards/:board_id/topics/:id', :action => 'show'
70 70 messages_views.connect 'boards/:board_id/topics/:id/edit', :action => 'edit'
71 71 end
72 72 messages_routes.with_options :conditions => {:method => :post} do |messages_actions|
73 73 messages_actions.connect 'boards/:board_id/topics/new', :action => 'new'
74 74 messages_actions.connect 'boards/:board_id/topics/:id/replies', :action => 'reply'
75 75 messages_actions.connect 'boards/:board_id/topics/:id/:action', :action => /edit|destroy/
76 76 end
77 77 end
78 78
79 79 map.with_options :controller => 'boards' do |board_routes|
80 80 board_routes.with_options :conditions => {:method => :get} do |board_views|
81 81 board_views.connect 'projects/:project_id/boards', :action => 'index'
82 82 board_views.connect 'projects/:project_id/boards/new', :action => 'new'
83 83 board_views.connect 'projects/:project_id/boards/:id', :action => 'show'
84 84 board_views.connect 'projects/:project_id/boards/:id.:format', :action => 'show'
85 85 board_views.connect 'projects/:project_id/boards/:id/edit', :action => 'edit'
86 86 end
87 87 board_routes.with_options :conditions => {:method => :post} do |board_actions|
88 88 board_actions.connect 'projects/:project_id/boards', :action => 'new'
89 89 board_actions.connect 'projects/:project_id/boards/:id/:action', :action => /edit|destroy/
90 90 end
91 91 end
92 92
93 93 map.with_options :controller => 'documents' do |document_routes|
94 94 document_routes.with_options :conditions => {:method => :get} do |document_views|
95 95 document_views.connect 'projects/:project_id/documents', :action => 'index'
96 96 document_views.connect 'projects/:project_id/documents/new', :action => 'new'
97 97 document_views.connect 'documents/:id', :action => 'show'
98 98 document_views.connect 'documents/:id/edit', :action => 'edit'
99 99 end
100 100 document_routes.with_options :conditions => {:method => :post} do |document_actions|
101 101 document_actions.connect 'projects/:project_id/documents', :action => 'new'
102 102 document_actions.connect 'documents/:id/:action', :action => /destroy|edit/
103 103 end
104 104 end
105 105
106 106 map.resources :issue_moves, :only => [:new, :create], :path_prefix => '/issues', :as => 'move'
107 107
108 108 # Misc issue routes. TODO: move into resources
109 109 map.auto_complete_issues '/issues/auto_complete', :controller => 'auto_completes', :action => 'issues'
110 110 map.preview_issue '/issues/preview/:id', :controller => 'previews', :action => 'issue' # TODO: would look nicer as /issues/:id/preview
111 111 map.issues_context_menu '/issues/context_menu', :controller => 'context_menus', :action => 'issues'
112 112 map.issue_changes '/issues/changes', :controller => 'journals', :action => 'index'
113 113 map.bulk_edit_issue 'issues/bulk_edit', :controller => 'issues', :action => 'bulk_edit', :conditions => { :method => :get }
114 114 map.bulk_update_issue 'issues/bulk_edit', :controller => 'issues', :action => 'bulk_update', :conditions => { :method => :post }
115 115 map.quoted_issue '/issues/:id/quoted', :controller => 'journals', :action => 'new', :id => /\d+/, :conditions => { :method => :post }
116 116 map.connect '/issues/:id/destroy', :controller => 'issues', :action => 'destroy', :conditions => { :method => :post } # legacy
117 117
118 118 map.resource :gantt, :path_prefix => '/issues', :controller => 'gantts', :only => [:show, :update]
119 119 map.resource :gantt, :path_prefix => '/projects/:project_id/issues', :controller => 'gantts', :only => [:show, :update]
120 120 map.resource :calendar, :path_prefix => '/issues', :controller => 'calendars', :only => [:show, :update]
121 121 map.resource :calendar, :path_prefix => '/projects/:project_id/issues', :controller => 'calendars', :only => [:show, :update]
122 122
123 123 map.with_options :controller => 'reports', :conditions => {:method => :get} do |reports|
124 124 reports.connect 'projects/:id/issues/report', :action => 'issue_report'
125 125 reports.connect 'projects/:id/issues/report/:detail', :action => 'issue_report_details'
126 126 end
127 127
128 128 # Following two routes conflict with the resources because #index allows POST
129 129 map.connect '/issues', :controller => 'issues', :action => 'index', :conditions => { :method => :post }
130 130 map.connect '/issues/create', :controller => 'issues', :action => 'index', :conditions => { :method => :post }
131 131
132 132 map.resources :issues, :member => { :edit => :post }, :collection => {}
133 133 map.resources :issues, :path_prefix => '/projects/:project_id', :collection => { :create => :post }
134 134
135 135 map.with_options :controller => 'issue_relations', :conditions => {:method => :post} do |relations|
136 136 relations.connect 'issues/:issue_id/relations/:id', :action => 'new'
137 137 relations.connect 'issues/:issue_id/relations/:id/destroy', :action => 'destroy'
138 138 end
139 139
140 140 map.with_options :controller => 'news' do |news_routes|
141 141 news_routes.with_options :conditions => {:method => :get} do |news_views|
142 142 news_views.connect 'news', :action => 'index'
143 143 news_views.connect 'projects/:project_id/news', :action => 'index'
144 144 news_views.connect 'projects/:project_id/news.:format', :action => 'index'
145 145 news_views.connect 'news.:format', :action => 'index'
146 146 news_views.connect 'projects/:project_id/news/new', :action => 'new'
147 147 news_views.connect 'news/:id', :action => 'show'
148 148 news_views.connect 'news/:id/edit', :action => 'edit'
149 149 end
150 150 news_routes.with_options do |news_actions|
151 151 news_actions.connect 'projects/:project_id/news', :action => 'new'
152 152 news_actions.connect 'news/:id/edit', :action => 'edit'
153 153 news_actions.connect 'news/:id/destroy', :action => 'destroy'
154 154 end
155 155 end
156 156
157 157 map.connect 'projects/:id/members/new', :controller => 'members', :action => 'new'
158 158
159 159 map.with_options :controller => 'users' do |users|
160 160 users.with_options :conditions => {:method => :get} do |user_views|
161 161 user_views.connect 'users', :action => 'index'
162 162 user_views.connect 'users/:id', :action => 'show', :id => /\d+/
163 163 user_views.connect 'users/new', :action => 'add'
164 164 user_views.connect 'users/:id/edit/:tab', :action => 'edit', :tab => nil
165 165 end
166 166 users.with_options :conditions => {:method => :post} do |user_actions|
167 167 user_actions.connect 'users', :action => 'add'
168 168 user_actions.connect 'users/new', :action => 'add'
169 169 user_actions.connect 'users/:id/edit', :action => 'edit'
170 170 user_actions.connect 'users/:id/memberships', :action => 'edit_membership'
171 171 user_actions.connect 'users/:id/memberships/:membership_id', :action => 'edit_membership'
172 172 user_actions.connect 'users/:id/memberships/:membership_id/destroy', :action => 'destroy_membership'
173 173 end
174 174 end
175 175
176 176 map.with_options :controller => 'projects' do |projects|
177 177 projects.with_options :conditions => {:method => :get} do |project_views|
178 178 project_views.connect 'projects', :action => 'index'
179 179 project_views.connect 'projects.:format', :action => 'index'
180 180 project_views.connect 'projects/new', :action => 'add'
181 181 project_views.connect 'projects/:id', :action => 'show'
182 182 project_views.connect 'projects/:id.:format', :action => 'show'
183 project_views.connect 'projects/:id/:action', :action => /roadmap|destroy|settings/
183 project_views.connect 'projects/:id/:action', :action => /destroy|settings/
184 184 project_views.connect 'projects/:id/files', :action => 'list_files'
185 185 project_views.connect 'projects/:id/files/new', :action => 'add_file'
186 186 project_views.connect 'projects/:id/settings/:tab', :action => 'settings'
187 187 project_views.connect 'projects/:project_id/issues/:copy_from/copy', :controller => 'issues', :action => 'new'
188 188 end
189 189
190 190 projects.with_options :controller => 'activities', :action => 'index', :conditions => {:method => :get} do |activity|
191 191 activity.connect 'projects/:id/activity'
192 192 activity.connect 'projects/:id/activity.:format'
193 193 activity.connect 'activity', :id => nil
194 194 activity.connect 'activity.:format', :id => nil
195 195 end
196 196
197 197 projects.with_options :conditions => {:method => :post} do |project_actions|
198 198 project_actions.connect 'projects/new', :action => 'add'
199 199 project_actions.connect 'projects', :action => 'add'
200 200 project_actions.connect 'projects.:format', :action => 'add', :format => /xml/
201 201 project_actions.connect 'projects/:id/:action', :action => /edit|destroy|archive|unarchive/
202 202 project_actions.connect 'projects/:id/files/new', :action => 'add_file'
203 203 project_actions.connect 'projects/:id/activities/save', :action => 'save_activities'
204 204 end
205 205
206 206 projects.with_options :conditions => {:method => :put} do |project_actions|
207 207 project_actions.conditions 'projects/:id.:format', :action => 'edit', :format => /xml/
208 208 end
209 209
210 210 projects.with_options :conditions => {:method => :delete} do |project_actions|
211 211 project_actions.conditions 'projects/:id.:format', :action => 'destroy', :format => /xml/
212 212 project_actions.conditions 'projects/:id/reset_activities', :action => 'reset_activities'
213 213 end
214 214 end
215 215
216 216 map.with_options :controller => 'versions' do |versions|
217 217 versions.connect 'projects/:project_id/versions/new', :action => 'new'
218 versions.connect 'projects/:project_id/roadmap', :action => 'index'
218 219 versions.with_options :conditions => {:method => :post} do |version_actions|
219 220 version_actions.connect 'projects/:project_id/versions/close_completed', :action => 'close_completed'
220 221 end
221 222 end
222 223
223 224 map.with_options :controller => 'issue_categories' do |categories|
224 225 categories.connect 'projects/:project_id/issue_categories/new', :action => 'new'
225 226 end
226 227
227 228 map.with_options :controller => 'repositories' do |repositories|
228 229 repositories.with_options :conditions => {:method => :get} do |repository_views|
229 230 repository_views.connect 'projects/:id/repository', :action => 'show'
230 231 repository_views.connect 'projects/:id/repository/edit', :action => 'edit'
231 232 repository_views.connect 'projects/:id/repository/statistics', :action => 'stats'
232 233 repository_views.connect 'projects/:id/repository/revisions', :action => 'revisions'
233 234 repository_views.connect 'projects/:id/repository/revisions.:format', :action => 'revisions'
234 235 repository_views.connect 'projects/:id/repository/revisions/:rev', :action => 'revision'
235 236 repository_views.connect 'projects/:id/repository/revisions/:rev/diff', :action => 'diff'
236 237 repository_views.connect 'projects/:id/repository/revisions/:rev/diff.:format', :action => 'diff'
237 238 repository_views.connect 'projects/:id/repository/revisions/:rev/raw/*path', :action => 'entry', :format => 'raw', :requirements => { :rev => /[a-z0-9\.\-_]+/ }
238 239 repository_views.connect 'projects/:id/repository/revisions/:rev/:action/*path', :requirements => { :rev => /[a-z0-9\.\-_]+/ }
239 240 repository_views.connect 'projects/:id/repository/raw/*path', :action => 'entry', :format => 'raw'
240 241 # TODO: why the following route is required?
241 242 repository_views.connect 'projects/:id/repository/entry/*path', :action => 'entry'
242 243 repository_views.connect 'projects/:id/repository/:action/*path'
243 244 end
244 245
245 246 repositories.connect 'projects/:id/repository/:action', :conditions => {:method => :post}
246 247 end
247 248
248 249 map.connect 'attachments/:id', :controller => 'attachments', :action => 'show', :id => /\d+/
249 250 map.connect 'attachments/:id/:filename', :controller => 'attachments', :action => 'show', :id => /\d+/, :filename => /.*/
250 251 map.connect 'attachments/download/:id/:filename', :controller => 'attachments', :action => 'download', :id => /\d+/, :filename => /.*/
251 252
252 253 map.resources :groups
253 254
254 255 #left old routes at the bottom for backwards compat
255 256 map.connect 'projects/:project_id/issues/:action', :controller => 'issues'
256 257 map.connect 'projects/:project_id/documents/:action', :controller => 'documents'
257 258 map.connect 'projects/:project_id/boards/:action/:id', :controller => 'boards'
258 259 map.connect 'boards/:board_id/topics/:action/:id', :controller => 'messages'
259 260 map.connect 'wiki/:id/:page/:action', :page => nil, :controller => 'wiki'
260 261 map.connect 'issues/:issue_id/relations/:action/:id', :controller => 'issue_relations'
261 262 map.connect 'projects/:project_id/news/:action', :controller => 'news'
262 263 map.connect 'projects/:project_id/timelog/:action/:id', :controller => 'timelog', :project_id => /.+/
263 264 map.with_options :controller => 'repositories' do |omap|
264 265 omap.repositories_show 'repositories/browse/:id/*path', :action => 'browse'
265 266 omap.repositories_changes 'repositories/changes/:id/*path', :action => 'changes'
266 267 omap.repositories_diff 'repositories/diff/:id/*path', :action => 'diff'
267 268 omap.repositories_entry 'repositories/entry/:id/*path', :action => 'entry'
268 269 omap.repositories_entry 'repositories/annotate/:id/*path', :action => 'annotate'
269 270 omap.connect 'repositories/revision/:id/:rev', :action => 'revision'
270 271 end
271 272
272 273 map.with_options :controller => 'sys' do |sys|
273 274 sys.connect 'sys/projects.:format', :action => 'projects', :conditions => {:method => :get}
274 275 sys.connect 'sys/projects/:id/repository.:format', :action => 'create_project_repository', :conditions => {:method => :post}
275 276 end
276 277
277 278 # Install the default route as the lowest priority.
278 279 map.connect ':controller/:action/:id'
279 280 map.connect 'robots.txt', :controller => 'welcome', :action => 'robots'
280 281 # Used for OpenID
281 282 map.root :controller => 'account', :action => 'login'
282 283 end
@@ -1,231 +1,230
1 1 require 'redmine/access_control'
2 2 require 'redmine/menu_manager'
3 3 require 'redmine/activity'
4 4 require 'redmine/search'
5 5 require 'redmine/custom_field_format'
6 6 require 'redmine/mime_type'
7 7 require 'redmine/core_ext'
8 8 require 'redmine/themes'
9 9 require 'redmine/hook'
10 10 require 'redmine/plugin'
11 11 require 'redmine/wiki_formatting'
12 12 require 'redmine/scm/base'
13 13
14 14 begin
15 15 require_library_or_gem 'RMagick' unless Object.const_defined?(:Magick)
16 16 rescue LoadError
17 17 # RMagick is not available
18 18 end
19 19
20 20 if RUBY_VERSION < '1.9'
21 21 require 'faster_csv'
22 22 else
23 23 require 'csv'
24 24 FCSV = CSV
25 25 end
26 26
27 27 Redmine::Scm::Base.add "Subversion"
28 28 Redmine::Scm::Base.add "Darcs"
29 29 Redmine::Scm::Base.add "Mercurial"
30 30 Redmine::Scm::Base.add "Cvs"
31 31 Redmine::Scm::Base.add "Bazaar"
32 32 Redmine::Scm::Base.add "Git"
33 33 Redmine::Scm::Base.add "Filesystem"
34 34
35 35 Redmine::CustomFieldFormat.map do |fields|
36 36 fields.register Redmine::CustomFieldFormat.new('string', :label => :label_string, :order => 1)
37 37 fields.register Redmine::CustomFieldFormat.new('text', :label => :label_text, :order => 2)
38 38 fields.register Redmine::CustomFieldFormat.new('int', :label => :label_integer, :order => 3)
39 39 fields.register Redmine::CustomFieldFormat.new('float', :label => :label_float, :order => 4)
40 40 fields.register Redmine::CustomFieldFormat.new('list', :label => :label_list, :order => 5)
41 41 fields.register Redmine::CustomFieldFormat.new('date', :label => :label_date, :order => 6)
42 42 fields.register Redmine::CustomFieldFormat.new('bool', :label => :label_boolean, :order => 7)
43 43 end
44 44
45 45 # Permissions
46 46 Redmine::AccessControl.map do |map|
47 47 map.permission :view_project, {:projects => [:show], :activities => [:index]}, :public => true
48 48 map.permission :search_project, {:search => :index}, :public => true
49 49 map.permission :add_project, {:projects => :add}, :require => :loggedin
50 50 map.permission :edit_project, {:projects => [:settings, :edit]}, :require => :member
51 51 map.permission :select_project_modules, {:projects => :modules}, :require => :member
52 52 map.permission :manage_members, {:projects => :settings, :members => [:new, :edit, :destroy, :autocomplete_for_member]}, :require => :member
53 53 map.permission :manage_versions, {:projects => :settings, :versions => [:new, :edit, :close_completed, :destroy]}, :require => :member
54 54 map.permission :add_subprojects, {:projects => :add}, :require => :member
55 55
56 56 map.project_module :issue_tracking do |map|
57 57 # Issue categories
58 58 map.permission :manage_categories, {:projects => :settings, :issue_categories => [:new, :edit, :destroy]}, :require => :member
59 59 # Issues
60 map.permission :view_issues, {:projects => :roadmap,
61 :issues => [:index, :show],
60 map.permission :view_issues, {:issues => [:index, :show],
62 61 :auto_complete => [:issues],
63 62 :context_menus => [:issues],
64 :versions => [:show, :status_by],
63 :versions => [:index, :show, :status_by],
65 64 :journals => :index,
66 65 :queries => :index,
67 66 :reports => [:issue_report, :issue_report_details]}
68 67 map.permission :add_issues, {:issues => [:new, :create, :update_form]}
69 68 map.permission :edit_issues, {:issues => [:edit, :update, :bulk_edit, :bulk_update, :update_form], :journals => [:new]}
70 69 map.permission :manage_issue_relations, {:issue_relations => [:new, :destroy]}
71 70 map.permission :manage_subtasks, {}
72 71 map.permission :add_issue_notes, {:issues => [:edit, :update], :journals => [:new]}
73 72 map.permission :edit_issue_notes, {:journals => :edit}, :require => :loggedin
74 73 map.permission :edit_own_issue_notes, {:journals => :edit}, :require => :loggedin
75 74 map.permission :move_issues, {:issue_moves => [:new, :create]}, :require => :loggedin
76 75 map.permission :delete_issues, {:issues => :destroy}, :require => :member
77 76 # Queries
78 77 map.permission :manage_public_queries, {:queries => [:new, :edit, :destroy]}, :require => :member
79 78 map.permission :save_queries, {:queries => [:new, :edit, :destroy]}, :require => :loggedin
80 79 # Watchers
81 80 map.permission :view_issue_watchers, {}
82 81 map.permission :add_issue_watchers, {:watchers => :new}
83 82 map.permission :delete_issue_watchers, {:watchers => :destroy}
84 83 end
85 84
86 85 map.project_module :time_tracking do |map|
87 86 map.permission :log_time, {:timelog => :edit}, :require => :loggedin
88 87 map.permission :view_time_entries, :timelog => [:details, :report]
89 88 map.permission :edit_time_entries, {:timelog => [:edit, :destroy]}, :require => :member
90 89 map.permission :edit_own_time_entries, {:timelog => [:edit, :destroy]}, :require => :loggedin
91 90 map.permission :manage_project_activities, {:projects => [:save_activities, :reset_activities]}, :require => :member
92 91 end
93 92
94 93 map.project_module :news do |map|
95 94 map.permission :manage_news, {:news => [:new, :edit, :destroy, :destroy_comment]}, :require => :member
96 95 map.permission :view_news, {:news => [:index, :show]}, :public => true
97 96 map.permission :comment_news, {:news => :add_comment}
98 97 end
99 98
100 99 map.project_module :documents do |map|
101 100 map.permission :manage_documents, {:documents => [:new, :edit, :destroy, :add_attachment]}, :require => :loggedin
102 101 map.permission :view_documents, :documents => [:index, :show, :download]
103 102 end
104 103
105 104 map.project_module :files do |map|
106 105 map.permission :manage_files, {:projects => :add_file}, :require => :loggedin
107 106 map.permission :view_files, :projects => :list_files, :versions => :download
108 107 end
109 108
110 109 map.project_module :wiki do |map|
111 110 map.permission :manage_wiki, {:wikis => [:edit, :destroy]}, :require => :member
112 111 map.permission :rename_wiki_pages, {:wiki => :rename}, :require => :member
113 112 map.permission :delete_wiki_pages, {:wiki => :destroy}, :require => :member
114 113 map.permission :view_wiki_pages, :wiki => [:index, :special]
115 114 map.permission :export_wiki_pages, {}
116 115 map.permission :view_wiki_edits, :wiki => [:history, :diff, :annotate]
117 116 map.permission :edit_wiki_pages, :wiki => [:edit, :preview, :add_attachment]
118 117 map.permission :delete_wiki_pages_attachments, {}
119 118 map.permission :protect_wiki_pages, {:wiki => :protect}, :require => :member
120 119 end
121 120
122 121 map.project_module :repository do |map|
123 122 map.permission :manage_repository, {:repositories => [:edit, :committers, :destroy]}, :require => :member
124 123 map.permission :browse_repository, :repositories => [:show, :browse, :entry, :annotate, :changes, :diff, :stats, :graph]
125 124 map.permission :view_changesets, :repositories => [:show, :revisions, :revision]
126 125 map.permission :commit_access, {}
127 126 end
128 127
129 128 map.project_module :boards do |map|
130 129 map.permission :manage_boards, {:boards => [:new, :edit, :destroy]}, :require => :member
131 130 map.permission :view_messages, {:boards => [:index, :show], :messages => [:show]}, :public => true
132 131 map.permission :add_messages, {:messages => [:new, :reply, :quote]}
133 132 map.permission :edit_messages, {:messages => :edit}, :require => :member
134 133 map.permission :edit_own_messages, {:messages => :edit}, :require => :loggedin
135 134 map.permission :delete_messages, {:messages => :destroy}, :require => :member
136 135 map.permission :delete_own_messages, {:messages => :destroy}, :require => :loggedin
137 136 end
138 137
139 138 map.project_module :calendar do |map|
140 139 map.permission :view_calendar, :calendars => [:show, :update]
141 140 end
142 141
143 142 map.project_module :gantt do |map|
144 143 map.permission :view_gantt, :gantts => [:show, :update]
145 144 end
146 145 end
147 146
148 147 Redmine::MenuManager.map :top_menu do |menu|
149 148 menu.push :home, :home_path
150 149 menu.push :my_page, { :controller => 'my', :action => 'page' }, :if => Proc.new { User.current.logged? }
151 150 menu.push :projects, { :controller => 'projects', :action => 'index' }, :caption => :label_project_plural
152 151 menu.push :administration, { :controller => 'admin', :action => 'index' }, :if => Proc.new { User.current.admin? }, :last => true
153 152 menu.push :help, Redmine::Info.help_url, :last => true
154 153 end
155 154
156 155 Redmine::MenuManager.map :account_menu do |menu|
157 156 menu.push :login, :signin_path, :if => Proc.new { !User.current.logged? }
158 157 menu.push :register, { :controller => 'account', :action => 'register' }, :if => Proc.new { !User.current.logged? && Setting.self_registration? }
159 158 menu.push :my_account, { :controller => 'my', :action => 'account' }, :if => Proc.new { User.current.logged? }
160 159 menu.push :logout, :signout_path, :if => Proc.new { User.current.logged? }
161 160 end
162 161
163 162 Redmine::MenuManager.map :application_menu do |menu|
164 163 # Empty
165 164 end
166 165
167 166 Redmine::MenuManager.map :admin_menu do |menu|
168 167 menu.push :projects, {:controller => 'admin', :action => 'projects'}, :caption => :label_project_plural
169 168 menu.push :users, {:controller => 'users'}, :caption => :label_user_plural
170 169 menu.push :groups, {:controller => 'groups'}, :caption => :label_group_plural
171 170 menu.push :roles, {:controller => 'roles'}, :caption => :label_role_and_permissions
172 171 menu.push :trackers, {:controller => 'trackers'}, :caption => :label_tracker_plural
173 172 menu.push :issue_statuses, {:controller => 'issue_statuses'}, :caption => :label_issue_status_plural,
174 173 :html => {:class => 'issue_statuses'}
175 174 menu.push :workflows, {:controller => 'workflows', :action => 'edit'}, :caption => :label_workflow
176 175 menu.push :custom_fields, {:controller => 'custom_fields'}, :caption => :label_custom_field_plural,
177 176 :html => {:class => 'custom_fields'}
178 177 menu.push :enumerations, {:controller => 'enumerations'}
179 178 menu.push :settings, {:controller => 'settings'}
180 179 menu.push :ldap_authentication, {:controller => 'ldap_auth_sources', :action => 'index'},
181 180 :html => {:class => 'server_authentication'}
182 181 menu.push :plugins, {:controller => 'admin', :action => 'plugins'}, :last => true
183 182 menu.push :info, {:controller => 'admin', :action => 'info'}, :caption => :label_information_plural, :last => true
184 183 end
185 184
186 185 Redmine::MenuManager.map :project_menu do |menu|
187 186 menu.push :overview, { :controller => 'projects', :action => 'show' }
188 187 menu.push :activity, { :controller => 'activities', :action => 'index' }
189 menu.push :roadmap, { :controller => 'projects', :action => 'roadmap' },
188 menu.push :roadmap, { :controller => 'versions', :action => 'index' }, :param => :project_id,
190 189 :if => Proc.new { |p| p.shared_versions.any? }
191 190 menu.push :issues, { :controller => 'issues', :action => 'index' }, :param => :project_id, :caption => :label_issue_plural
192 191 menu.push :new_issue, { :controller => 'issues', :action => 'new' }, :param => :project_id, :caption => :label_issue_new,
193 192 :html => { :accesskey => Redmine::AccessKeys.key_for(:new_issue) }
194 193 menu.push :gantt, { :controller => 'gantts', :action => 'show' }, :param => :project_id, :caption => :label_gantt
195 194 menu.push :calendar, { :controller => 'calendars', :action => 'show' }, :param => :project_id, :caption => :label_calendar
196 195 menu.push :news, { :controller => 'news', :action => 'index' }, :param => :project_id, :caption => :label_news_plural
197 196 menu.push :documents, { :controller => 'documents', :action => 'index' }, :param => :project_id, :caption => :label_document_plural
198 197 menu.push :wiki, { :controller => 'wiki', :action => 'index', :page => nil },
199 198 :if => Proc.new { |p| p.wiki && !p.wiki.new_record? }
200 199 menu.push :boards, { :controller => 'boards', :action => 'index', :id => nil }, :param => :project_id,
201 200 :if => Proc.new { |p| p.boards.any? }, :caption => :label_board_plural
202 201 menu.push :files, { :controller => 'projects', :action => 'list_files' }, :caption => :label_file_plural
203 202 menu.push :repository, { :controller => 'repositories', :action => 'show' },
204 203 :if => Proc.new { |p| p.repository && !p.repository.new_record? }
205 204 menu.push :settings, { :controller => 'projects', :action => 'settings' }, :last => true
206 205 end
207 206
208 207 Redmine::Activity.map do |activity|
209 208 activity.register :issues, :class_name => %w(Issue Journal)
210 209 activity.register :changesets
211 210 activity.register :news
212 211 activity.register :documents, :class_name => %w(Document Attachment)
213 212 activity.register :files, :class_name => 'Attachment'
214 213 activity.register :wiki_edits, :class_name => 'WikiContent::Version', :default => false
215 214 activity.register :messages, :default => false
216 215 activity.register :time_entries, :default => false
217 216 end
218 217
219 218 Redmine::Search.map do |search|
220 219 search.register :issues
221 220 search.register :news
222 221 search.register :documents
223 222 search.register :changesets
224 223 search.register :wiki_pages
225 224 search.register :messages
226 225 search.register :projects
227 226 end
228 227
229 228 Redmine::WikiFormatting.map do |format|
230 229 format.register :textile, Redmine::WikiFormatting::Textile::Formatter, Redmine::WikiFormatting::Textile::Helper
231 230 end
@@ -1,663 +1,631
1 1 # Redmine - project management software
2 2 # Copyright (C) 2006-2008 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 require File.dirname(__FILE__) + '/../test_helper'
19 19 require 'projects_controller'
20 20
21 21 # Re-raise errors caught by the controller.
22 22 class ProjectsController; def rescue_action(e) raise e end; end
23 23
24 24 class ProjectsControllerTest < ActionController::TestCase
25 25 fixtures :projects, :versions, :users, :roles, :members, :member_roles, :issues, :journals, :journal_details,
26 26 :trackers, :projects_trackers, :issue_statuses, :enabled_modules, :enumerations, :boards, :messages,
27 27 :attachments, :custom_fields, :custom_values, :time_entries
28 28
29 29 def setup
30 30 @controller = ProjectsController.new
31 31 @request = ActionController::TestRequest.new
32 32 @response = ActionController::TestResponse.new
33 33 @request.session[:user_id] = nil
34 34 Setting.default_language = 'en'
35 35 end
36 36
37 37 def test_index
38 38 get :index
39 39 assert_response :success
40 40 assert_template 'index'
41 41 assert_not_nil assigns(:projects)
42 42
43 43 assert_tag :ul, :child => {:tag => 'li',
44 44 :descendant => {:tag => 'a', :content => 'eCookbook'},
45 45 :child => { :tag => 'ul',
46 46 :descendant => { :tag => 'a',
47 47 :content => 'Child of private child'
48 48 }
49 49 }
50 50 }
51 51
52 52 assert_no_tag :a, :content => /Private child of eCookbook/
53 53 end
54 54
55 55 def test_index_atom
56 56 get :index, :format => 'atom'
57 57 assert_response :success
58 58 assert_template 'common/feed.atom.rxml'
59 59 assert_select 'feed>title', :text => 'Redmine: Latest projects'
60 60 assert_select 'feed>entry', :count => Project.count(:conditions => Project.visible_by(User.current))
61 61 end
62 62
63 63 context "#index" do
64 64 context "by non-admin user with view_time_entries permission" do
65 65 setup do
66 66 @request.session[:user_id] = 3
67 67 end
68 68 should "show overall spent time link" do
69 69 get :index
70 70 assert_template 'index'
71 71 assert_tag :a, :attributes => {:href => '/time_entries'}
72 72 end
73 73 end
74 74
75 75 context "by non-admin user without view_time_entries permission" do
76 76 setup do
77 77 Role.find(2).remove_permission! :view_time_entries
78 78 Role.non_member.remove_permission! :view_time_entries
79 79 Role.anonymous.remove_permission! :view_time_entries
80 80 @request.session[:user_id] = 3
81 81 end
82 82 should "not show overall spent time link" do
83 83 get :index
84 84 assert_template 'index'
85 85 assert_no_tag :a, :attributes => {:href => '/time_entries'}
86 86 end
87 87 end
88 88 end
89 89
90 90 context "#add" do
91 91 context "by admin user" do
92 92 setup do
93 93 @request.session[:user_id] = 1
94 94 end
95 95
96 96 should "accept get" do
97 97 get :add
98 98 assert_response :success
99 99 assert_template 'add'
100 100 end
101 101
102 102 should "accept post" do
103 103 post :add, :project => { :name => "blog",
104 104 :description => "weblog",
105 105 :identifier => "blog",
106 106 :is_public => 1,
107 107 :custom_field_values => { '3' => 'Beta' }
108 108 }
109 109 assert_redirected_to '/projects/blog/settings'
110 110
111 111 project = Project.find_by_name('blog')
112 112 assert_kind_of Project, project
113 113 assert_equal 'weblog', project.description
114 114 assert_equal true, project.is_public?
115 115 assert_nil project.parent
116 116 end
117 117
118 118 should "accept post with parent" do
119 119 post :add, :project => { :name => "blog",
120 120 :description => "weblog",
121 121 :identifier => "blog",
122 122 :is_public => 1,
123 123 :custom_field_values => { '3' => 'Beta' },
124 124 :parent_id => 1
125 125 }
126 126 assert_redirected_to '/projects/blog/settings'
127 127
128 128 project = Project.find_by_name('blog')
129 129 assert_kind_of Project, project
130 130 assert_equal Project.find(1), project.parent
131 131 end
132 132 end
133 133
134 134 context "by non-admin user with add_project permission" do
135 135 setup do
136 136 Role.non_member.add_permission! :add_project
137 137 @request.session[:user_id] = 9
138 138 end
139 139
140 140 should "accept get" do
141 141 get :add
142 142 assert_response :success
143 143 assert_template 'add'
144 144 assert_no_tag :select, :attributes => {:name => 'project[parent_id]'}
145 145 end
146 146
147 147 should "accept post" do
148 148 post :add, :project => { :name => "blog",
149 149 :description => "weblog",
150 150 :identifier => "blog",
151 151 :is_public => 1,
152 152 :custom_field_values => { '3' => 'Beta' }
153 153 }
154 154
155 155 assert_redirected_to '/projects/blog/settings'
156 156
157 157 project = Project.find_by_name('blog')
158 158 assert_kind_of Project, project
159 159 assert_equal 'weblog', project.description
160 160 assert_equal true, project.is_public?
161 161
162 162 # User should be added as a project member
163 163 assert User.find(9).member_of?(project)
164 164 assert_equal 1, project.members.size
165 165 end
166 166
167 167 should "fail with parent_id" do
168 168 assert_no_difference 'Project.count' do
169 169 post :add, :project => { :name => "blog",
170 170 :description => "weblog",
171 171 :identifier => "blog",
172 172 :is_public => 1,
173 173 :custom_field_values => { '3' => 'Beta' },
174 174 :parent_id => 1
175 175 }
176 176 end
177 177 assert_response :success
178 178 project = assigns(:project)
179 179 assert_kind_of Project, project
180 180 assert_not_nil project.errors.on(:parent_id)
181 181 end
182 182 end
183 183
184 184 context "by non-admin user with add_subprojects permission" do
185 185 setup do
186 186 Role.find(1).remove_permission! :add_project
187 187 Role.find(1).add_permission! :add_subprojects
188 188 @request.session[:user_id] = 2
189 189 end
190 190
191 191 should "accept get" do
192 192 get :add, :parent_id => 'ecookbook'
193 193 assert_response :success
194 194 assert_template 'add'
195 195 # parent project selected
196 196 assert_tag :select, :attributes => {:name => 'project[parent_id]'},
197 197 :child => {:tag => 'option', :attributes => {:value => '1', :selected => 'selected'}}
198 198 # no empty value
199 199 assert_no_tag :select, :attributes => {:name => 'project[parent_id]'},
200 200 :child => {:tag => 'option', :attributes => {:value => ''}}
201 201 end
202 202
203 203 should "accept post with parent_id" do
204 204 post :add, :project => { :name => "blog",
205 205 :description => "weblog",
206 206 :identifier => "blog",
207 207 :is_public => 1,
208 208 :custom_field_values => { '3' => 'Beta' },
209 209 :parent_id => 1
210 210 }
211 211 assert_redirected_to '/projects/blog/settings'
212 212 project = Project.find_by_name('blog')
213 213 end
214 214
215 215 should "fail without parent_id" do
216 216 assert_no_difference 'Project.count' do
217 217 post :add, :project => { :name => "blog",
218 218 :description => "weblog",
219 219 :identifier => "blog",
220 220 :is_public => 1,
221 221 :custom_field_values => { '3' => 'Beta' }
222 222 }
223 223 end
224 224 assert_response :success
225 225 project = assigns(:project)
226 226 assert_kind_of Project, project
227 227 assert_not_nil project.errors.on(:parent_id)
228 228 end
229 229
230 230 should "fail with unauthorized parent_id" do
231 231 assert !User.find(2).member_of?(Project.find(6))
232 232 assert_no_difference 'Project.count' do
233 233 post :add, :project => { :name => "blog",
234 234 :description => "weblog",
235 235 :identifier => "blog",
236 236 :is_public => 1,
237 237 :custom_field_values => { '3' => 'Beta' },
238 238 :parent_id => 6
239 239 }
240 240 end
241 241 assert_response :success
242 242 project = assigns(:project)
243 243 assert_kind_of Project, project
244 244 assert_not_nil project.errors.on(:parent_id)
245 245 end
246 246 end
247 247 end
248 248
249 249 def test_show_by_id
250 250 get :show, :id => 1
251 251 assert_response :success
252 252 assert_template 'show'
253 253 assert_not_nil assigns(:project)
254 254 end
255 255
256 256 def test_show_by_identifier
257 257 get :show, :id => 'ecookbook'
258 258 assert_response :success
259 259 assert_template 'show'
260 260 assert_not_nil assigns(:project)
261 261 assert_equal Project.find_by_identifier('ecookbook'), assigns(:project)
262 262 end
263 263
264 264 def test_show_should_not_fail_when_custom_values_are_nil
265 265 project = Project.find_by_identifier('ecookbook')
266 266 project.custom_values.first.update_attribute(:value, nil)
267 267 get :show, :id => 'ecookbook'
268 268 assert_response :success
269 269 assert_template 'show'
270 270 assert_not_nil assigns(:project)
271 271 assert_equal Project.find_by_identifier('ecookbook'), assigns(:project)
272 272 end
273 273
274 274 def test_private_subprojects_hidden
275 275 get :show, :id => 'ecookbook'
276 276 assert_response :success
277 277 assert_template 'show'
278 278 assert_no_tag :tag => 'a', :content => /Private child/
279 279 end
280 280
281 281 def test_private_subprojects_visible
282 282 @request.session[:user_id] = 2 # manager who is a member of the private subproject
283 283 get :show, :id => 'ecookbook'
284 284 assert_response :success
285 285 assert_template 'show'
286 286 assert_tag :tag => 'a', :content => /Private child/
287 287 end
288 288
289 289 def test_settings
290 290 @request.session[:user_id] = 2 # manager
291 291 get :settings, :id => 1
292 292 assert_response :success
293 293 assert_template 'settings'
294 294 end
295 295
296 296 def test_edit
297 297 @request.session[:user_id] = 2 # manager
298 298 post :edit, :id => 1, :project => {:name => 'Test changed name',
299 299 :issue_custom_field_ids => ['']}
300 300 assert_redirected_to 'projects/ecookbook/settings'
301 301 project = Project.find(1)
302 302 assert_equal 'Test changed name', project.name
303 303 end
304 304
305 305 def test_get_destroy
306 306 @request.session[:user_id] = 1 # admin
307 307 get :destroy, :id => 1
308 308 assert_response :success
309 309 assert_template 'destroy'
310 310 assert_not_nil Project.find_by_id(1)
311 311 end
312 312
313 313 def test_post_destroy
314 314 @request.session[:user_id] = 1 # admin
315 315 post :destroy, :id => 1, :confirm => 1
316 316 assert_redirected_to 'admin/projects'
317 317 assert_nil Project.find_by_id(1)
318 318 end
319 319
320 320 def test_add_file
321 321 set_tmp_attachments_directory
322 322 @request.session[:user_id] = 2
323 323 Setting.notified_events = ['file_added']
324 324 ActionMailer::Base.deliveries.clear
325 325
326 326 assert_difference 'Attachment.count' do
327 327 post :add_file, :id => 1, :version_id => '',
328 328 :attachments => {'1' => {'file' => uploaded_test_file('testfile.txt', 'text/plain')}}
329 329 end
330 330 assert_redirected_to 'projects/ecookbook/files'
331 331 a = Attachment.find(:first, :order => 'created_on DESC')
332 332 assert_equal 'testfile.txt', a.filename
333 333 assert_equal Project.find(1), a.container
334 334
335 335 mail = ActionMailer::Base.deliveries.last
336 336 assert_kind_of TMail::Mail, mail
337 337 assert_equal "[eCookbook] New file", mail.subject
338 338 assert mail.body.include?('testfile.txt')
339 339 end
340 340
341 341 def test_add_version_file
342 342 set_tmp_attachments_directory
343 343 @request.session[:user_id] = 2
344 344 Setting.notified_events = ['file_added']
345 345
346 346 assert_difference 'Attachment.count' do
347 347 post :add_file, :id => 1, :version_id => '2',
348 348 :attachments => {'1' => {'file' => uploaded_test_file('testfile.txt', 'text/plain')}}
349 349 end
350 350 assert_redirected_to 'projects/ecookbook/files'
351 351 a = Attachment.find(:first, :order => 'created_on DESC')
352 352 assert_equal 'testfile.txt', a.filename
353 353 assert_equal Version.find(2), a.container
354 354 end
355 355
356 356 def test_list_files
357 357 get :list_files, :id => 1
358 358 assert_response :success
359 359 assert_template 'list_files'
360 360 assert_not_nil assigns(:containers)
361 361
362 362 # file attached to the project
363 363 assert_tag :a, :content => 'project_file.zip',
364 364 :attributes => { :href => '/attachments/download/8/project_file.zip' }
365 365
366 366 # file attached to a project's version
367 367 assert_tag :a, :content => 'version_file.zip',
368 368 :attributes => { :href => '/attachments/download/9/version_file.zip' }
369 369 end
370 370
371 def test_roadmap
372 get :roadmap, :id => 1
373 assert_response :success
374 assert_template 'roadmap'
375 assert_not_nil assigns(:versions)
376 # Version with no date set appears
377 assert assigns(:versions).include?(Version.find(3))
378 # Completed version doesn't appear
379 assert !assigns(:versions).include?(Version.find(1))
380 end
381
382 def test_roadmap_with_completed_versions
383 get :roadmap, :id => 1, :completed => 1
384 assert_response :success
385 assert_template 'roadmap'
386 assert_not_nil assigns(:versions)
387 # Version with no date set appears
388 assert assigns(:versions).include?(Version.find(3))
389 # Completed version appears
390 assert assigns(:versions).include?(Version.find(1))
391 end
392
393 def test_roadmap_showing_subprojects_versions
394 @subproject_version = Version.generate!(:project => Project.find(3))
395 get :roadmap, :id => 1, :with_subprojects => 1
396 assert_response :success
397 assert_template 'roadmap'
398 assert_not_nil assigns(:versions)
399
400 assert assigns(:versions).include?(Version.find(4)), "Shared version not found"
401 assert assigns(:versions).include?(@subproject_version), "Subproject version not found"
402 end
403 371 def test_archive
404 372 @request.session[:user_id] = 1 # admin
405 373 post :archive, :id => 1
406 374 assert_redirected_to 'admin/projects'
407 375 assert !Project.find(1).active?
408 376 end
409 377
410 378 def test_unarchive
411 379 @request.session[:user_id] = 1 # admin
412 380 Project.find(1).archive
413 381 post :unarchive, :id => 1
414 382 assert_redirected_to 'admin/projects'
415 383 assert Project.find(1).active?
416 384 end
417 385
418 386 def test_project_breadcrumbs_should_be_limited_to_3_ancestors
419 387 CustomField.delete_all
420 388 parent = nil
421 389 6.times do |i|
422 390 p = Project.create!(:name => "Breadcrumbs #{i}", :identifier => "breadcrumbs-#{i}")
423 391 p.set_parent!(parent)
424 392 get :show, :id => p
425 393 assert_tag :h1, :parent => { :attributes => {:id => 'header'}},
426 394 :children => { :count => [i, 3].min,
427 395 :only => { :tag => 'a' } }
428 396
429 397 parent = p
430 398 end
431 399 end
432 400
433 401 def test_copy_with_project
434 402 @request.session[:user_id] = 1 # admin
435 403 get :copy, :id => 1
436 404 assert_response :success
437 405 assert_template 'copy'
438 406 assert assigns(:project)
439 407 assert_equal Project.find(1).description, assigns(:project).description
440 408 assert_nil assigns(:project).id
441 409 end
442 410
443 411 def test_copy_without_project
444 412 @request.session[:user_id] = 1 # admin
445 413 get :copy
446 414 assert_response :redirect
447 415 assert_redirected_to :controller => 'admin', :action => 'projects'
448 416 end
449 417
450 418 def test_jump_should_redirect_to_active_tab
451 419 get :show, :id => 1, :jump => 'issues'
452 420 assert_redirected_to 'projects/ecookbook/issues'
453 421 end
454 422
455 423 def test_jump_should_not_redirect_to_inactive_tab
456 424 get :show, :id => 3, :jump => 'documents'
457 425 assert_response :success
458 426 assert_template 'show'
459 427 end
460 428
461 429 def test_jump_should_not_redirect_to_unknown_tab
462 430 get :show, :id => 3, :jump => 'foobar'
463 431 assert_response :success
464 432 assert_template 'show'
465 433 end
466 434
467 435 def test_reset_activities
468 436 @request.session[:user_id] = 2 # manager
469 437 project_activity = TimeEntryActivity.new({
470 438 :name => 'Project Specific',
471 439 :parent => TimeEntryActivity.find(:first),
472 440 :project => Project.find(1),
473 441 :active => true
474 442 })
475 443 assert project_activity.save
476 444 project_activity_two = TimeEntryActivity.new({
477 445 :name => 'Project Specific Two',
478 446 :parent => TimeEntryActivity.find(:last),
479 447 :project => Project.find(1),
480 448 :active => true
481 449 })
482 450 assert project_activity_two.save
483 451
484 452 delete :reset_activities, :id => 1
485 453 assert_response :redirect
486 454 assert_redirected_to 'projects/ecookbook/settings/activities'
487 455
488 456 assert_nil TimeEntryActivity.find_by_id(project_activity.id)
489 457 assert_nil TimeEntryActivity.find_by_id(project_activity_two.id)
490 458 end
491 459
492 460 def test_reset_activities_should_reassign_time_entries_back_to_the_system_activity
493 461 @request.session[:user_id] = 2 # manager
494 462 project_activity = TimeEntryActivity.new({
495 463 :name => 'Project Specific Design',
496 464 :parent => TimeEntryActivity.find(9),
497 465 :project => Project.find(1),
498 466 :active => true
499 467 })
500 468 assert project_activity.save
501 469 assert TimeEntry.update_all("activity_id = '#{project_activity.id}'", ["project_id = ? AND activity_id = ?", 1, 9])
502 470 assert 3, TimeEntry.find_all_by_activity_id_and_project_id(project_activity.id, 1).size
503 471
504 472 delete :reset_activities, :id => 1
505 473 assert_response :redirect
506 474 assert_redirected_to 'projects/ecookbook/settings/activities'
507 475
508 476 assert_nil TimeEntryActivity.find_by_id(project_activity.id)
509 477 assert_equal 0, TimeEntry.find_all_by_activity_id_and_project_id(project_activity.id, 1).size, "TimeEntries still assigned to project specific activity"
510 478 assert_equal 3, TimeEntry.find_all_by_activity_id_and_project_id(9, 1).size, "TimeEntries still assigned to project specific activity"
511 479 end
512 480
513 481 def test_save_activities_to_override_system_activities
514 482 @request.session[:user_id] = 2 # manager
515 483 billable_field = TimeEntryActivityCustomField.find_by_name("Billable")
516 484
517 485 post :save_activities, :id => 1, :enumerations => {
518 486 "9"=> {"parent_id"=>"9", "custom_field_values"=>{"7" => "1"}, "active"=>"0"}, # Design, De-activate
519 487 "10"=> {"parent_id"=>"10", "custom_field_values"=>{"7"=>"0"}, "active"=>"1"}, # Development, Change custom value
520 488 "14"=>{"parent_id"=>"14", "custom_field_values"=>{"7"=>"1"}, "active"=>"1"}, # Inactive Activity, Activate with custom value
521 489 "11"=>{"parent_id"=>"11", "custom_field_values"=>{"7"=>"1"}, "active"=>"1"} # QA, no changes
522 490 }
523 491
524 492 assert_response :redirect
525 493 assert_redirected_to 'projects/ecookbook/settings/activities'
526 494
527 495 # Created project specific activities...
528 496 project = Project.find('ecookbook')
529 497
530 498 # ... Design
531 499 design = project.time_entry_activities.find_by_name("Design")
532 500 assert design, "Project activity not found"
533 501
534 502 assert_equal 9, design.parent_id # Relate to the system activity
535 503 assert_not_equal design.parent.id, design.id # Different records
536 504 assert_equal design.parent.name, design.name # Same name
537 505 assert !design.active?
538 506
539 507 # ... Development
540 508 development = project.time_entry_activities.find_by_name("Development")
541 509 assert development, "Project activity not found"
542 510
543 511 assert_equal 10, development.parent_id # Relate to the system activity
544 512 assert_not_equal development.parent.id, development.id # Different records
545 513 assert_equal development.parent.name, development.name # Same name
546 514 assert development.active?
547 515 assert_equal "0", development.custom_value_for(billable_field).value
548 516
549 517 # ... Inactive Activity
550 518 previously_inactive = project.time_entry_activities.find_by_name("Inactive Activity")
551 519 assert previously_inactive, "Project activity not found"
552 520
553 521 assert_equal 14, previously_inactive.parent_id # Relate to the system activity
554 522 assert_not_equal previously_inactive.parent.id, previously_inactive.id # Different records
555 523 assert_equal previously_inactive.parent.name, previously_inactive.name # Same name
556 524 assert previously_inactive.active?
557 525 assert_equal "1", previously_inactive.custom_value_for(billable_field).value
558 526
559 527 # ... QA
560 528 assert_equal nil, project.time_entry_activities.find_by_name("QA"), "Custom QA activity created when it wasn't modified"
561 529 end
562 530
563 531 def test_save_activities_will_update_project_specific_activities
564 532 @request.session[:user_id] = 2 # manager
565 533
566 534 project_activity = TimeEntryActivity.new({
567 535 :name => 'Project Specific',
568 536 :parent => TimeEntryActivity.find(:first),
569 537 :project => Project.find(1),
570 538 :active => true
571 539 })
572 540 assert project_activity.save
573 541 project_activity_two = TimeEntryActivity.new({
574 542 :name => 'Project Specific Two',
575 543 :parent => TimeEntryActivity.find(:last),
576 544 :project => Project.find(1),
577 545 :active => true
578 546 })
579 547 assert project_activity_two.save
580 548
581 549
582 550 post :save_activities, :id => 1, :enumerations => {
583 551 project_activity.id => {"custom_field_values"=>{"7" => "1"}, "active"=>"0"}, # De-activate
584 552 project_activity_two.id => {"custom_field_values"=>{"7" => "1"}, "active"=>"0"} # De-activate
585 553 }
586 554
587 555 assert_response :redirect
588 556 assert_redirected_to 'projects/ecookbook/settings/activities'
589 557
590 558 # Created project specific activities...
591 559 project = Project.find('ecookbook')
592 560 assert_equal 2, project.time_entry_activities.count
593 561
594 562 activity_one = project.time_entry_activities.find_by_name(project_activity.name)
595 563 assert activity_one, "Project activity not found"
596 564 assert_equal project_activity.id, activity_one.id
597 565 assert !activity_one.active?
598 566
599 567 activity_two = project.time_entry_activities.find_by_name(project_activity_two.name)
600 568 assert activity_two, "Project activity not found"
601 569 assert_equal project_activity_two.id, activity_two.id
602 570 assert !activity_two.active?
603 571 end
604 572
605 573 def test_save_activities_when_creating_new_activities_will_convert_existing_data
606 574 assert_equal 3, TimeEntry.find_all_by_activity_id_and_project_id(9, 1).size
607 575
608 576 @request.session[:user_id] = 2 # manager
609 577 post :save_activities, :id => 1, :enumerations => {
610 578 "9"=> {"parent_id"=>"9", "custom_field_values"=>{"7" => "1"}, "active"=>"0"} # Design, De-activate
611 579 }
612 580 assert_response :redirect
613 581
614 582 # No more TimeEntries using the system activity
615 583 assert_equal 0, TimeEntry.find_all_by_activity_id_and_project_id(9, 1).size, "Time Entries still assigned to system activities"
616 584 # All TimeEntries using project activity
617 585 project_specific_activity = TimeEntryActivity.find_by_parent_id_and_project_id(9, 1)
618 586 assert_equal 3, TimeEntry.find_all_by_activity_id_and_project_id(project_specific_activity.id, 1).size, "No Time Entries assigned to the project activity"
619 587 end
620 588
621 589 def test_save_activities_when_creating_new_activities_will_not_convert_existing_data_if_an_exception_is_raised
622 590 # TODO: Need to cause an exception on create but these tests
623 591 # aren't setup for mocking. Just create a record now so the
624 592 # second one is a dupicate
625 593 parent = TimeEntryActivity.find(9)
626 594 TimeEntryActivity.create!({:name => parent.name, :project_id => 1, :position => parent.position, :active => true})
627 595 TimeEntry.create!({:project_id => 1, :hours => 1.0, :user => User.find(1), :issue_id => 3, :activity_id => 10, :spent_on => '2009-01-01'})
628 596
629 597 assert_equal 3, TimeEntry.find_all_by_activity_id_and_project_id(9, 1).size
630 598 assert_equal 1, TimeEntry.find_all_by_activity_id_and_project_id(10, 1).size
631 599
632 600 @request.session[:user_id] = 2 # manager
633 601 post :save_activities, :id => 1, :enumerations => {
634 602 "9"=> {"parent_id"=>"9", "custom_field_values"=>{"7" => "1"}, "active"=>"0"}, # Design
635 603 "10"=> {"parent_id"=>"10", "custom_field_values"=>{"7"=>"0"}, "active"=>"1"} # Development, Change custom value
636 604 }
637 605 assert_response :redirect
638 606
639 607 # TimeEntries shouldn't have been reassigned on the failed record
640 608 assert_equal 3, TimeEntry.find_all_by_activity_id_and_project_id(9, 1).size, "Time Entries are not assigned to system activities"
641 609 # TimeEntries shouldn't have been reassigned on the saved record either
642 610 assert_equal 1, TimeEntry.find_all_by_activity_id_and_project_id(10, 1).size, "Time Entries are not assigned to system activities"
643 611 end
644 612
645 613 # A hook that is manually registered later
646 614 class ProjectBasedTemplate < Redmine::Hook::ViewListener
647 615 def view_layouts_base_html_head(context)
648 616 # Adds a project stylesheet
649 617 stylesheet_link_tag(context[:project].identifier) if context[:project]
650 618 end
651 619 end
652 620 # Don't use this hook now
653 621 Redmine::Hook.clear_listeners
654 622
655 623 def test_hook_response
656 624 Redmine::Hook.add_listener(ProjectBasedTemplate)
657 625 get :show, :id => 1
658 626 assert_tag :tag => 'link', :attributes => {:href => '/stylesheets/ecookbook.css'},
659 627 :parent => {:tag => 'head'}
660 628
661 629 Redmine::Hook.clear_listeners
662 630 end
663 631 end
@@ -1,104 +1,137
1 1 # redMine - project management software
2 2 # Copyright (C) 2006-2007 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 require File.dirname(__FILE__) + '/../test_helper'
19 19 require 'versions_controller'
20 20
21 21 # Re-raise errors caught by the controller.
22 22 class VersionsController; def rescue_action(e) raise e end; end
23 23
24 24 class VersionsControllerTest < ActionController::TestCase
25 25 fixtures :projects, :versions, :issues, :users, :roles, :members, :member_roles, :enabled_modules
26 26
27 27 def setup
28 28 @controller = VersionsController.new
29 29 @request = ActionController::TestRequest.new
30 30 @response = ActionController::TestResponse.new
31 31 User.current = nil
32 32 end
33 33
34 def test_index
35 get :index, :project_id => 1
36 assert_response :success
37 assert_template 'index'
38 assert_not_nil assigns(:versions)
39 # Version with no date set appears
40 assert assigns(:versions).include?(Version.find(3))
41 # Completed version doesn't appear
42 assert !assigns(:versions).include?(Version.find(1))
43 end
44
45 def test_index_with_completed_versions
46 get :index, :project_id => 1, :completed => 1
47 assert_response :success
48 assert_template 'index'
49 assert_not_nil assigns(:versions)
50 # Version with no date set appears
51 assert assigns(:versions).include?(Version.find(3))
52 # Completed version appears
53 assert assigns(:versions).include?(Version.find(1))
54 end
55
56 def test_index_showing_subprojects_versions
57 @subproject_version = Version.generate!(:project => Project.find(3))
58 get :index, :project_id => 1, :with_subprojects => 1
59 assert_response :success
60 assert_template 'index'
61 assert_not_nil assigns(:versions)
62
63 assert assigns(:versions).include?(Version.find(4)), "Shared version not found"
64 assert assigns(:versions).include?(@subproject_version), "Subproject version not found"
65 end
66
34 67 def test_show
35 68 get :show, :id => 2
36 69 assert_response :success
37 70 assert_template 'show'
38 71 assert_not_nil assigns(:version)
39 72
40 73 assert_tag :tag => 'h2', :content => /1.0/
41 74 end
42 75
43 76 def test_new
44 77 @request.session[:user_id] = 2 # manager
45 78 assert_difference 'Version.count' do
46 79 post :new, :project_id => '1', :version => {:name => 'test_add_version'}
47 80 end
48 81 assert_redirected_to '/projects/ecookbook/settings/versions'
49 82 version = Version.find_by_name('test_add_version')
50 83 assert_not_nil version
51 84 assert_equal 1, version.project_id
52 85 end
53 86
54 87 def test_new_from_issue_form
55 88 @request.session[:user_id] = 2 # manager
56 89 assert_difference 'Version.count' do
57 90 xhr :post, :new, :project_id => '1', :version => {:name => 'test_add_version_from_issue_form'}
58 91 end
59 92 assert_response :success
60 93 assert_select_rjs :replace, 'issue_fixed_version_id'
61 94 version = Version.find_by_name('test_add_version_from_issue_form')
62 95 assert_not_nil version
63 96 assert_equal 1, version.project_id
64 97 end
65 98
66 99 def test_get_edit
67 100 @request.session[:user_id] = 2
68 101 get :edit, :id => 2
69 102 assert_response :success
70 103 assert_template 'edit'
71 104 end
72 105
73 106 def test_close_completed
74 107 Version.update_all("status = 'open'")
75 108 @request.session[:user_id] = 2
76 109 post :close_completed, :project_id => 'ecookbook'
77 110 assert_redirected_to :controller => 'projects', :action => 'settings', :tab => 'versions', :id => 'ecookbook'
78 111 assert_not_nil Version.find_by_status('closed')
79 112 end
80 113
81 114 def test_post_edit
82 115 @request.session[:user_id] = 2
83 116 post :edit, :id => 2,
84 117 :version => { :name => 'New version name',
85 118 :effective_date => Date.today.strftime("%Y-%m-%d")}
86 119 assert_redirected_to :controller => 'projects', :action => 'settings', :tab => 'versions', :id => 'ecookbook'
87 120 version = Version.find(2)
88 121 assert_equal 'New version name', version.name
89 122 assert_equal Date.today, version.effective_date
90 123 end
91 124
92 125 def test_destroy
93 126 @request.session[:user_id] = 2
94 127 post :destroy, :id => 3
95 128 assert_redirected_to :controller => 'projects', :action => 'settings', :tab => 'versions', :id => 'ecookbook'
96 129 assert_nil Version.find_by_id(3)
97 130 end
98 131
99 132 def test_issue_status_by
100 133 xhr :get, :status_by, :id => 2
101 134 assert_response :success
102 135 assert_template '_issue_counts'
103 136 end
104 137 end
@@ -1,291 +1,291
1 1 # redMine - project management software
2 2 # Copyright (C) 2006-2010 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 require "#{File.dirname(__FILE__)}/../test_helper"
19 19
20 20 class RoutingTest < ActionController::IntegrationTest
21 21 context "activities" do
22 22 should_route :get, "/activity", :controller => 'activities', :action => 'index', :id => nil
23 23 should_route :get, "/activity.atom", :controller => 'activities', :action => 'index', :id => nil, :format => 'atom'
24 24 end
25 25
26 26 context "attachments" do
27 27 should_route :get, "/attachments/1", :controller => 'attachments', :action => 'show', :id => '1'
28 28 should_route :get, "/attachments/1/filename.ext", :controller => 'attachments', :action => 'show', :id => '1', :filename => 'filename.ext'
29 29 should_route :get, "/attachments/download/1", :controller => 'attachments', :action => 'download', :id => '1'
30 30 should_route :get, "/attachments/download/1/filename.ext", :controller => 'attachments', :action => 'download', :id => '1', :filename => 'filename.ext'
31 31 end
32 32
33 33 context "boards" do
34 34 should_route :get, "/projects/world_domination/boards", :controller => 'boards', :action => 'index', :project_id => 'world_domination'
35 35 should_route :get, "/projects/world_domination/boards/new", :controller => 'boards', :action => 'new', :project_id => 'world_domination'
36 36 should_route :get, "/projects/world_domination/boards/44", :controller => 'boards', :action => 'show', :project_id => 'world_domination', :id => '44'
37 37 should_route :get, "/projects/world_domination/boards/44.atom", :controller => 'boards', :action => 'show', :project_id => 'world_domination', :id => '44', :format => 'atom'
38 38 should_route :get, "/projects/world_domination/boards/44/edit", :controller => 'boards', :action => 'edit', :project_id => 'world_domination', :id => '44'
39 39
40 40 should_route :post, "/projects/world_domination/boards/new", :controller => 'boards', :action => 'new', :project_id => 'world_domination'
41 41 should_route :post, "/projects/world_domination/boards/44/edit", :controller => 'boards', :action => 'edit', :project_id => 'world_domination', :id => '44'
42 42 should_route :post, "/projects/world_domination/boards/44/destroy", :controller => 'boards', :action => 'destroy', :project_id => 'world_domination', :id => '44'
43 43
44 44 end
45 45
46 46 context "documents" do
47 47 should_route :get, "/projects/567/documents", :controller => 'documents', :action => 'index', :project_id => '567'
48 48 should_route :get, "/projects/567/documents/new", :controller => 'documents', :action => 'new', :project_id => '567'
49 49 should_route :get, "/documents/22", :controller => 'documents', :action => 'show', :id => '22'
50 50 should_route :get, "/documents/22/edit", :controller => 'documents', :action => 'edit', :id => '22'
51 51
52 52 should_route :post, "/projects/567/documents/new", :controller => 'documents', :action => 'new', :project_id => '567'
53 53 should_route :post, "/documents/567/edit", :controller => 'documents', :action => 'edit', :id => '567'
54 54 should_route :post, "/documents/567/destroy", :controller => 'documents', :action => 'destroy', :id => '567'
55 55 end
56 56
57 57 context "issues" do
58 58 # REST actions
59 59 should_route :get, "/issues", :controller => 'issues', :action => 'index'
60 60 should_route :get, "/issues.pdf", :controller => 'issues', :action => 'index', :format => 'pdf'
61 61 should_route :get, "/issues.atom", :controller => 'issues', :action => 'index', :format => 'atom'
62 62 should_route :get, "/issues.xml", :controller => 'issues', :action => 'index', :format => 'xml'
63 63 should_route :get, "/projects/23/issues", :controller => 'issues', :action => 'index', :project_id => '23'
64 64 should_route :get, "/projects/23/issues.pdf", :controller => 'issues', :action => 'index', :project_id => '23', :format => 'pdf'
65 65 should_route :get, "/projects/23/issues.atom", :controller => 'issues', :action => 'index', :project_id => '23', :format => 'atom'
66 66 should_route :get, "/projects/23/issues.xml", :controller => 'issues', :action => 'index', :project_id => '23', :format => 'xml'
67 67 should_route :get, "/issues/64", :controller => 'issues', :action => 'show', :id => '64'
68 68 should_route :get, "/issues/64.pdf", :controller => 'issues', :action => 'show', :id => '64', :format => 'pdf'
69 69 should_route :get, "/issues/64.atom", :controller => 'issues', :action => 'show', :id => '64', :format => 'atom'
70 70 should_route :get, "/issues/64.xml", :controller => 'issues', :action => 'show', :id => '64', :format => 'xml'
71 71
72 72 should_route :get, "/projects/23/issues/new", :controller => 'issues', :action => 'new', :project_id => '23'
73 73 should_route :post, "/projects/23/issues", :controller => 'issues', :action => 'create', :project_id => '23'
74 74 should_route :post, "/issues.xml", :controller => 'issues', :action => 'create', :format => 'xml'
75 75
76 76 should_route :get, "/issues/64/edit", :controller => 'issues', :action => 'edit', :id => '64'
77 77 # TODO: Should use PUT
78 78 should_route :post, "/issues/64/edit", :controller => 'issues', :action => 'edit', :id => '64'
79 79 should_route :put, "/issues/1.xml", :controller => 'issues', :action => 'update', :id => '1', :format => 'xml'
80 80
81 81 # TODO: Should use DELETE
82 82 should_route :post, "/issues/64/destroy", :controller => 'issues', :action => 'destroy', :id => '64'
83 83 should_route :delete, "/issues/1.xml", :controller => 'issues', :action => 'destroy', :id => '1', :format => 'xml'
84 84
85 85 # Extra actions
86 86 should_route :get, "/projects/23/issues/64/copy", :controller => 'issues', :action => 'new', :project_id => '23', :copy_from => '64'
87 87
88 88 should_route :get, "/issues/move/new", :controller => 'issue_moves', :action => 'new'
89 89 should_route :post, "/issues/move", :controller => 'issue_moves', :action => 'create'
90 90
91 91 should_route :post, "/issues/1/quoted", :controller => 'journals', :action => 'new', :id => '1'
92 92
93 93 should_route :get, "/issues/calendar", :controller => 'calendars', :action => 'show'
94 94 should_route :put, "/issues/calendar", :controller => 'calendars', :action => 'update'
95 95 should_route :get, "/projects/project-name/issues/calendar", :controller => 'calendars', :action => 'show', :project_id => 'project-name'
96 96 should_route :put, "/projects/project-name/issues/calendar", :controller => 'calendars', :action => 'update', :project_id => 'project-name'
97 97
98 98 should_route :get, "/issues/gantt", :controller => 'gantts', :action => 'show'
99 99 should_route :put, "/issues/gantt", :controller => 'gantts', :action => 'update'
100 100 should_route :get, "/projects/project-name/issues/gantt", :controller => 'gantts', :action => 'show', :project_id => 'project-name'
101 101 should_route :put, "/projects/project-name/issues/gantt", :controller => 'gantts', :action => 'update', :project_id => 'project-name'
102 102
103 103 should_route :get, "/issues/auto_complete", :controller => 'auto_completes', :action => 'issues'
104 104
105 105 should_route :get, "/issues/preview/123", :controller => 'previews', :action => 'issue', :id => '123'
106 106 should_route :post, "/issues/preview/123", :controller => 'previews', :action => 'issue', :id => '123'
107 107 should_route :get, "/issues/context_menu", :controller => 'context_menus', :action => 'issues'
108 108 should_route :post, "/issues/context_menu", :controller => 'context_menus', :action => 'issues'
109 109
110 110 should_route :get, "/issues/changes", :controller => 'journals', :action => 'index'
111 111
112 112 should_route :get, "/issues/bulk_edit", :controller => 'issues', :action => 'bulk_edit'
113 113 should_route :post, "/issues/bulk_edit", :controller => 'issues', :action => 'bulk_update'
114 114 end
115 115
116 116 context "issue categories" do
117 117 should_route :get, "/projects/test/issue_categories/new", :controller => 'issue_categories', :action => 'new', :project_id => 'test'
118 118
119 119 should_route :post, "/projects/test/issue_categories/new", :controller => 'issue_categories', :action => 'new', :project_id => 'test'
120 120 end
121 121
122 122 context "issue relations" do
123 123 should_route :post, "/issues/1/relations", :controller => 'issue_relations', :action => 'new', :issue_id => '1'
124 124 should_route :post, "/issues/1/relations/23/destroy", :controller => 'issue_relations', :action => 'destroy', :issue_id => '1', :id => '23'
125 125 end
126 126
127 127 context "issue reports" do
128 128 should_route :get, "/projects/567/issues/report", :controller => 'reports', :action => 'issue_report', :id => '567'
129 129 should_route :get, "/projects/567/issues/report/assigned_to", :controller => 'reports', :action => 'issue_report_details', :id => '567', :detail => 'assigned_to'
130 130 end
131 131
132 132 context "members" do
133 133 should_route :post, "/projects/5234/members/new", :controller => 'members', :action => 'new', :id => '5234'
134 134 end
135 135
136 136 context "messages" do
137 137 should_route :get, "/boards/22/topics/2", :controller => 'messages', :action => 'show', :id => '2', :board_id => '22'
138 138 should_route :get, "/boards/lala/topics/new", :controller => 'messages', :action => 'new', :board_id => 'lala'
139 139 should_route :get, "/boards/lala/topics/22/edit", :controller => 'messages', :action => 'edit', :id => '22', :board_id => 'lala'
140 140
141 141 should_route :post, "/boards/lala/topics/new", :controller => 'messages', :action => 'new', :board_id => 'lala'
142 142 should_route :post, "/boards/lala/topics/22/edit", :controller => 'messages', :action => 'edit', :id => '22', :board_id => 'lala'
143 143 should_route :post, "/boards/22/topics/555/replies", :controller => 'messages', :action => 'reply', :id => '555', :board_id => '22'
144 144 should_route :post, "/boards/22/topics/555/destroy", :controller => 'messages', :action => 'destroy', :id => '555', :board_id => '22'
145 145 end
146 146
147 147 context "news" do
148 148 should_route :get, "/news", :controller => 'news', :action => 'index'
149 149 should_route :get, "/news.atom", :controller => 'news', :action => 'index', :format => 'atom'
150 150 should_route :get, "/news.xml", :controller => 'news', :action => 'index', :format => 'xml'
151 151 should_route :get, "/news.json", :controller => 'news', :action => 'index', :format => 'json'
152 152 should_route :get, "/projects/567/news", :controller => 'news', :action => 'index', :project_id => '567'
153 153 should_route :get, "/projects/567/news.atom", :controller => 'news', :action => 'index', :format => 'atom', :project_id => '567'
154 154 should_route :get, "/projects/567/news.xml", :controller => 'news', :action => 'index', :format => 'xml', :project_id => '567'
155 155 should_route :get, "/projects/567/news.json", :controller => 'news', :action => 'index', :format => 'json', :project_id => '567'
156 156 should_route :get, "/news/2", :controller => 'news', :action => 'show', :id => '2'
157 157 should_route :get, "/projects/567/news/new", :controller => 'news', :action => 'new', :project_id => '567'
158 158 should_route :get, "/news/234", :controller => 'news', :action => 'show', :id => '234'
159 159
160 160 should_route :post, "/projects/567/news/new", :controller => 'news', :action => 'new', :project_id => '567'
161 161 should_route :post, "/news/567/edit", :controller => 'news', :action => 'edit', :id => '567'
162 162 should_route :post, "/news/567/destroy", :controller => 'news', :action => 'destroy', :id => '567'
163 163 end
164 164
165 165 context "projects" do
166 166 should_route :get, "/projects", :controller => 'projects', :action => 'index'
167 167 should_route :get, "/projects.atom", :controller => 'projects', :action => 'index', :format => 'atom'
168 168 should_route :get, "/projects.xml", :controller => 'projects', :action => 'index', :format => 'xml'
169 169 should_route :get, "/projects/new", :controller => 'projects', :action => 'add'
170 170 should_route :get, "/projects/test", :controller => 'projects', :action => 'show', :id => 'test'
171 171 should_route :get, "/projects/1.xml", :controller => 'projects', :action => 'show', :id => '1', :format => 'xml'
172 172 should_route :get, "/projects/4223/settings", :controller => 'projects', :action => 'settings', :id => '4223'
173 173 should_route :get, "/projects/4223/settings/members", :controller => 'projects', :action => 'settings', :id => '4223', :tab => 'members'
174 174 should_route :get, "/projects/567/destroy", :controller => 'projects', :action => 'destroy', :id => '567'
175 175 should_route :get, "/projects/33/files", :controller => 'projects', :action => 'list_files', :id => '33'
176 176 should_route :get, "/projects/33/files/new", :controller => 'projects', :action => 'add_file', :id => '33'
177 should_route :get, "/projects/33/roadmap", :controller => 'projects', :action => 'roadmap', :id => '33'
177 should_route :get, "/projects/33/roadmap", :controller => 'versions', :action => 'index', :project_id => '33'
178 178 should_route :get, "/projects/33/activity", :controller => 'activities', :action => 'index', :id => '33'
179 179 should_route :get, "/projects/33/activity.atom", :controller => 'activities', :action => 'index', :id => '33', :format => 'atom'
180 180
181 181 should_route :post, "/projects/new", :controller => 'projects', :action => 'add'
182 182 should_route :post, "/projects.xml", :controller => 'projects', :action => 'add', :format => 'xml'
183 183 should_route :post, "/projects/4223/edit", :controller => 'projects', :action => 'edit', :id => '4223'
184 184 should_route :post, "/projects/64/destroy", :controller => 'projects', :action => 'destroy', :id => '64'
185 185 should_route :post, "/projects/33/files/new", :controller => 'projects', :action => 'add_file', :id => '33'
186 186 should_route :post, "/projects/64/archive", :controller => 'projects', :action => 'archive', :id => '64'
187 187 should_route :post, "/projects/64/unarchive", :controller => 'projects', :action => 'unarchive', :id => '64'
188 188 should_route :post, "/projects/64/activities/save", :controller => 'projects', :action => 'save_activities', :id => '64'
189 189
190 190 should_route :put, "/projects/1.xml", :controller => 'projects', :action => 'edit', :id => '1', :format => 'xml'
191 191
192 192 should_route :delete, "/projects/1.xml", :controller => 'projects', :action => 'destroy', :id => '1', :format => 'xml'
193 193 should_route :delete, "/projects/64/reset_activities", :controller => 'projects', :action => 'reset_activities', :id => '64'
194 194 end
195 195
196 196 context "repositories" do
197 197 should_route :get, "/projects/redmine/repository", :controller => 'repositories', :action => 'show', :id => 'redmine'
198 198 should_route :get, "/projects/redmine/repository/edit", :controller => 'repositories', :action => 'edit', :id => 'redmine'
199 199 should_route :get, "/projects/redmine/repository/revisions", :controller => 'repositories', :action => 'revisions', :id => 'redmine'
200 200 should_route :get, "/projects/redmine/repository/revisions.atom", :controller => 'repositories', :action => 'revisions', :id => 'redmine', :format => 'atom'
201 201 should_route :get, "/projects/redmine/repository/revisions/2457", :controller => 'repositories', :action => 'revision', :id => 'redmine', :rev => '2457'
202 202 should_route :get, "/projects/redmine/repository/revisions/2457/diff", :controller => 'repositories', :action => 'diff', :id => 'redmine', :rev => '2457'
203 203 should_route :get, "/projects/redmine/repository/revisions/2457/diff.diff", :controller => 'repositories', :action => 'diff', :id => 'redmine', :rev => '2457', :format => 'diff'
204 204 should_route :get, "/projects/redmine/repository/diff/path/to/file.c", :controller => 'repositories', :action => 'diff', :id => 'redmine', :path => %w[path to file.c]
205 205 should_route :get, "/projects/redmine/repository/revisions/2/diff/path/to/file.c", :controller => 'repositories', :action => 'diff', :id => 'redmine', :path => %w[path to file.c], :rev => '2'
206 206 should_route :get, "/projects/redmine/repository/browse/path/to/file.c", :controller => 'repositories', :action => 'browse', :id => 'redmine', :path => %w[path to file.c]
207 207 should_route :get, "/projects/redmine/repository/entry/path/to/file.c", :controller => 'repositories', :action => 'entry', :id => 'redmine', :path => %w[path to file.c]
208 208 should_route :get, "/projects/redmine/repository/revisions/2/entry/path/to/file.c", :controller => 'repositories', :action => 'entry', :id => 'redmine', :path => %w[path to file.c], :rev => '2'
209 209 should_route :get, "/projects/redmine/repository/raw/path/to/file.c", :controller => 'repositories', :action => 'entry', :id => 'redmine', :path => %w[path to file.c], :format => 'raw'
210 210 should_route :get, "/projects/redmine/repository/revisions/2/raw/path/to/file.c", :controller => 'repositories', :action => 'entry', :id => 'redmine', :path => %w[path to file.c], :rev => '2', :format => 'raw'
211 211 should_route :get, "/projects/redmine/repository/annotate/path/to/file.c", :controller => 'repositories', :action => 'annotate', :id => 'redmine', :path => %w[path to file.c]
212 212 should_route :get, "/projects/redmine/repository/changes/path/to/file.c", :controller => 'repositories', :action => 'changes', :id => 'redmine', :path => %w[path to file.c]
213 213 should_route :get, "/projects/redmine/repository/statistics", :controller => 'repositories', :action => 'stats', :id => 'redmine'
214 214
215 215
216 216 should_route :post, "/projects/redmine/repository/edit", :controller => 'repositories', :action => 'edit', :id => 'redmine'
217 217 end
218 218
219 219 context "timelogs" do
220 220 should_route :get, "/issues/567/time_entries/new", :controller => 'timelog', :action => 'edit', :issue_id => '567'
221 221 should_route :get, "/projects/ecookbook/time_entries/new", :controller => 'timelog', :action => 'edit', :project_id => 'ecookbook'
222 222 should_route :get, "/projects/ecookbook/issues/567/time_entries/new", :controller => 'timelog', :action => 'edit', :project_id => 'ecookbook', :issue_id => '567'
223 223 should_route :get, "/time_entries/22/edit", :controller => 'timelog', :action => 'edit', :id => '22'
224 224 should_route :get, "/time_entries/report", :controller => 'timelog', :action => 'report'
225 225 should_route :get, "/projects/567/time_entries/report", :controller => 'timelog', :action => 'report', :project_id => '567'
226 226 should_route :get, "/projects/567/time_entries/report.csv", :controller => 'timelog', :action => 'report', :project_id => '567', :format => 'csv'
227 227 should_route :get, "/time_entries", :controller => 'timelog', :action => 'details'
228 228 should_route :get, "/time_entries.csv", :controller => 'timelog', :action => 'details', :format => 'csv'
229 229 should_route :get, "/time_entries.atom", :controller => 'timelog', :action => 'details', :format => 'atom'
230 230 should_route :get, "/projects/567/time_entries", :controller => 'timelog', :action => 'details', :project_id => '567'
231 231 should_route :get, "/projects/567/time_entries.csv", :controller => 'timelog', :action => 'details', :project_id => '567', :format => 'csv'
232 232 should_route :get, "/projects/567/time_entries.atom", :controller => 'timelog', :action => 'details', :project_id => '567', :format => 'atom'
233 233 should_route :get, "/issues/234/time_entries", :controller => 'timelog', :action => 'details', :issue_id => '234'
234 234 should_route :get, "/issues/234/time_entries.csv", :controller => 'timelog', :action => 'details', :issue_id => '234', :format => 'csv'
235 235 should_route :get, "/issues/234/time_entries.atom", :controller => 'timelog', :action => 'details', :issue_id => '234', :format => 'atom'
236 236 should_route :get, "/projects/ecookbook/issues/123/time_entries", :controller => 'timelog', :action => 'details', :project_id => 'ecookbook', :issue_id => '123'
237 237
238 238 should_route :post, "/time_entries/55/destroy", :controller => 'timelog', :action => 'destroy', :id => '55'
239 239 end
240 240
241 241 context "users" do
242 242 should_route :get, "/users", :controller => 'users', :action => 'index'
243 243 should_route :get, "/users/44", :controller => 'users', :action => 'show', :id => '44'
244 244 should_route :get, "/users/new", :controller => 'users', :action => 'add'
245 245 should_route :get, "/users/444/edit", :controller => 'users', :action => 'edit', :id => '444'
246 246 should_route :get, "/users/222/edit/membership", :controller => 'users', :action => 'edit', :id => '222', :tab => 'membership'
247 247
248 248 should_route :post, "/users/new", :controller => 'users', :action => 'add'
249 249 should_route :post, "/users/444/edit", :controller => 'users', :action => 'edit', :id => '444'
250 250 should_route :post, "/users/123/memberships", :controller => 'users', :action => 'edit_membership', :id => '123'
251 251 should_route :post, "/users/123/memberships/55", :controller => 'users', :action => 'edit_membership', :id => '123', :membership_id => '55'
252 252 should_route :post, "/users/567/memberships/12/destroy", :controller => 'users', :action => 'destroy_membership', :id => '567', :membership_id => '12'
253 253 end
254 254
255 255 context "versions" do
256 256 should_route :get, "/projects/foo/versions/new", :controller => 'versions', :action => 'new', :project_id => 'foo'
257 257
258 258 should_route :post, "/projects/foo/versions/new", :controller => 'versions', :action => 'new', :project_id => 'foo'
259 259 end
260 260
261 261 context "wiki (singular, project's pages)" do
262 262 should_route :get, "/projects/567/wiki", :controller => 'wiki', :action => 'index', :id => '567'
263 263 should_route :get, "/projects/567/wiki/lalala", :controller => 'wiki', :action => 'index', :id => '567', :page => 'lalala'
264 264 should_route :get, "/projects/567/wiki/my_page/edit", :controller => 'wiki', :action => 'edit', :id => '567', :page => 'my_page'
265 265 should_route :get, "/projects/1/wiki/CookBook_documentation/history", :controller => 'wiki', :action => 'history', :id => '1', :page => 'CookBook_documentation'
266 266 should_route :get, "/projects/1/wiki/CookBook_documentation/diff/2/vs/1", :controller => 'wiki', :action => 'diff', :id => '1', :page => 'CookBook_documentation', :version => '2', :version_from => '1'
267 267 should_route :get, "/projects/1/wiki/CookBook_documentation/annotate/2", :controller => 'wiki', :action => 'annotate', :id => '1', :page => 'CookBook_documentation', :version => '2'
268 268 should_route :get, "/projects/22/wiki/ladida/rename", :controller => 'wiki', :action => 'rename', :id => '22', :page => 'ladida'
269 269 should_route :get, "/projects/567/wiki/page_index", :controller => 'wiki', :action => 'special', :id => '567', :page => 'page_index'
270 270 should_route :get, "/projects/567/wiki/Page_Index", :controller => 'wiki', :action => 'special', :id => '567', :page => 'Page_Index'
271 271 should_route :get, "/projects/567/wiki/date_index", :controller => 'wiki', :action => 'special', :id => '567', :page => 'date_index'
272 272 should_route :get, "/projects/567/wiki/export", :controller => 'wiki', :action => 'special', :id => '567', :page => 'export'
273 273
274 274 should_route :post, "/projects/567/wiki/my_page/edit", :controller => 'wiki', :action => 'edit', :id => '567', :page => 'my_page'
275 275 should_route :post, "/projects/567/wiki/CookBook_documentation/preview", :controller => 'wiki', :action => 'preview', :id => '567', :page => 'CookBook_documentation'
276 276 should_route :post, "/projects/22/wiki/ladida/rename", :controller => 'wiki', :action => 'rename', :id => '22', :page => 'ladida'
277 277 should_route :post, "/projects/22/wiki/ladida/destroy", :controller => 'wiki', :action => 'destroy', :id => '22', :page => 'ladida'
278 278 should_route :post, "/projects/22/wiki/ladida/protect", :controller => 'wiki', :action => 'protect', :id => '22', :page => 'ladida'
279 279 end
280 280
281 281 context "wikis (plural, admin setup)" do
282 282 should_route :get, "/projects/ladida/wiki/destroy", :controller => 'wikis', :action => 'destroy', :id => 'ladida'
283 283
284 284 should_route :post, "/projects/ladida/wiki", :controller => 'wikis', :action => 'edit', :id => 'ladida'
285 285 should_route :post, "/projects/ladida/wiki/destroy", :controller => 'wikis', :action => 'destroy', :id => 'ladida'
286 286 end
287 287
288 288 context "administration panel" do
289 289 should_route :get, "/admin/projects", :controller => 'admin', :action => 'projects'
290 290 end
291 291 end
General Comments 0
You need to be logged in to leave comments. Login now