##// END OF EJS Templates
Refactor: rename TimelogController#details to #index...
Eric Davis -
r4121:2ecca7e4df96
parent child
Show More
@@ -1,190 +1,190
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 class TimelogController < ApplicationController
19 19 menu_item :issues
20 20 before_filter :find_project, :authorize, :only => [:edit, :destroy]
21 before_filter :find_optional_project, :only => [:details]
21 before_filter :find_optional_project, :only => [:index]
22 22
23 verify :method => :post, :only => :destroy, :redirect_to => { :action => :details }
23 verify :method => :post, :only => :destroy, :redirect_to => { :action => :index }
24 24
25 25 helper :sort
26 26 include SortHelper
27 27 helper :issues
28 28 include TimelogHelper
29 29 helper :custom_fields
30 30 include CustomFieldsHelper
31 31
32 def details
32 def index
33 33 sort_init 'spent_on', 'desc'
34 34 sort_update 'spent_on' => 'spent_on',
35 35 'user' => 'user_id',
36 36 'activity' => 'activity_id',
37 37 'project' => "#{Project.table_name}.name",
38 38 'issue' => 'issue_id',
39 39 'hours' => 'hours'
40 40
41 41 cond = ARCondition.new
42 42 if @project.nil?
43 43 cond << Project.allowed_to_condition(User.current, :view_time_entries)
44 44 elsif @issue.nil?
45 45 cond << @project.project_condition(Setting.display_subprojects_issues?)
46 46 else
47 47 cond << "#{Issue.table_name}.root_id = #{@issue.root_id} AND #{Issue.table_name}.lft >= #{@issue.lft} AND #{Issue.table_name}.rgt <= #{@issue.rgt}"
48 48 end
49 49
50 50 retrieve_date_range
51 51 cond << ['spent_on BETWEEN ? AND ?', @from, @to]
52 52
53 53 TimeEntry.visible_by(User.current) do
54 54 respond_to do |format|
55 55 format.html {
56 56 # Paginate results
57 57 @entry_count = TimeEntry.count(:include => [:project, :issue], :conditions => cond.conditions)
58 58 @entry_pages = Paginator.new self, @entry_count, per_page_option, params['page']
59 59 @entries = TimeEntry.find(:all,
60 60 :include => [:project, :activity, :user, {:issue => :tracker}],
61 61 :conditions => cond.conditions,
62 62 :order => sort_clause,
63 63 :limit => @entry_pages.items_per_page,
64 64 :offset => @entry_pages.current.offset)
65 65 @total_hours = TimeEntry.sum(:hours, :include => [:project, :issue], :conditions => cond.conditions).to_f
66 66
67 67 render :layout => !request.xhr?
68 68 }
69 69 format.atom {
70 70 entries = TimeEntry.find(:all,
71 71 :include => [:project, :activity, :user, {:issue => :tracker}],
72 72 :conditions => cond.conditions,
73 73 :order => "#{TimeEntry.table_name}.created_on DESC",
74 74 :limit => Setting.feeds_limit.to_i)
75 75 render_feed(entries, :title => l(:label_spent_time))
76 76 }
77 77 format.csv {
78 78 # Export all entries
79 79 @entries = TimeEntry.find(:all,
80 80 :include => [:project, :activity, :user, {:issue => [:tracker, :assigned_to, :priority]}],
81 81 :conditions => cond.conditions,
82 82 :order => sort_clause)
83 83 send_data(entries_to_csv(@entries), :type => 'text/csv; header=present', :filename => 'timelog.csv')
84 84 }
85 85 end
86 86 end
87 87 end
88 88
89 89 def edit
90 90 (render_403; return) if @time_entry && !@time_entry.editable_by?(User.current)
91 91 @time_entry ||= TimeEntry.new(:project => @project, :issue => @issue, :user => User.current, :spent_on => User.current.today)
92 92 @time_entry.attributes = params[:time_entry]
93 93
94 94 call_hook(:controller_timelog_edit_before_save, { :params => params, :time_entry => @time_entry })
95 95
96 96 if request.post? and @time_entry.save
97 97 flash[:notice] = l(:notice_successful_update)
98 redirect_back_or_default :action => 'details', :project_id => @time_entry.project
98 redirect_back_or_default :action => 'index', :project_id => @time_entry.project
99 99 return
100 100 end
101 101 end
102 102
103 103 def destroy
104 104 (render_404; return) unless @time_entry
105 105 (render_403; return) unless @time_entry.editable_by?(User.current)
106 106 if @time_entry.destroy && @time_entry.destroyed?
107 107 flash[:notice] = l(:notice_successful_delete)
108 108 else
109 109 flash[:error] = l(:notice_unable_delete_time_entry)
110 110 end
111 111 redirect_to :back
112 112 rescue ::ActionController::RedirectBackError
113 redirect_to :action => 'details', :project_id => @time_entry.project
113 redirect_to :action => 'index', :project_id => @time_entry.project
114 114 end
115 115
116 116 private
117 117 def find_project
118 118 if params[:id]
119 119 @time_entry = TimeEntry.find(params[:id])
120 120 @project = @time_entry.project
121 121 elsif params[:issue_id]
122 122 @issue = Issue.find(params[:issue_id])
123 123 @project = @issue.project
124 124 elsif params[:project_id]
125 125 @project = Project.find(params[:project_id])
126 126 else
127 127 render_404
128 128 return false
129 129 end
130 130 rescue ActiveRecord::RecordNotFound
131 131 render_404
132 132 end
133 133
134 134 def find_optional_project
135 135 if !params[:issue_id].blank?
136 136 @issue = Issue.find(params[:issue_id])
137 137 @project = @issue.project
138 138 elsif !params[:project_id].blank?
139 139 @project = Project.find(params[:project_id])
140 140 end
141 141 deny_access unless User.current.allowed_to?(:view_time_entries, @project, :global => true)
142 142 end
143 143
144 144 # Retrieves the date range based on predefined ranges or specific from/to param dates
145 145 def retrieve_date_range
146 146 @free_period = false
147 147 @from, @to = nil, nil
148 148
149 149 if params[:period_type] == '1' || (params[:period_type].nil? && !params[:period].nil?)
150 150 case params[:period].to_s
151 151 when 'today'
152 152 @from = @to = Date.today
153 153 when 'yesterday'
154 154 @from = @to = Date.today - 1
155 155 when 'current_week'
156 156 @from = Date.today - (Date.today.cwday - 1)%7
157 157 @to = @from + 6
158 158 when 'last_week'
159 159 @from = Date.today - 7 - (Date.today.cwday - 1)%7
160 160 @to = @from + 6
161 161 when '7_days'
162 162 @from = Date.today - 7
163 163 @to = Date.today
164 164 when 'current_month'
165 165 @from = Date.civil(Date.today.year, Date.today.month, 1)
166 166 @to = (@from >> 1) - 1
167 167 when 'last_month'
168 168 @from = Date.civil(Date.today.year, Date.today.month, 1) << 1
169 169 @to = (@from >> 1) - 1
170 170 when '30_days'
171 171 @from = Date.today - 30
172 172 @to = Date.today
173 173 when 'current_year'
174 174 @from = Date.civil(Date.today.year, 1, 1)
175 175 @to = Date.civil(Date.today.year, 12, 31)
176 176 end
177 177 elsif params[:period_type] == '2' || (params[:period_type].nil? && (!params[:from].nil? || !params[:to].nil?))
178 178 begin; @from = params[:from].to_s.to_date unless params[:from].blank?; rescue; end
179 179 begin; @to = params[:to].to_s.to_date unless params[:to].blank?; rescue; end
180 180 @free_period = true
181 181 else
182 182 # default
183 183 end
184 184
185 185 @from, @to = @to, @from if @from && @to && @from > @to
186 186 @from ||= (TimeEntry.earilest_date_for_project(@project) || Date.today)
187 187 @to ||= (TimeEntry.latest_date_for_project(@project) || Date.today)
188 188 end
189 189
190 190 end
@@ -1,134 +1,134
1 1 <%= render :partial => 'action_menu' %>
2 2
3 3 <h2><%= @issue.tracker.name %> #<%= @issue.id %></h2>
4 4
5 5 <div class="<%= @issue.css_classes %> details">
6 6 <%= avatar(@issue.author, :size => "50") %>
7 7
8 8 <div class="subject">
9 9 <%= render_issue_subject_with_tree(@issue) %>
10 10 </div>
11 11 <p class="author">
12 12 <%= authoring @issue.created_on, @issue.author %>.
13 13 <% if @issue.created_on != @issue.updated_on %>
14 14 <%= l(:label_updated_time, time_tag(@issue.updated_on)) %>.
15 15 <% end %>
16 16 </p>
17 17
18 18 <table class="attributes">
19 19 <tr>
20 20 <th class="status"><%=l(:field_status)%>:</th><td class="status"><%= @issue.status.name %></td>
21 21 <th class="start-date"><%=l(:field_start_date)%>:</th><td class="start-date"><%= format_date(@issue.start_date) %></td>
22 22 </tr>
23 23 <tr>
24 24 <th class="priority"><%=l(:field_priority)%>:</th><td class="priority"><%= @issue.priority.name %></td>
25 25 <th class="due-date"><%=l(:field_due_date)%>:</th><td class="due-date"><%= format_date(@issue.due_date) %></td>
26 26 </tr>
27 27 <tr>
28 28 <th class="assigned-to"><%=l(:field_assigned_to)%>:</th><td class="assigned-to"><%= avatar(@issue.assigned_to, :size => "14") %><%= @issue.assigned_to ? link_to_user(@issue.assigned_to) : "-" %></td>
29 29 <th class="progress"><%=l(:field_done_ratio)%>:</th><td class="progress"><%= progress_bar @issue.done_ratio, :width => '80px', :legend => "#{@issue.done_ratio}%" %></td>
30 30 </tr>
31 31 <tr>
32 32 <th class="category"><%=l(:field_category)%>:</th><td class="category"><%=h @issue.category ? @issue.category.name : "-" %></td>
33 33 <% if User.current.allowed_to?(:view_time_entries, @project) %>
34 34 <th class="spent-time"><%=l(:label_spent_time)%>:</th>
35 <td class="spent-time"><%= @issue.spent_hours > 0 ? (link_to l_hours(@issue.spent_hours), {:controller => 'timelog', :action => 'details', :project_id => @project, :issue_id => @issue}) : "-" %></td>
35 <td class="spent-time"><%= @issue.spent_hours > 0 ? (link_to l_hours(@issue.spent_hours), {:controller => 'timelog', :action => 'index', :project_id => @project, :issue_id => @issue}) : "-" %></td>
36 36 <% end %>
37 37 </tr>
38 38 <tr>
39 39 <th class="fixed-version"><%=l(:field_fixed_version)%>:</th><td class="fixed-version"><%= @issue.fixed_version ? link_to_version(@issue.fixed_version) : "-" %></td>
40 40 <% if @issue.estimated_hours %>
41 41 <th class="estimated-hours"><%=l(:field_estimated_hours)%>:</th><td class="estimated-hours"><%= l_hours(@issue.estimated_hours) %></td>
42 42 <% end %>
43 43 </tr>
44 44 <%= render_custom_fields_rows(@issue) %>
45 45 <%= call_hook(:view_issues_show_details_bottom, :issue => @issue) %>
46 46 </table>
47 47 <hr />
48 48
49 49 <div class="contextual">
50 50 <%= link_to_remote_if_authorized(l(:button_quote), { :url => {:action => 'reply', :id => @issue} }, :class => 'icon icon-comment') unless @issue.description.blank? %>
51 51 </div>
52 52
53 53 <p><strong><%=l(:field_description)%></strong></p>
54 54 <div class="wiki">
55 55 <%= textilizable @issue, :description, :attachments => @issue.attachments %>
56 56 </div>
57 57
58 58 <%= link_to_attachments @issue %>
59 59
60 60 <%= call_hook(:view_issues_show_description_bottom, :issue => @issue) %>
61 61
62 62 <% if !@issue.leaf? || User.current.allowed_to?(:manage_subtasks, @project) %>
63 63 <hr />
64 64 <div id="issue_tree">
65 65 <div class="contextual">
66 66 <%= link_to(l(:button_add), {:controller => 'issues', :action => 'new', :project_id => @project, :issue => {:parent_issue_id => @issue}}) if User.current.allowed_to?(:manage_subtasks, @project) %>
67 67 </div>
68 68 <p><strong><%=l(:label_subtask_plural)%></strong></p>
69 69 <%= render_descendants_tree(@issue) unless @issue.leaf? %>
70 70 </div>
71 71 <% end %>
72 72
73 73 <% if authorize_for('issue_relations', 'new') || @issue.relations.present? %>
74 74 <hr />
75 75 <div id="relations">
76 76 <%= render :partial => 'relations' %>
77 77 </div>
78 78 <% end %>
79 79
80 80 </div>
81 81
82 82 <% if @changesets.present? %>
83 83 <div id="issue-changesets">
84 84 <h3><%=l(:label_associated_revisions)%></h3>
85 85 <%= render :partial => 'changesets', :locals => { :changesets => @changesets} %>
86 86 </div>
87 87 <% end %>
88 88
89 89 <% if @journals.present? %>
90 90 <div id="history">
91 91 <h3><%=l(:label_history)%></h3>
92 92 <%= render :partial => 'history', :locals => { :issue => @issue, :journals => @journals } %>
93 93 </div>
94 94 <% end %>
95 95
96 96
97 97 <div style="clear: both;"></div>
98 98 <%= render :partial => 'action_menu', :locals => {:replace_watcher => 'watcher2' } %>
99 99
100 100 <div style="clear: both;"></div>
101 101 <% if authorize_for('issues', 'edit') %>
102 102 <div id="update" style="display:none;">
103 103 <h3><%= l(:button_update) %></h3>
104 104 <%= render :partial => 'edit' %>
105 105 </div>
106 106 <% end %>
107 107
108 108 <% other_formats_links do |f| %>
109 109 <%= f.link_to 'Atom', :url => {:key => User.current.rss_key} %>
110 110 <%= f.link_to 'PDF' %>
111 111 <% end %>
112 112
113 113 <% html_title "#{@issue.tracker.name} ##{@issue.id}: #{@issue.subject}" %>
114 114
115 115 <% content_for :sidebar do %>
116 116 <%= render :partial => 'issues/sidebar' %>
117 117
118 118 <% if User.current.allowed_to?(:add_issue_watchers, @project) ||
119 119 (@issue.watchers.present? && User.current.allowed_to?(:view_issue_watchers, @project)) %>
120 120 <div id="watchers">
121 121 <%= render :partial => 'watchers/watchers', :locals => {:watched => @issue} %>
122 122 </div>
123 123 <% end %>
124 124 <% end %>
125 125
126 126 <% content_for :header_tags do %>
127 127 <%= auto_discovery_link_tag(:atom, {:format => 'atom', :key => User.current.rss_key}, :title => "#{@issue.project} - #{@issue.tracker} ##{@issue.id}: #{@issue.subject}") %>
128 128 <%= stylesheet_link_tag 'scm' %>
129 129 <%= javascript_include_tag 'context_menu' %>
130 130 <%= stylesheet_link_tag 'context_menu' %>
131 131 <%= stylesheet_link_tag 'context_menu_rtl' if l(:direction) == 'rtl' %>
132 132 <% end %>
133 133 <div id="context-menu" style="display: none;"></div>
134 134 <%= javascript_tag "new ContextMenu('#{issues_context_menu_path}')" %>
@@ -1,80 +1,80
1 1 <div class="contextual">
2 2 <% if User.current.allowed_to?(:add_subprojects, @project) %>
3 3 <%= link_to l(:label_subproject_new), {:controller => 'projects', :action => 'new', :parent_id => @project}, :class => 'icon icon-add' %>
4 4 <% end %>
5 5 </div>
6 6
7 7 <h2><%=l(:label_overview)%></h2>
8 8
9 9 <div class="splitcontentleft">
10 10 <div class="wiki">
11 11 <%= textilizable @project.description %>
12 12 </div>
13 13 <ul>
14 14 <% unless @project.homepage.blank? %><li><%=l(:field_homepage)%>: <%= auto_link(h(@project.homepage)) %></li><% end %>
15 15 <% if @subprojects.any? %>
16 16 <li><%=l(:label_subproject_plural)%>:
17 17 <%= @subprojects.collect{|p| link_to(h(p), :action => 'show', :id => p)}.join(", ") %></li>
18 18 <% end %>
19 19 <% @project.custom_values.each do |custom_value| %>
20 20 <% if !custom_value.value.blank? %>
21 21 <li><%= custom_value.custom_field.name%>: <%=h show_value(custom_value) %></li>
22 22 <% end %>
23 23 <% end %>
24 24 </ul>
25 25
26 26 <% if User.current.allowed_to?(:view_issues, @project) %>
27 27 <div class="issues box">
28 28 <h3><%=l(:label_issue_tracking)%></h3>
29 29 <ul>
30 30 <% for tracker in @trackers %>
31 31 <li><%= link_to tracker.name, :controller => 'issues', :action => 'index', :project_id => @project,
32 32 :set_filter => 1,
33 33 "tracker_id" => tracker.id %>:
34 34 <%= l(:label_x_open_issues_abbr_on_total, :count => @open_issues_by_tracker[tracker].to_i,
35 35 :total => @total_issues_by_tracker[tracker].to_i) %>
36 36 </li>
37 37 <% end %>
38 38 </ul>
39 39 <p>
40 40 <%= link_to l(:label_issue_view_all), :controller => 'issues', :action => 'index', :project_id => @project, :set_filter => 1 %>
41 41 <% if User.current.allowed_to?(:view_calendar, @project, :global => true) %>
42 42 | <%= link_to(l(:label_calendar), :controller => 'calendars', :action => 'show', :project_id => @project) %>
43 43 <% end %>
44 44 <% if User.current.allowed_to?(:view_gantt, @project, :global => true) %>
45 45 | <%= link_to(l(:label_gantt), :controller => 'gantts', :action => 'show', :project_id => @project) %>
46 46 <% end %>
47 47 </p>
48 48 </div>
49 49 <% end %>
50 50 <%= call_hook(:view_projects_show_left, :project => @project) %>
51 51 </div>
52 52
53 53 <div class="splitcontentright">
54 54 <%= render :partial => 'members_box' %>
55 55
56 56 <% if @news.any? && authorize_for('news', 'index') %>
57 57 <div class="news box">
58 58 <h3><%=l(:label_news_latest)%></h3>
59 59 <%= render :partial => 'news/news', :collection => @news %>
60 60 <p><%= link_to l(:label_news_view_all), :controller => 'news', :action => 'index', :project_id => @project %></p>
61 61 </div>
62 62 <% end %>
63 63 <%= call_hook(:view_projects_show_right, :project => @project) %>
64 64 </div>
65 65
66 66 <% content_for :sidebar do %>
67 67 <% if @total_hours && User.current.allowed_to?(:view_time_entries, @project) %>
68 68 <h3><%= l(:label_spent_time) %></h3>
69 69 <p><span class="icon icon-time"><%= l_hours(@total_hours) %></span></p>
70 <p><%= link_to(l(:label_details), {:controller => 'timelog', :action => 'details', :project_id => @project}) %> |
70 <p><%= link_to(l(:label_details), {:controller => 'timelog', :action => 'index', :project_id => @project}) %> |
71 71 <%= link_to(l(:label_report), {:controller => 'time_entry_reports', :action => 'report', :project_id => @project}) %></p>
72 72 <% end %>
73 73 <%= call_hook(:view_projects_show_sidebar_bottom, :project => @project) %>
74 74 <% end %>
75 75
76 76 <% content_for :header_tags do %>
77 77 <%= auto_discovery_link_tag(:atom, {:controller => 'activities', :action => 'index', :id => @project, :format => 'atom', :key => User.current.rss_key}) %>
78 78 <% end %>
79 79
80 80 <% html_title(l(:label_overview)) -%>
@@ -1,36 +1,36
1 1 <fieldset id="date-range" class="collapsible">
2 2 <legend onclick="toggleFieldset(this);"><%= l(:label_date_range) %></legend>
3 3 <div>
4 4 <p>
5 5 <%= radio_button_tag 'period_type', '1', !@free_period %>
6 6 <%= select_tag 'period', options_for_period_select(params[:period]),
7 7 :onchange => 'this.form.onsubmit();',
8 8 :onfocus => '$("period_type_1").checked = true;' %>
9 9 </p>
10 10 <p>
11 11 <%= radio_button_tag 'period_type', '2', @free_period %>
12 12 <span onclick="$('period_type_2').checked = true;">
13 13 <%= l(:label_date_from_to, :start => (text_field_tag('from', @from, :size => 10) + calendar_for('from')),
14 14 :end => (text_field_tag('to', @to, :size => 10) + calendar_for('to'))) %>
15 15 </span>
16 16 </p>
17 17 </div>
18 18 </fieldset>
19 19 <p class="buttons">
20 20 <%= link_to_remote l(:button_apply),
21 21 { :url => { },
22 22 :update => "content",
23 23 :with => "Form.serialize('query_form')",
24 24 :method => :get
25 25 }, :class => 'icon icon-checked' %>
26 26 </p>
27 27
28 28 <div class="tabs">
29 29 <% url_params = @free_period ? { :from => @from, :to => @to } : { :period => params[:period] } %>
30 30 <ul>
31 <li><%= link_to(l(:label_details), url_params.merge({:controller => 'timelog', :action => 'details', :project_id => @project, :issue_id => @issue }),
32 :class => (@controller.action_name == 'details' ? 'selected' : nil)) %></li>
31 <li><%= link_to(l(:label_details), url_params.merge({:controller => 'timelog', :action => 'index', :project_id => @project, :issue_id => @issue }),
32 :class => (@controller.action_name == 'index' ? 'selected' : nil)) %></li>
33 33 <li><%= link_to(l(:label_report), url_params.merge({:controller => 'time_entry_reports', :action => 'report', :project_id => @project, :issue_id => @issue}),
34 34 :class => (@controller.action_name == 'report' ? 'selected' : nil)) %></li>
35 35 </ul>
36 36 </div>
1 NO CONTENT: file renamed from app/views/timelog/details.rhtml to app/views/timelog/index.html.erb
@@ -1,258 +1,258
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 timelog.connect 'projects/:project_id/time_entries', :action => 'details'
22 timelog.connect 'projects/:project_id/time_entries', :action => 'index'
23 23
24 timelog.with_options :action => 'details', :conditions => {:method => :get} do |time_details|
24 timelog.with_options :action => 'index', :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', :controller => 'time_entry_reports', :action => 'report'
34 34 timelog.with_options :controller => 'time_entry_reports', :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.connect 'projects/:id/members/new', :controller => 'members', :action => 'new'
141 141
142 142 map.with_options :controller => 'users' do |users|
143 143 users.connect 'users/:id/edit/:tab', :action => 'edit', :tab => nil, :conditions => {:method => :get}
144 144
145 145 users.with_options :conditions => {:method => :post} do |user_actions|
146 146 user_actions.connect 'users/:id/memberships', :action => 'edit_membership'
147 147 user_actions.connect 'users/:id/memberships/:membership_id', :action => 'edit_membership'
148 148 user_actions.connect 'users/:id/memberships/:membership_id/destroy', :action => 'destroy_membership'
149 149 end
150 150 end
151 151
152 152 map.resources :users, :member => {
153 153 :edit_membership => :post,
154 154 :destroy_membership => :post
155 155 },
156 156 :except => [:destroy]
157 157
158 158 # For nice "roadmap" in the url for the index action
159 159 map.connect 'projects/:project_id/roadmap', :controller => 'versions', :action => 'index'
160 160
161 161 map.all_news 'news', :controller => 'news', :action => 'index'
162 162 map.formatted_all_news 'news.:format', :controller => 'news', :action => 'index'
163 163 map.preview_news '/news/preview', :controller => 'previews', :action => 'news'
164 164 map.connect 'news/:id/comments', :controller => 'comments', :action => 'create', :conditions => {:method => :post}
165 165 map.connect 'news/:id/comments/:comment_id', :controller => 'comments', :action => 'destroy', :conditions => {:method => :delete}
166 166
167 167 map.resources :projects, :member => {
168 168 :copy => [:get, :post],
169 169 :settings => :get,
170 170 :modules => :post,
171 171 :archive => :post,
172 172 :unarchive => :post
173 173 } do |project|
174 174 project.resource :project_enumerations, :as => 'enumerations', :only => [:update, :destroy]
175 175 project.resources :files, :only => [:index, :new, :create]
176 176 project.resources :versions, :collection => {:close_completed => :put}, :member => {:status_by => :post}
177 177 project.resources :news, :shallow => true
178 178 end
179 179
180 180 # Destroy uses a get request to prompt the user before the actual DELETE request
181 181 map.project_destroy_confirm 'projects/:id/destroy', :controller => 'projects', :action => 'destroy', :conditions => {:method => :get}
182 182
183 183 # TODO: port to be part of the resources route(s)
184 184 map.with_options :controller => 'projects' do |project_mapper|
185 185 project_mapper.with_options :conditions => {:method => :get} do |project_views|
186 186 project_views.connect 'projects/:id/settings/:tab', :controller => 'projects', :action => 'settings'
187 187 project_views.connect 'projects/:project_id/issues/:copy_from/copy', :controller => 'issues', :action => 'new'
188 188 end
189 189 end
190 190
191 191 map.with_options :controller => 'activities', :action => 'index', :conditions => {:method => :get} do |activity|
192 192 activity.connect 'projects/:id/activity'
193 193 activity.connect 'projects/:id/activity.:format'
194 194 activity.connect 'activity', :id => nil
195 195 activity.connect 'activity.:format', :id => nil
196 196 end
197 197
198 198
199 199 map.with_options :controller => 'issue_categories' do |categories|
200 200 categories.connect 'projects/:project_id/issue_categories/new', :action => 'new'
201 201 end
202 202
203 203 map.with_options :controller => 'repositories' do |repositories|
204 204 repositories.with_options :conditions => {:method => :get} do |repository_views|
205 205 repository_views.connect 'projects/:id/repository', :action => 'show'
206 206 repository_views.connect 'projects/:id/repository/edit', :action => 'edit'
207 207 repository_views.connect 'projects/:id/repository/statistics', :action => 'stats'
208 208 repository_views.connect 'projects/:id/repository/revisions', :action => 'revisions'
209 209 repository_views.connect 'projects/:id/repository/revisions.:format', :action => 'revisions'
210 210 repository_views.connect 'projects/:id/repository/revisions/:rev', :action => 'revision'
211 211 repository_views.connect 'projects/:id/repository/revisions/:rev/diff', :action => 'diff'
212 212 repository_views.connect 'projects/:id/repository/revisions/:rev/diff.:format', :action => 'diff'
213 213 repository_views.connect 'projects/:id/repository/revisions/:rev/raw/*path', :action => 'entry', :format => 'raw', :requirements => { :rev => /[a-z0-9\.\-_]+/ }
214 214 repository_views.connect 'projects/:id/repository/revisions/:rev/:action/*path', :requirements => { :rev => /[a-z0-9\.\-_]+/ }
215 215 repository_views.connect 'projects/:id/repository/raw/*path', :action => 'entry', :format => 'raw'
216 216 # TODO: why the following route is required?
217 217 repository_views.connect 'projects/:id/repository/entry/*path', :action => 'entry'
218 218 repository_views.connect 'projects/:id/repository/:action/*path'
219 219 end
220 220
221 221 repositories.connect 'projects/:id/repository/:action', :conditions => {:method => :post}
222 222 end
223 223
224 224 map.connect 'attachments/:id', :controller => 'attachments', :action => 'show', :id => /\d+/
225 225 map.connect 'attachments/:id/:filename', :controller => 'attachments', :action => 'show', :id => /\d+/, :filename => /.*/
226 226 map.connect 'attachments/download/:id/:filename', :controller => 'attachments', :action => 'download', :id => /\d+/, :filename => /.*/
227 227
228 228 map.resources :groups
229 229
230 230 #left old routes at the bottom for backwards compat
231 231 map.connect 'projects/:project_id/issues/:action', :controller => 'issues'
232 232 map.connect 'projects/:project_id/documents/:action', :controller => 'documents'
233 233 map.connect 'projects/:project_id/boards/:action/:id', :controller => 'boards'
234 234 map.connect 'boards/:board_id/topics/:action/:id', :controller => 'messages'
235 235 map.connect 'wiki/:id/:page/:action', :page => nil, :controller => 'wiki'
236 236 map.connect 'issues/:issue_id/relations/:action/:id', :controller => 'issue_relations'
237 237 map.connect 'projects/:project_id/news/:action', :controller => 'news'
238 238 map.connect 'projects/:project_id/timelog/:action/:id', :controller => 'timelog', :project_id => /.+/
239 239 map.with_options :controller => 'repositories' do |omap|
240 240 omap.repositories_show 'repositories/browse/:id/*path', :action => 'browse'
241 241 omap.repositories_changes 'repositories/changes/:id/*path', :action => 'changes'
242 242 omap.repositories_diff 'repositories/diff/:id/*path', :action => 'diff'
243 243 omap.repositories_entry 'repositories/entry/:id/*path', :action => 'entry'
244 244 omap.repositories_entry 'repositories/annotate/:id/*path', :action => 'annotate'
245 245 omap.connect 'repositories/revision/:id/:rev', :action => 'revision'
246 246 end
247 247
248 248 map.with_options :controller => 'sys' do |sys|
249 249 sys.connect 'sys/projects.:format', :action => 'projects', :conditions => {:method => :get}
250 250 sys.connect 'sys/projects/:id/repository.:format', :action => 'create_project_repository', :conditions => {:method => :post}
251 251 end
252 252
253 253 # Install the default route as the lowest priority.
254 254 map.connect ':controller/:action/:id'
255 255 map.connect 'robots.txt', :controller => 'welcome', :action => 'robots'
256 256 # Used for OpenID
257 257 map.root :controller => 'account', :action => 'login'
258 258 end
@@ -1,231 +1,231
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/notifiable'
12 12 require 'redmine/wiki_formatting'
13 13 require 'redmine/scm/base'
14 14
15 15 begin
16 16 require_library_or_gem 'RMagick' unless Object.const_defined?(:Magick)
17 17 rescue LoadError
18 18 # RMagick is not available
19 19 end
20 20
21 21 if RUBY_VERSION < '1.9'
22 22 require 'faster_csv'
23 23 else
24 24 require 'csv'
25 25 FCSV = CSV
26 26 end
27 27
28 28 Redmine::Scm::Base.add "Subversion"
29 29 Redmine::Scm::Base.add "Darcs"
30 30 Redmine::Scm::Base.add "Mercurial"
31 31 Redmine::Scm::Base.add "Cvs"
32 32 Redmine::Scm::Base.add "Bazaar"
33 33 Redmine::Scm::Base.add "Git"
34 34 Redmine::Scm::Base.add "Filesystem"
35 35
36 36 Redmine::CustomFieldFormat.map do |fields|
37 37 fields.register Redmine::CustomFieldFormat.new('string', :label => :label_string, :order => 1)
38 38 fields.register Redmine::CustomFieldFormat.new('text', :label => :label_text, :order => 2)
39 39 fields.register Redmine::CustomFieldFormat.new('int', :label => :label_integer, :order => 3)
40 40 fields.register Redmine::CustomFieldFormat.new('float', :label => :label_float, :order => 4)
41 41 fields.register Redmine::CustomFieldFormat.new('list', :label => :label_list, :order => 5)
42 42 fields.register Redmine::CustomFieldFormat.new('date', :label => :label_date, :order => 6)
43 43 fields.register Redmine::CustomFieldFormat.new('bool', :label => :label_boolean, :order => 7)
44 44 end
45 45
46 46 # Permissions
47 47 Redmine::AccessControl.map do |map|
48 48 map.permission :view_project, {:projects => [:show], :activities => [:index]}, :public => true
49 49 map.permission :search_project, {:search => :index}, :public => true
50 50 map.permission :add_project, {:projects => [:new, :create]}, :require => :loggedin
51 51 map.permission :edit_project, {:projects => [:settings, :edit, :update]}, :require => :member
52 52 map.permission :select_project_modules, {:projects => :modules}, :require => :member
53 53 map.permission :manage_members, {:projects => :settings, :members => [:new, :edit, :destroy, :autocomplete_for_member]}, :require => :member
54 54 map.permission :manage_versions, {:projects => :settings, :versions => [:new, :create, :edit, :update, :close_completed, :destroy]}, :require => :member
55 55 map.permission :add_subprojects, {:projects => [:new, :create]}, :require => :member
56 56
57 57 map.project_module :issue_tracking do |map|
58 58 # Issue categories
59 59 map.permission :manage_categories, {:projects => :settings, :issue_categories => [:new, :edit, :destroy]}, :require => :member
60 60 # Issues
61 61 map.permission :view_issues, {:issues => [:index, :show],
62 62 :auto_complete => [:issues],
63 63 :context_menus => [:issues],
64 64 :versions => [:index, :show, :status_by],
65 65 :journals => :index,
66 66 :queries => :index,
67 67 :reports => [:issue_report, :issue_report_details]}
68 68 map.permission :add_issues, {:issues => [:new, :create, :update_form]}
69 69 map.permission :edit_issues, {:issues => [:edit, :update, :bulk_edit, :bulk_update, :update_form], :journals => [:new]}
70 70 map.permission :manage_issue_relations, {:issue_relations => [:new, :destroy]}
71 71 map.permission :manage_subtasks, {}
72 72 map.permission :add_issue_notes, {:issues => [:edit, :update], :journals => [:new]}
73 73 map.permission :edit_issue_notes, {:journals => :edit}, :require => :loggedin
74 74 map.permission :edit_own_issue_notes, {:journals => :edit}, :require => :loggedin
75 75 map.permission :move_issues, {:issue_moves => [:new, :create]}, :require => :loggedin
76 76 map.permission :delete_issues, {:issues => :destroy}, :require => :member
77 77 # Queries
78 78 map.permission :manage_public_queries, {:queries => [:new, :edit, :destroy]}, :require => :member
79 79 map.permission :save_queries, {:queries => [:new, :edit, :destroy]}, :require => :loggedin
80 80 # Watchers
81 81 map.permission :view_issue_watchers, {}
82 82 map.permission :add_issue_watchers, {:watchers => :new}
83 83 map.permission :delete_issue_watchers, {:watchers => :destroy}
84 84 end
85 85
86 86 map.project_module :time_tracking do |map|
87 87 map.permission :log_time, {:timelog => :edit}, :require => :loggedin
88 map.permission :view_time_entries, :timelog => [:details], :time_entry_reports => [:report]
88 map.permission :view_time_entries, :timelog => [:index], :time_entry_reports => [:report]
89 89 map.permission :edit_time_entries, {:timelog => [:edit, :destroy]}, :require => :member
90 90 map.permission :edit_own_time_entries, {:timelog => [:edit, :destroy]}, :require => :loggedin
91 91 map.permission :manage_project_activities, {:project_enumerations => [:update, :destroy]}, :require => :member
92 92 end
93 93
94 94 map.project_module :news do |map|
95 95 map.permission :manage_news, {:news => [:new, :create, :edit, :update, :destroy], :comments => [:destroy]}, :require => :member
96 96 map.permission :view_news, {:news => [:index, :show]}, :public => true
97 97 map.permission :comment_news, {:comments => :create}
98 98 end
99 99
100 100 map.project_module :documents do |map|
101 101 map.permission :manage_documents, {:documents => [:new, :edit, :destroy, :add_attachment]}, :require => :loggedin
102 102 map.permission :view_documents, :documents => [:index, :show, :download]
103 103 end
104 104
105 105 map.project_module :files do |map|
106 106 map.permission :manage_files, {:files => [:new, :create]}, :require => :loggedin
107 107 map.permission :view_files, :files => :index, :versions => :download
108 108 end
109 109
110 110 map.project_module :wiki do |map|
111 111 map.permission :manage_wiki, {:wikis => [:edit, :destroy]}, :require => :member
112 112 map.permission :rename_wiki_pages, {:wiki => :rename}, :require => :member
113 113 map.permission :delete_wiki_pages, {:wiki => :destroy}, :require => :member
114 114 map.permission :view_wiki_pages, :wiki => [:index, :special]
115 115 map.permission :export_wiki_pages, {}
116 116 map.permission :view_wiki_edits, :wiki => [:history, :diff, :annotate]
117 117 map.permission :edit_wiki_pages, :wiki => [:edit, :preview, :add_attachment]
118 118 map.permission :delete_wiki_pages_attachments, {}
119 119 map.permission :protect_wiki_pages, {:wiki => :protect}, :require => :member
120 120 end
121 121
122 122 map.project_module :repository do |map|
123 123 map.permission :manage_repository, {:repositories => [:edit, :committers, :destroy]}, :require => :member
124 124 map.permission :browse_repository, :repositories => [:show, :browse, :entry, :annotate, :changes, :diff, :stats, :graph]
125 125 map.permission :view_changesets, :repositories => [:show, :revisions, :revision]
126 126 map.permission :commit_access, {}
127 127 end
128 128
129 129 map.project_module :boards do |map|
130 130 map.permission :manage_boards, {:boards => [:new, :edit, :destroy]}, :require => :member
131 131 map.permission :view_messages, {:boards => [:index, :show], :messages => [:show]}, :public => true
132 132 map.permission :add_messages, {:messages => [:new, :reply, :quote]}
133 133 map.permission :edit_messages, {:messages => :edit}, :require => :member
134 134 map.permission :edit_own_messages, {:messages => :edit}, :require => :loggedin
135 135 map.permission :delete_messages, {:messages => :destroy}, :require => :member
136 136 map.permission :delete_own_messages, {:messages => :destroy}, :require => :loggedin
137 137 end
138 138
139 139 map.project_module :calendar do |map|
140 140 map.permission :view_calendar, :calendars => [:show, :update]
141 141 end
142 142
143 143 map.project_module :gantt do |map|
144 144 map.permission :view_gantt, :gantts => [:show, :update]
145 145 end
146 146 end
147 147
148 148 Redmine::MenuManager.map :top_menu do |menu|
149 149 menu.push :home, :home_path
150 150 menu.push :my_page, { :controller => 'my', :action => 'page' }, :if => Proc.new { User.current.logged? }
151 151 menu.push :projects, { :controller => 'projects', :action => 'index' }, :caption => :label_project_plural
152 152 menu.push :administration, { :controller => 'admin', :action => 'index' }, :if => Proc.new { User.current.admin? }, :last => true
153 153 menu.push :help, Redmine::Info.help_url, :last => true
154 154 end
155 155
156 156 Redmine::MenuManager.map :account_menu do |menu|
157 157 menu.push :login, :signin_path, :if => Proc.new { !User.current.logged? }
158 158 menu.push :register, { :controller => 'account', :action => 'register' }, :if => Proc.new { !User.current.logged? && Setting.self_registration? }
159 159 menu.push :my_account, { :controller => 'my', :action => 'account' }, :if => Proc.new { User.current.logged? }
160 160 menu.push :logout, :signout_path, :if => Proc.new { User.current.logged? }
161 161 end
162 162
163 163 Redmine::MenuManager.map :application_menu do |menu|
164 164 # Empty
165 165 end
166 166
167 167 Redmine::MenuManager.map :admin_menu do |menu|
168 168 menu.push :projects, {:controller => 'admin', :action => 'projects'}, :caption => :label_project_plural
169 169 menu.push :users, {:controller => 'users'}, :caption => :label_user_plural
170 170 menu.push :groups, {:controller => 'groups'}, :caption => :label_group_plural
171 171 menu.push :roles, {:controller => 'roles'}, :caption => :label_role_and_permissions
172 172 menu.push :trackers, {:controller => 'trackers'}, :caption => :label_tracker_plural
173 173 menu.push :issue_statuses, {:controller => 'issue_statuses'}, :caption => :label_issue_status_plural,
174 174 :html => {:class => 'issue_statuses'}
175 175 menu.push :workflows, {:controller => 'workflows', :action => 'edit'}, :caption => :label_workflow
176 176 menu.push :custom_fields, {:controller => 'custom_fields'}, :caption => :label_custom_field_plural,
177 177 :html => {:class => 'custom_fields'}
178 178 menu.push :enumerations, {:controller => 'enumerations'}
179 179 menu.push :settings, {:controller => 'settings'}
180 180 menu.push :ldap_authentication, {:controller => 'ldap_auth_sources', :action => 'index'},
181 181 :html => {:class => 'server_authentication'}
182 182 menu.push :plugins, {:controller => 'admin', :action => 'plugins'}, :last => true
183 183 menu.push :info, {:controller => 'admin', :action => 'info'}, :caption => :label_information_plural, :last => true
184 184 end
185 185
186 186 Redmine::MenuManager.map :project_menu do |menu|
187 187 menu.push :overview, { :controller => 'projects', :action => 'show' }
188 188 menu.push :activity, { :controller => 'activities', :action => 'index' }
189 189 menu.push :roadmap, { :controller => 'versions', :action => 'index' }, :param => :project_id,
190 190 :if => Proc.new { |p| p.shared_versions.any? }
191 191 menu.push :issues, { :controller => 'issues', :action => 'index' }, :param => :project_id, :caption => :label_issue_plural
192 192 menu.push :new_issue, { :controller => 'issues', :action => 'new' }, :param => :project_id, :caption => :label_issue_new,
193 193 :html => { :accesskey => Redmine::AccessKeys.key_for(:new_issue) }
194 194 menu.push :gantt, { :controller => 'gantts', :action => 'show' }, :param => :project_id, :caption => :label_gantt
195 195 menu.push :calendar, { :controller => 'calendars', :action => 'show' }, :param => :project_id, :caption => :label_calendar
196 196 menu.push :news, { :controller => 'news', :action => 'index' }, :param => :project_id, :caption => :label_news_plural
197 197 menu.push :documents, { :controller => 'documents', :action => 'index' }, :param => :project_id, :caption => :label_document_plural
198 198 menu.push :wiki, { :controller => 'wiki', :action => 'index', :page => nil },
199 199 :if => Proc.new { |p| p.wiki && !p.wiki.new_record? }
200 200 menu.push :boards, { :controller => 'boards', :action => 'index', :id => nil }, :param => :project_id,
201 201 :if => Proc.new { |p| p.boards.any? }, :caption => :label_board_plural
202 202 menu.push :files, { :controller => 'files', :action => 'index' }, :caption => :label_file_plural, :param => :project_id
203 203 menu.push :repository, { :controller => 'repositories', :action => 'show' },
204 204 :if => Proc.new { |p| p.repository && !p.repository.new_record? }
205 205 menu.push :settings, { :controller => 'projects', :action => 'settings' }, :last => true
206 206 end
207 207
208 208 Redmine::Activity.map do |activity|
209 209 activity.register :issues, :class_name => %w(Issue Journal)
210 210 activity.register :changesets
211 211 activity.register :news
212 212 activity.register :documents, :class_name => %w(Document Attachment)
213 213 activity.register :files, :class_name => 'Attachment'
214 214 activity.register :wiki_edits, :class_name => 'WikiContent::Version', :default => false
215 215 activity.register :messages, :default => false
216 216 activity.register :time_entries, :default => false
217 217 end
218 218
219 219 Redmine::Search.map do |search|
220 220 search.register :issues
221 221 search.register :news
222 222 search.register :documents
223 223 search.register :changesets
224 224 search.register :wiki_pages
225 225 search.register :messages
226 226 search.register :projects
227 227 end
228 228
229 229 Redmine::WikiFormatting.map do |format|
230 230 format.register :textile, Redmine::WikiFormatting::Textile::Formatter, Redmine::WikiFormatting::Textile::Helper
231 231 end
@@ -1,231 +1,231
1 1 # -*- coding: utf-8 -*-
2 2 # redMine - project management software
3 3 # Copyright (C) 2006-2007 Jean-Philippe Lang
4 4 #
5 5 # This program is free software; you can redistribute it and/or
6 6 # modify it under the terms of the GNU General Public License
7 7 # as published by the Free Software Foundation; either version 2
8 8 # of the License, or (at your option) any later version.
9 9 #
10 10 # This program is distributed in the hope that it will be useful,
11 11 # but WITHOUT ANY WARRANTY; without even the implied warranty of
12 12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 13 # GNU General Public License for more details.
14 14 #
15 15 # You should have received a copy of the GNU General Public License
16 16 # along with this program; if not, write to the Free Software
17 17 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 18
19 19 require File.dirname(__FILE__) + '/../test_helper'
20 20 require 'timelog_controller'
21 21
22 22 # Re-raise errors caught by the controller.
23 23 class TimelogController; def rescue_action(e) raise e end; end
24 24
25 25 class TimelogControllerTest < ActionController::TestCase
26 26 fixtures :projects, :enabled_modules, :roles, :members, :member_roles, :issues, :time_entries, :users, :trackers, :enumerations, :issue_statuses, :custom_fields, :custom_values
27 27
28 28 def setup
29 29 @controller = TimelogController.new
30 30 @request = ActionController::TestRequest.new
31 31 @response = ActionController::TestResponse.new
32 32 end
33 33
34 34 def test_get_edit
35 35 @request.session[:user_id] = 3
36 36 get :edit, :project_id => 1
37 37 assert_response :success
38 38 assert_template 'edit'
39 39 # Default activity selected
40 40 assert_tag :tag => 'option', :attributes => { :selected => 'selected' },
41 41 :content => 'Development'
42 42 end
43 43
44 44 def test_get_edit_existing_time
45 45 @request.session[:user_id] = 2
46 46 get :edit, :id => 2, :project_id => nil
47 47 assert_response :success
48 48 assert_template 'edit'
49 49 # Default activity selected
50 50 assert_tag :tag => 'form', :attributes => { :action => '/projects/ecookbook/timelog/edit/2' }
51 51 end
52 52
53 53 def test_get_edit_should_only_show_active_time_entry_activities
54 54 @request.session[:user_id] = 3
55 55 get :edit, :project_id => 1
56 56 assert_response :success
57 57 assert_template 'edit'
58 58 assert_no_tag :tag => 'option', :content => 'Inactive Activity'
59 59
60 60 end
61 61
62 62 def test_get_edit_with_an_existing_time_entry_with_inactive_activity
63 63 te = TimeEntry.find(1)
64 64 te.activity = TimeEntryActivity.find_by_name("Inactive Activity")
65 65 te.save!
66 66
67 67 @request.session[:user_id] = 1
68 68 get :edit, :project_id => 1, :id => 1
69 69 assert_response :success
70 70 assert_template 'edit'
71 71 # Blank option since nothing is pre-selected
72 72 assert_tag :tag => 'option', :content => '--- Please select ---'
73 73 end
74 74
75 75 def test_post_edit
76 76 # TODO: should POST to issues’ time log instead of project. change form
77 77 # and routing
78 78 @request.session[:user_id] = 3
79 79 post :edit, :project_id => 1,
80 80 :time_entry => {:comments => 'Some work on TimelogControllerTest',
81 81 # Not the default activity
82 82 :activity_id => '11',
83 83 :spent_on => '2008-03-14',
84 84 :issue_id => '1',
85 85 :hours => '7.3'}
86 assert_redirected_to :action => 'details', :project_id => 'ecookbook'
86 assert_redirected_to :action => 'index', :project_id => 'ecookbook'
87 87
88 88 i = Issue.find(1)
89 89 t = TimeEntry.find_by_comments('Some work on TimelogControllerTest')
90 90 assert_not_nil t
91 91 assert_equal 11, t.activity_id
92 92 assert_equal 7.3, t.hours
93 93 assert_equal 3, t.user_id
94 94 assert_equal i, t.issue
95 95 assert_equal i.project, t.project
96 96 end
97 97
98 98 def test_update
99 99 entry = TimeEntry.find(1)
100 100 assert_equal 1, entry.issue_id
101 101 assert_equal 2, entry.user_id
102 102
103 103 @request.session[:user_id] = 1
104 104 post :edit, :id => 1,
105 105 :time_entry => {:issue_id => '2',
106 106 :hours => '8'}
107 assert_redirected_to :action => 'details', :project_id => 'ecookbook'
107 assert_redirected_to :action => 'index', :project_id => 'ecookbook'
108 108 entry.reload
109 109
110 110 assert_equal 8, entry.hours
111 111 assert_equal 2, entry.issue_id
112 112 assert_equal 2, entry.user_id
113 113 end
114 114
115 115 def test_destroy
116 116 @request.session[:user_id] = 2
117 117 post :destroy, :id => 1
118 assert_redirected_to :action => 'details', :project_id => 'ecookbook'
118 assert_redirected_to :action => 'index', :project_id => 'ecookbook'
119 119 assert_equal I18n.t(:notice_successful_delete), flash[:notice]
120 120 assert_nil TimeEntry.find_by_id(1)
121 121 end
122 122
123 123 def test_destroy_should_fail
124 124 # simulate that this fails (e.g. due to a plugin), see #5700
125 125 TimeEntry.class_eval do
126 126 before_destroy :stop_callback_chain
127 127 def stop_callback_chain ; return false ; end
128 128 end
129 129
130 130 @request.session[:user_id] = 2
131 131 post :destroy, :id => 1
132 assert_redirected_to :action => 'details', :project_id => 'ecookbook'
132 assert_redirected_to :action => 'index', :project_id => 'ecookbook'
133 133 assert_equal I18n.t(:notice_unable_delete_time_entry), flash[:error]
134 134 assert_not_nil TimeEntry.find_by_id(1)
135 135
136 136 # remove the simulation
137 137 TimeEntry.before_destroy.reject! {|callback| callback.method == :stop_callback_chain }
138 138 end
139 139
140 def test_details_all_projects
141 get :details
140 def test_index_all_projects
141 get :index
142 142 assert_response :success
143 assert_template 'details'
143 assert_template 'index'
144 144 assert_not_nil assigns(:total_hours)
145 145 assert_equal "162.90", "%.2f" % assigns(:total_hours)
146 146 end
147 147
148 def test_details_at_project_level
149 get :details, :project_id => 1
148 def test_index_at_project_level
149 get :index, :project_id => 1
150 150 assert_response :success
151 assert_template 'details'
151 assert_template 'index'
152 152 assert_not_nil assigns(:entries)
153 153 assert_equal 4, assigns(:entries).size
154 154 # project and subproject
155 155 assert_equal [1, 3], assigns(:entries).collect(&:project_id).uniq.sort
156 156 assert_not_nil assigns(:total_hours)
157 157 assert_equal "162.90", "%.2f" % assigns(:total_hours)
158 158 # display all time by default
159 159 assert_equal '2007-03-12'.to_date, assigns(:from)
160 160 assert_equal '2007-04-22'.to_date, assigns(:to)
161 161 end
162 162
163 def test_details_at_project_level_with_date_range
164 get :details, :project_id => 1, :from => '2007-03-20', :to => '2007-04-30'
163 def test_index_at_project_level_with_date_range
164 get :index, :project_id => 1, :from => '2007-03-20', :to => '2007-04-30'
165 165 assert_response :success
166 assert_template 'details'
166 assert_template 'index'
167 167 assert_not_nil assigns(:entries)
168 168 assert_equal 3, assigns(:entries).size
169 169 assert_not_nil assigns(:total_hours)
170 170 assert_equal "12.90", "%.2f" % assigns(:total_hours)
171 171 assert_equal '2007-03-20'.to_date, assigns(:from)
172 172 assert_equal '2007-04-30'.to_date, assigns(:to)
173 173 end
174 174
175 def test_details_at_project_level_with_period
176 get :details, :project_id => 1, :period => '7_days'
175 def test_index_at_project_level_with_period
176 get :index, :project_id => 1, :period => '7_days'
177 177 assert_response :success
178 assert_template 'details'
178 assert_template 'index'
179 179 assert_not_nil assigns(:entries)
180 180 assert_not_nil assigns(:total_hours)
181 181 assert_equal Date.today - 7, assigns(:from)
182 182 assert_equal Date.today, assigns(:to)
183 183 end
184 184
185 def test_details_one_day
186 get :details, :project_id => 1, :from => "2007-03-23", :to => "2007-03-23"
185 def test_index_one_day
186 get :index, :project_id => 1, :from => "2007-03-23", :to => "2007-03-23"
187 187 assert_response :success
188 assert_template 'details'
188 assert_template 'index'
189 189 assert_not_nil assigns(:total_hours)
190 190 assert_equal "4.25", "%.2f" % assigns(:total_hours)
191 191 end
192 192
193 def test_details_at_issue_level
194 get :details, :issue_id => 1
193 def test_index_at_issue_level
194 get :index, :issue_id => 1
195 195 assert_response :success
196 assert_template 'details'
196 assert_template 'index'
197 197 assert_not_nil assigns(:entries)
198 198 assert_equal 2, assigns(:entries).size
199 199 assert_not_nil assigns(:total_hours)
200 200 assert_equal 154.25, assigns(:total_hours)
201 201 # display all time based on what's been logged
202 202 assert_equal '2007-03-12'.to_date, assigns(:from)
203 203 assert_equal '2007-04-22'.to_date, assigns(:to)
204 204 end
205 205
206 def test_details_atom_feed
207 get :details, :project_id => 1, :format => 'atom'
206 def test_index_atom_feed
207 get :index, :project_id => 1, :format => 'atom'
208 208 assert_response :success
209 209 assert_equal 'application/atom+xml', @response.content_type
210 210 assert_not_nil assigns(:items)
211 211 assert assigns(:items).first.is_a?(TimeEntry)
212 212 end
213 213
214 def test_details_all_projects_csv_export
214 def test_index_all_projects_csv_export
215 215 Setting.date_format = '%m/%d/%Y'
216 get :details, :format => 'csv'
216 get :index, :format => 'csv'
217 217 assert_response :success
218 218 assert_equal 'text/csv', @response.content_type
219 219 assert @response.body.include?("Date,User,Activity,Project,Issue,Tracker,Subject,Hours,Comment\n")
220 220 assert @response.body.include?("\n04/21/2007,redMine Admin,Design,eCookbook,3,Bug,Error 281 when updating a recipe,1.0,\"\"\n")
221 221 end
222 222
223 def test_details_csv_export
223 def test_index_csv_export
224 224 Setting.date_format = '%m/%d/%Y'
225 get :details, :project_id => 1, :format => 'csv'
225 get :index, :project_id => 1, :format => 'csv'
226 226 assert_response :success
227 227 assert_equal 'text/csv', @response.content_type
228 228 assert @response.body.include?("Date,User,Activity,Project,Issue,Tracker,Subject,Hours,Comment\n")
229 229 assert @response.body.include?("\n04/21/2007,redMine Admin,Design,eCookbook,3,Bug,Error 281 when updating a recipe,1.0,\"\"\n")
230 230 end
231 231 end
@@ -1,308 +1,307
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 should_route :get, "/news/567/edit", :controller => 'news', :action => 'edit', :id => '567'
160 160 should_route :get, "/news/preview", :controller => 'previews', :action => 'news'
161 161
162 162 should_route :post, "/projects/567/news", :controller => 'news', :action => 'create', :project_id => '567'
163 163 should_route :post, "/news/567/comments", :controller => 'comments', :action => 'create', :id => '567'
164 164
165 165 should_route :put, "/news/567", :controller => 'news', :action => 'update', :id => '567'
166 166
167 167 should_route :delete, "/news/567", :controller => 'news', :action => 'destroy', :id => '567'
168 168 should_route :delete, "/news/567/comments/15", :controller => 'comments', :action => 'destroy', :id => '567', :comment_id => '15'
169 169 end
170 170
171 171 context "projects" do
172 172 should_route :get, "/projects", :controller => 'projects', :action => 'index'
173 173 should_route :get, "/projects.atom", :controller => 'projects', :action => 'index', :format => 'atom'
174 174 should_route :get, "/projects.xml", :controller => 'projects', :action => 'index', :format => 'xml'
175 175 should_route :get, "/projects/new", :controller => 'projects', :action => 'new'
176 176 should_route :get, "/projects/test", :controller => 'projects', :action => 'show', :id => 'test'
177 177 should_route :get, "/projects/1.xml", :controller => 'projects', :action => 'show', :id => '1', :format => 'xml'
178 178 should_route :get, "/projects/4223/settings", :controller => 'projects', :action => 'settings', :id => '4223'
179 179 should_route :get, "/projects/4223/settings/members", :controller => 'projects', :action => 'settings', :id => '4223', :tab => 'members'
180 180 should_route :get, "/projects/33/files", :controller => 'files', :action => 'index', :project_id => '33'
181 181 should_route :get, "/projects/33/files/new", :controller => 'files', :action => 'new', :project_id => '33'
182 182 should_route :get, "/projects/33/roadmap", :controller => 'versions', :action => 'index', :project_id => '33'
183 183 should_route :get, "/projects/33/activity", :controller => 'activities', :action => 'index', :id => '33'
184 184 should_route :get, "/projects/33/activity.atom", :controller => 'activities', :action => 'index', :id => '33', :format => 'atom'
185 185
186 186 should_route :post, "/projects", :controller => 'projects', :action => 'create'
187 187 should_route :post, "/projects.xml", :controller => 'projects', :action => 'create', :format => 'xml'
188 188 should_route :post, "/projects/33/files", :controller => 'files', :action => 'create', :project_id => '33'
189 189 should_route :post, "/projects/64/archive", :controller => 'projects', :action => 'archive', :id => '64'
190 190 should_route :post, "/projects/64/unarchive", :controller => 'projects', :action => 'unarchive', :id => '64'
191 191
192 192 should_route :put, "/projects/64/enumerations", :controller => 'project_enumerations', :action => 'update', :project_id => '64'
193 193 should_route :put, "/projects/4223", :controller => 'projects', :action => 'update', :id => '4223'
194 194 should_route :put, "/projects/1.xml", :controller => 'projects', :action => 'update', :id => '1', :format => 'xml'
195 195
196 196 should_route :delete, "/projects/64", :controller => 'projects', :action => 'destroy', :id => '64'
197 197 should_route :delete, "/projects/1.xml", :controller => 'projects', :action => 'destroy', :id => '1', :format => 'xml'
198 198 should_route :delete, "/projects/64/enumerations", :controller => 'project_enumerations', :action => 'destroy', :project_id => '64'
199 199 end
200 200
201 201 context "repositories" do
202 202 should_route :get, "/projects/redmine/repository", :controller => 'repositories', :action => 'show', :id => 'redmine'
203 203 should_route :get, "/projects/redmine/repository/edit", :controller => 'repositories', :action => 'edit', :id => 'redmine'
204 204 should_route :get, "/projects/redmine/repository/revisions", :controller => 'repositories', :action => 'revisions', :id => 'redmine'
205 205 should_route :get, "/projects/redmine/repository/revisions.atom", :controller => 'repositories', :action => 'revisions', :id => 'redmine', :format => 'atom'
206 206 should_route :get, "/projects/redmine/repository/revisions/2457", :controller => 'repositories', :action => 'revision', :id => 'redmine', :rev => '2457'
207 207 should_route :get, "/projects/redmine/repository/revisions/2457/diff", :controller => 'repositories', :action => 'diff', :id => 'redmine', :rev => '2457'
208 208 should_route :get, "/projects/redmine/repository/revisions/2457/diff.diff", :controller => 'repositories', :action => 'diff', :id => 'redmine', :rev => '2457', :format => 'diff'
209 209 should_route :get, "/projects/redmine/repository/diff/path/to/file.c", :controller => 'repositories', :action => 'diff', :id => 'redmine', :path => %w[path to file.c]
210 210 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'
211 211 should_route :get, "/projects/redmine/repository/browse/path/to/file.c", :controller => 'repositories', :action => 'browse', :id => 'redmine', :path => %w[path to file.c]
212 212 should_route :get, "/projects/redmine/repository/entry/path/to/file.c", :controller => 'repositories', :action => 'entry', :id => 'redmine', :path => %w[path to file.c]
213 213 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'
214 214 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'
215 215 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'
216 216 should_route :get, "/projects/redmine/repository/annotate/path/to/file.c", :controller => 'repositories', :action => 'annotate', :id => 'redmine', :path => %w[path to file.c]
217 217 should_route :get, "/projects/redmine/repository/changes/path/to/file.c", :controller => 'repositories', :action => 'changes', :id => 'redmine', :path => %w[path to file.c]
218 218 should_route :get, "/projects/redmine/repository/statistics", :controller => 'repositories', :action => 'stats', :id => 'redmine'
219 219
220 220
221 221 should_route :post, "/projects/redmine/repository/edit", :controller => 'repositories', :action => 'edit', :id => 'redmine'
222 222 end
223 223
224 224 context "timelogs" do
225 should_route :get, "/time_entries", :controller => 'timelog', :action => 'index'
226 should_route :get, "/time_entries.csv", :controller => 'timelog', :action => 'index', :format => 'csv'
227 should_route :get, "/time_entries.atom", :controller => 'timelog', :action => 'index', :format => 'atom'
228 should_route :get, "/projects/567/time_entries", :controller => 'timelog', :action => 'index', :project_id => '567'
229 should_route :get, "/projects/567/time_entries.csv", :controller => 'timelog', :action => 'index', :project_id => '567', :format => 'csv'
230 should_route :get, "/projects/567/time_entries.atom", :controller => 'timelog', :action => 'index', :project_id => '567', :format => 'atom'
231 should_route :get, "/issues/234/time_entries", :controller => 'timelog', :action => 'index', :issue_id => '234'
232 should_route :get, "/issues/234/time_entries.csv", :controller => 'timelog', :action => 'index', :issue_id => '234', :format => 'csv'
233 should_route :get, "/issues/234/time_entries.atom", :controller => 'timelog', :action => 'index', :issue_id => '234', :format => 'atom'
234 should_route :get, "/projects/ecookbook/issues/123/time_entries", :controller => 'timelog', :action => 'index', :project_id => 'ecookbook', :issue_id => '123'
235
225 236 should_route :get, "/issues/567/time_entries/new", :controller => 'timelog', :action => 'edit', :issue_id => '567'
226 237 should_route :get, "/projects/ecookbook/time_entries/new", :controller => 'timelog', :action => 'edit', :project_id => 'ecookbook'
227 238 should_route :get, "/projects/ecookbook/issues/567/time_entries/new", :controller => 'timelog', :action => 'edit', :project_id => 'ecookbook', :issue_id => '567'
228 239 should_route :get, "/time_entries/22/edit", :controller => 'timelog', :action => 'edit', :id => '22'
229 240
230 241 should_route :post, "/time_entries/55/destroy", :controller => 'timelog', :action => 'destroy', :id => '55'
231 242 end
232 243
233 244 context "time_entry_reports" do
234 245 should_route :get, "/time_entries/report", :controller => 'time_entry_reports', :action => 'report'
235 246 should_route :get, "/projects/567/time_entries/report", :controller => 'time_entry_reports', :action => 'report', :project_id => '567'
236 247 should_route :get, "/projects/567/time_entries/report.csv", :controller => 'time_entry_reports', :action => 'report', :project_id => '567', :format => 'csv'
237
238 should_route :get, "/time_entries", :controller => 'timelog', :action => 'details'
239 should_route :get, "/time_entries.csv", :controller => 'timelog', :action => 'details', :format => 'csv'
240 should_route :get, "/time_entries.atom", :controller => 'timelog', :action => 'details', :format => 'atom'
241 should_route :get, "/projects/567/time_entries", :controller => 'timelog', :action => 'details', :project_id => '567'
242 should_route :get, "/projects/567/time_entries.csv", :controller => 'timelog', :action => 'details', :project_id => '567', :format => 'csv'
243 should_route :get, "/projects/567/time_entries.atom", :controller => 'timelog', :action => 'details', :project_id => '567', :format => 'atom'
244 should_route :get, "/issues/234/time_entries", :controller => 'timelog', :action => 'details', :issue_id => '234'
245 should_route :get, "/issues/234/time_entries.csv", :controller => 'timelog', :action => 'details', :issue_id => '234', :format => 'csv'
246 should_route :get, "/issues/234/time_entries.atom", :controller => 'timelog', :action => 'details', :issue_id => '234', :format => 'atom'
247 should_route :get, "/projects/ecookbook/issues/123/time_entries", :controller => 'timelog', :action => 'details', :project_id => 'ecookbook', :issue_id => '123'
248
249 248 end
250 249
251 250 context "users" do
252 251 should_route :get, "/users", :controller => 'users', :action => 'index'
253 252 should_route :get, "/users/44", :controller => 'users', :action => 'show', :id => '44'
254 253 should_route :get, "/users/new", :controller => 'users', :action => 'new'
255 254 should_route :get, "/users/444/edit", :controller => 'users', :action => 'edit', :id => '444'
256 255 should_route :get, "/users/222/edit/membership", :controller => 'users', :action => 'edit', :id => '222', :tab => 'membership'
257 256
258 257 should_route :post, "/users", :controller => 'users', :action => 'create'
259 258 should_route :post, "/users/123/memberships", :controller => 'users', :action => 'edit_membership', :id => '123'
260 259 should_route :post, "/users/123/memberships/55", :controller => 'users', :action => 'edit_membership', :id => '123', :membership_id => '55'
261 260 should_route :post, "/users/567/memberships/12/destroy", :controller => 'users', :action => 'destroy_membership', :id => '567', :membership_id => '12'
262 261
263 262 should_route :put, "/users/444", :controller => 'users', :action => 'update', :id => '444'
264 263 end
265 264
266 265 # TODO: should they all be scoped under /projects/:project_id ?
267 266 context "versions" do
268 267 should_route :get, "/projects/foo/versions/new", :controller => 'versions', :action => 'new', :project_id => 'foo'
269 268 should_route :get, "/versions/show/1", :controller => 'versions', :action => 'show', :id => '1'
270 269 should_route :get, "/versions/edit/1", :controller => 'versions', :action => 'edit', :id => '1'
271 270
272 271 should_route :post, "/projects/foo/versions", :controller => 'versions', :action => 'create', :project_id => 'foo'
273 272 should_route :post, "/versions/update/1", :controller => 'versions', :action => 'update', :id => '1'
274 273
275 274 should_route :delete, "/versions/destroy/1", :controller => 'versions', :action => 'destroy', :id => '1'
276 275 end
277 276
278 277 context "wiki (singular, project's pages)" do
279 278 should_route :get, "/projects/567/wiki", :controller => 'wiki', :action => 'index', :id => '567'
280 279 should_route :get, "/projects/567/wiki/lalala", :controller => 'wiki', :action => 'index', :id => '567', :page => 'lalala'
281 280 should_route :get, "/projects/567/wiki/my_page/edit", :controller => 'wiki', :action => 'edit', :id => '567', :page => 'my_page'
282 281 should_route :get, "/projects/1/wiki/CookBook_documentation/history", :controller => 'wiki', :action => 'history', :id => '1', :page => 'CookBook_documentation'
283 282 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'
284 283 should_route :get, "/projects/1/wiki/CookBook_documentation/annotate/2", :controller => 'wiki', :action => 'annotate', :id => '1', :page => 'CookBook_documentation', :version => '2'
285 284 should_route :get, "/projects/22/wiki/ladida/rename", :controller => 'wiki', :action => 'rename', :id => '22', :page => 'ladida'
286 285 should_route :get, "/projects/567/wiki/page_index", :controller => 'wiki', :action => 'special', :id => '567', :page => 'page_index'
287 286 should_route :get, "/projects/567/wiki/Page_Index", :controller => 'wiki', :action => 'special', :id => '567', :page => 'Page_Index'
288 287 should_route :get, "/projects/567/wiki/date_index", :controller => 'wiki', :action => 'special', :id => '567', :page => 'date_index'
289 288 should_route :get, "/projects/567/wiki/export", :controller => 'wiki', :action => 'special', :id => '567', :page => 'export'
290 289
291 290 should_route :post, "/projects/567/wiki/my_page/edit", :controller => 'wiki', :action => 'edit', :id => '567', :page => 'my_page'
292 291 should_route :post, "/projects/567/wiki/CookBook_documentation/preview", :controller => 'wiki', :action => 'preview', :id => '567', :page => 'CookBook_documentation'
293 292 should_route :post, "/projects/22/wiki/ladida/rename", :controller => 'wiki', :action => 'rename', :id => '22', :page => 'ladida'
294 293 should_route :post, "/projects/22/wiki/ladida/destroy", :controller => 'wiki', :action => 'destroy', :id => '22', :page => 'ladida'
295 294 should_route :post, "/projects/22/wiki/ladida/protect", :controller => 'wiki', :action => 'protect', :id => '22', :page => 'ladida'
296 295 end
297 296
298 297 context "wikis (plural, admin setup)" do
299 298 should_route :get, "/projects/ladida/wiki/destroy", :controller => 'wikis', :action => 'destroy', :id => 'ladida'
300 299
301 300 should_route :post, "/projects/ladida/wiki", :controller => 'wikis', :action => 'edit', :id => 'ladida'
302 301 should_route :post, "/projects/ladida/wiki/destroy", :controller => 'wikis', :action => 'destroy', :id => 'ladida'
303 302 end
304 303
305 304 context "administration panel" do
306 305 should_route :get, "/admin/projects", :controller => 'admin', :action => 'projects'
307 306 end
308 307 end
General Comments 0
You need to be logged in to leave comments. Login now