@@ -0,0 +1,17 | |||
|
1 | ||
|
2 | ActiveRecord::Errors.default_error_messages = { | |
|
3 | :inclusion => "activerecord_error_inclusion", | |
|
4 | :exclusion => "activerecord_error_exclusion", | |
|
5 | :invalid => "activerecord_error_invalid", | |
|
6 | :confirmation => "activerecord_error_confirmation", | |
|
7 | :accepted => "activerecord_error_accepted", | |
|
8 | :empty => "activerecord_error_empty", | |
|
9 | :blank => "activerecord_error_blank", | |
|
10 | :too_long => "activerecord_error_too_long", | |
|
11 | :too_short => "activerecord_error_too_short", | |
|
12 | :wrong_length => "activerecord_error_wrong_length", | |
|
13 | :taken => "activerecord_error_taken", | |
|
14 | :not_a_number => "activerecord_error_not_a_number" | |
|
15 | } | |
|
16 | ||
|
17 | ActionView::Base.field_error_proc = Proc.new{ |html_tag, instance| "#{html_tag}" } |
@@ -0,0 +1,4 | |||
|
1 | # Add new mime types for use in respond_to blocks: | |
|
2 | ||
|
3 | Mime::SET << Mime::CSV unless Mime::SET.include?(Mime::CSV) | |
|
4 | Mime::Type.register 'application/pdf', :pdf |
@@ -0,0 +1,7 | |||
|
1 | GLoc.set_config :default_language => :en | |
|
2 | GLoc.clear_strings | |
|
3 | GLoc.set_kcode | |
|
4 | GLoc.load_localized_strings | |
|
5 | GLoc.set_config(:raise_string_not_found_errors => false) | |
|
6 | ||
|
7 | require 'redmine' |
@@ -0,0 +1,3 | |||
|
1 | #!/usr/bin/env ruby | |
|
2 | require File.dirname(__FILE__) + '/../config/boot' | |
|
3 | require 'commands/dbconsole' |
@@ -0,0 +1,3 | |||
|
1 | #!/usr/bin/env ruby | |
|
2 | require File.dirname(__FILE__) + '/../../config/boot' | |
|
3 | require 'commands/performance/request' |
@@ -0,0 +1,3 | |||
|
1 | #!/usr/bin/env ruby | |
|
2 | require File.dirname(__FILE__) + '/../../config/boot' | |
|
3 | require 'commands/process/inspector' |
@@ -1,434 +1,434 | |||
|
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 ProjectsController < ApplicationController |
|
19 | 19 | layout 'base' |
|
20 | 20 | menu_item :overview |
|
21 | 21 | menu_item :activity, :only => :activity |
|
22 | 22 | menu_item :roadmap, :only => :roadmap |
|
23 | 23 | menu_item :files, :only => [:list_files, :add_file] |
|
24 | 24 | menu_item :settings, :only => :settings |
|
25 | 25 | menu_item :issues, :only => [:changelog] |
|
26 | 26 | |
|
27 | 27 | before_filter :find_project, :except => [ :index, :list, :add, :activity ] |
|
28 | 28 | before_filter :find_optional_project, :only => :activity |
|
29 | 29 | before_filter :authorize, :except => [ :index, :list, :add, :archive, :unarchive, :destroy, :activity ] |
|
30 | 30 | before_filter :require_admin, :only => [ :add, :archive, :unarchive, :destroy ] |
|
31 | 31 | accept_key_auth :activity, :calendar |
|
32 | 32 | |
|
33 | 33 | helper :sort |
|
34 | 34 | include SortHelper |
|
35 | 35 | helper :custom_fields |
|
36 | 36 | include CustomFieldsHelper |
|
37 | 37 | helper :ifpdf |
|
38 | 38 | include IfpdfHelper |
|
39 | 39 | helper :issues |
|
40 | 40 | helper IssuesHelper |
|
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 | projects = Project.find :all, |
|
50 | 50 | :conditions => Project.visible_by(User.current), |
|
51 | 51 | :include => :parent |
|
52 | 52 | respond_to do |format| |
|
53 | 53 | format.html { |
|
54 | 54 | @project_tree = projects.group_by {|p| p.parent || p} |
|
55 |
@project_tree.each |
|
|
55 | @project_tree.keys.each {|p| @project_tree[p] -= [p]} | |
|
56 | 56 | } |
|
57 | 57 | format.atom { |
|
58 | 58 | render_feed(projects.sort_by(&:created_on).reverse.slice(0, Setting.feeds_limit.to_i), |
|
59 | 59 | :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 | @root_projects = Project.find(:all, |
|
69 | 69 | :conditions => "parent_id IS NULL AND status = #{Project::STATUS_ACTIVE}", |
|
70 | 70 | :order => 'name') |
|
71 | 71 | @project = Project.new(params[:project]) |
|
72 | 72 | if request.get? |
|
73 | 73 | @project.trackers = Tracker.all |
|
74 | 74 | @project.is_public = Setting.default_projects_public? |
|
75 | 75 | @project.enabled_module_names = Redmine::AccessControl.available_project_modules |
|
76 | 76 | else |
|
77 | 77 | @project.enabled_module_names = params[:enabled_modules] |
|
78 | 78 | if @project.save |
|
79 | 79 | flash[:notice] = l(:notice_successful_create) |
|
80 | 80 | redirect_to :controller => 'admin', :action => 'projects' |
|
81 | 81 | end |
|
82 | 82 | end |
|
83 | 83 | end |
|
84 | 84 | |
|
85 | 85 | # Show @project |
|
86 | 86 | def show |
|
87 | 87 | @members_by_role = @project.members.find(:all, :include => [:user, :role], :order => 'position').group_by {|m| m.role} |
|
88 | 88 | @subprojects = @project.children.find(:all, :conditions => Project.visible_by(User.current)) |
|
89 | 89 | @news = @project.news.find(:all, :limit => 5, :include => [ :author, :project ], :order => "#{News.table_name}.created_on DESC") |
|
90 | 90 | @trackers = @project.rolled_up_trackers |
|
91 | 91 | |
|
92 | 92 | cond = @project.project_condition(Setting.display_subprojects_issues?) |
|
93 | 93 | Issue.visible_by(User.current) do |
|
94 | 94 | @open_issues_by_tracker = Issue.count(:group => :tracker, |
|
95 | 95 | :include => [:project, :status, :tracker], |
|
96 | 96 | :conditions => ["(#{cond}) AND #{IssueStatus.table_name}.is_closed=?", false]) |
|
97 | 97 | @total_issues_by_tracker = Issue.count(:group => :tracker, |
|
98 | 98 | :include => [:project, :status, :tracker], |
|
99 | 99 | :conditions => cond) |
|
100 | 100 | end |
|
101 | 101 | TimeEntry.visible_by(User.current) do |
|
102 | 102 | @total_hours = TimeEntry.sum(:hours, |
|
103 | 103 | :include => :project, |
|
104 | 104 | :conditions => cond).to_f |
|
105 | 105 | end |
|
106 | 106 | @key = User.current.rss_key |
|
107 | 107 | end |
|
108 | 108 | |
|
109 | 109 | def settings |
|
110 | 110 | @root_projects = Project.find(:all, |
|
111 | 111 | :conditions => ["parent_id IS NULL AND status = #{Project::STATUS_ACTIVE} AND id <> ?", @project.id], |
|
112 | 112 | :order => 'name') |
|
113 | 113 | @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position") |
|
114 | 114 | @issue_category ||= IssueCategory.new |
|
115 | 115 | @member ||= @project.members.new |
|
116 | 116 | @trackers = Tracker.all |
|
117 | 117 | @repository ||= @project.repository |
|
118 | 118 | @wiki ||= @project.wiki |
|
119 | 119 | end |
|
120 | 120 | |
|
121 | 121 | # Edit @project |
|
122 | 122 | def edit |
|
123 | 123 | if request.post? |
|
124 | 124 | @project.attributes = params[:project] |
|
125 | 125 | if @project.save |
|
126 | 126 | flash[:notice] = l(:notice_successful_update) |
|
127 | 127 | redirect_to :action => 'settings', :id => @project |
|
128 | 128 | else |
|
129 | 129 | settings |
|
130 | 130 | render :action => 'settings' |
|
131 | 131 | end |
|
132 | 132 | end |
|
133 | 133 | end |
|
134 | 134 | |
|
135 | 135 | def modules |
|
136 | 136 | @project.enabled_module_names = params[:enabled_modules] |
|
137 | 137 | redirect_to :action => 'settings', :id => @project, :tab => 'modules' |
|
138 | 138 | end |
|
139 | 139 | |
|
140 | 140 | def archive |
|
141 | 141 | @project.archive if request.post? && @project.active? |
|
142 | 142 | redirect_to :controller => 'admin', :action => 'projects' |
|
143 | 143 | end |
|
144 | 144 | |
|
145 | 145 | def unarchive |
|
146 | 146 | @project.unarchive if request.post? && !@project.active? |
|
147 | 147 | redirect_to :controller => 'admin', :action => 'projects' |
|
148 | 148 | end |
|
149 | 149 | |
|
150 | 150 | # Delete @project |
|
151 | 151 | def destroy |
|
152 | 152 | @project_to_destroy = @project |
|
153 | 153 | if request.post? and params[:confirm] |
|
154 | 154 | @project_to_destroy.destroy |
|
155 | 155 | redirect_to :controller => 'admin', :action => 'projects' |
|
156 | 156 | end |
|
157 | 157 | # hide project in layout |
|
158 | 158 | @project = nil |
|
159 | 159 | end |
|
160 | 160 | |
|
161 | 161 | # Add a new issue category to @project |
|
162 | 162 | def add_issue_category |
|
163 | 163 | @category = @project.issue_categories.build(params[:category]) |
|
164 | 164 | if request.post? and @category.save |
|
165 | 165 | respond_to do |format| |
|
166 | 166 | format.html do |
|
167 | 167 | flash[:notice] = l(:notice_successful_create) |
|
168 | 168 | redirect_to :action => 'settings', :tab => 'categories', :id => @project |
|
169 | 169 | end |
|
170 | 170 | format.js do |
|
171 | 171 | # IE doesn't support the replace_html rjs method for select box options |
|
172 | 172 | render(:update) {|page| page.replace "issue_category_id", |
|
173 | 173 | content_tag('select', '<option></option>' + options_from_collection_for_select(@project.issue_categories, 'id', 'name', @category.id), :id => 'issue_category_id', :name => 'issue[category_id]') |
|
174 | 174 | } |
|
175 | 175 | end |
|
176 | 176 | end |
|
177 | 177 | end |
|
178 | 178 | end |
|
179 | 179 | |
|
180 | 180 | # Add a new version to @project |
|
181 | 181 | def add_version |
|
182 | 182 | @version = @project.versions.build(params[:version]) |
|
183 | 183 | if request.post? and @version.save |
|
184 | 184 | flash[:notice] = l(:notice_successful_create) |
|
185 | 185 | redirect_to :action => 'settings', :tab => 'versions', :id => @project |
|
186 | 186 | end |
|
187 | 187 | end |
|
188 | 188 | |
|
189 | 189 | def add_file |
|
190 | 190 | if request.post? |
|
191 | 191 | @version = @project.versions.find_by_id(params[:version_id]) |
|
192 | 192 | attachments = attach_files(@version, params[:attachments]) |
|
193 | 193 | Mailer.deliver_attachments_added(attachments) if !attachments.empty? && Setting.notified_events.include?('file_added') |
|
194 | 194 | redirect_to :controller => 'projects', :action => 'list_files', :id => @project |
|
195 | 195 | end |
|
196 | 196 | @versions = @project.versions.sort |
|
197 | 197 | end |
|
198 | 198 | |
|
199 | 199 | def list_files |
|
200 | 200 | sort_init "#{Attachment.table_name}.filename", "asc" |
|
201 | 201 | sort_update |
|
202 | 202 | @versions = @project.versions.find(:all, :include => :attachments, :order => sort_clause).sort.reverse |
|
203 | 203 | render :layout => !request.xhr? |
|
204 | 204 | end |
|
205 | 205 | |
|
206 | 206 | # Show changelog for @project |
|
207 | 207 | def changelog |
|
208 | 208 | @trackers = @project.trackers.find(:all, :conditions => ["is_in_chlog=?", true], :order => 'position') |
|
209 | 209 | retrieve_selected_tracker_ids(@trackers) |
|
210 | 210 | @versions = @project.versions.sort |
|
211 | 211 | end |
|
212 | 212 | |
|
213 | 213 | def roadmap |
|
214 | 214 | @trackers = @project.trackers.find(:all, :conditions => ["is_in_roadmap=?", true]) |
|
215 | 215 | retrieve_selected_tracker_ids(@trackers) |
|
216 | 216 | @versions = @project.versions.sort |
|
217 | 217 | @versions = @versions.select {|v| !v.completed? } unless params[:completed] |
|
218 | 218 | end |
|
219 | 219 | |
|
220 | 220 | def activity |
|
221 | 221 | @days = Setting.activity_days_default.to_i |
|
222 | 222 | |
|
223 | 223 | if params[:from] |
|
224 | 224 | begin; @date_to = params[:from].to_date; rescue; end |
|
225 | 225 | end |
|
226 | 226 | |
|
227 | 227 | @date_to ||= Date.today + 1 |
|
228 | 228 | @date_from = @date_to - @days |
|
229 | 229 | |
|
230 | 230 | @event_types = %w(issues news files documents changesets wiki_pages messages) |
|
231 | 231 | if @project |
|
232 | 232 | @event_types.delete('wiki_pages') unless @project.wiki |
|
233 | 233 | @event_types.delete('changesets') unless @project.repository |
|
234 | 234 | @event_types.delete('messages') unless @project.boards.any? |
|
235 | 235 | # only show what the user is allowed to view |
|
236 | 236 | @event_types = @event_types.select {|o| User.current.allowed_to?("view_#{o}".to_sym, @project)} |
|
237 | 237 | @with_subprojects = params[:with_subprojects].nil? ? Setting.display_subprojects_issues? : (params[:with_subprojects] == '1') |
|
238 | 238 | end |
|
239 | 239 | @scope = @event_types.select {|t| params["show_#{t}"]} |
|
240 | 240 | # default events if none is specified in parameters |
|
241 | 241 | @scope = (@event_types - %w(wiki_pages messages))if @scope.empty? |
|
242 | 242 | |
|
243 | 243 | @events = [] |
|
244 | 244 | |
|
245 | 245 | if @scope.include?('issues') |
|
246 | 246 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_issues, :project => @project, :with_subprojects => @with_subprojects)) |
|
247 | 247 | cond.add(["#{Issue.table_name}.created_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
248 | 248 | @events += Issue.find(:all, :include => [:project, :author, :tracker], :conditions => cond.conditions) |
|
249 | 249 | |
|
250 | 250 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_issues, :project => @project, :with_subprojects => @with_subprojects)) |
|
251 | 251 | cond.add(["#{Journal.table_name}.journalized_type = 'Issue' AND #{Journal.table_name}.created_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
252 | 252 | cond.add("#{JournalDetail.table_name}.prop_key = 'status_id' OR #{Journal.table_name}.notes <> ''") |
|
253 | 253 | @events += Journal.find(:all, :include => [{:issue => :project}, :details, :user], :conditions => cond.conditions) |
|
254 | 254 | end |
|
255 | 255 | |
|
256 | 256 | if @scope.include?('news') |
|
257 | 257 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_news, :project => @project, :with_subprojects => @with_subprojects)) |
|
258 | 258 | cond.add(["#{News.table_name}.created_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
259 | 259 | @events += News.find(:all, :include => [:project, :author], :conditions => cond.conditions) |
|
260 | 260 | end |
|
261 | 261 | |
|
262 | 262 | if @scope.include?('files') |
|
263 | 263 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_files, :project => @project, :with_subprojects => @with_subprojects)) |
|
264 | 264 | cond.add(["#{Attachment.table_name}.created_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
265 | 265 | @events += Attachment.find(:all, :select => "#{Attachment.table_name}.*", |
|
266 | 266 | :joins => "LEFT JOIN #{Version.table_name} ON #{Attachment.table_name}.container_type='Version' AND #{Version.table_name}.id = #{Attachment.table_name}.container_id " + |
|
267 | 267 | "LEFT JOIN #{Project.table_name} ON #{Version.table_name}.project_id = #{Project.table_name}.id", |
|
268 | 268 | :conditions => cond.conditions) |
|
269 | 269 | end |
|
270 | 270 | |
|
271 | 271 | if @scope.include?('documents') |
|
272 | 272 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_documents, :project => @project, :with_subprojects => @with_subprojects)) |
|
273 | 273 | cond.add(["#{Document.table_name}.created_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
274 | 274 | @events += Document.find(:all, :include => :project, :conditions => cond.conditions) |
|
275 | 275 | |
|
276 | 276 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_documents, :project => @project, :with_subprojects => @with_subprojects)) |
|
277 | 277 | cond.add(["#{Attachment.table_name}.created_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
278 | 278 | @events += Attachment.find(:all, :select => "#{Attachment.table_name}.*", |
|
279 | 279 | :joins => "LEFT JOIN #{Document.table_name} ON #{Attachment.table_name}.container_type='Document' AND #{Document.table_name}.id = #{Attachment.table_name}.container_id " + |
|
280 | 280 | "LEFT JOIN #{Project.table_name} ON #{Document.table_name}.project_id = #{Project.table_name}.id", |
|
281 | 281 | :conditions => cond.conditions) |
|
282 | 282 | end |
|
283 | 283 | |
|
284 | 284 | if @scope.include?('wiki_pages') |
|
285 | 285 | select = "#{WikiContent.versioned_table_name}.updated_on, #{WikiContent.versioned_table_name}.comments, " + |
|
286 | 286 | "#{WikiContent.versioned_table_name}.#{WikiContent.version_column}, #{WikiPage.table_name}.title, " + |
|
287 | 287 | "#{WikiContent.versioned_table_name}.page_id, #{WikiContent.versioned_table_name}.author_id, " + |
|
288 | 288 | "#{WikiContent.versioned_table_name}.id" |
|
289 | 289 | joins = "LEFT JOIN #{WikiPage.table_name} ON #{WikiPage.table_name}.id = #{WikiContent.versioned_table_name}.page_id " + |
|
290 | 290 | "LEFT JOIN #{Wiki.table_name} ON #{Wiki.table_name}.id = #{WikiPage.table_name}.wiki_id " + |
|
291 | 291 | "LEFT JOIN #{Project.table_name} ON #{Project.table_name}.id = #{Wiki.table_name}.project_id" |
|
292 | 292 | |
|
293 | 293 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_wiki_pages, :project => @project, :with_subprojects => @with_subprojects)) |
|
294 | 294 | cond.add(["#{WikiContent.versioned_table_name}.updated_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
295 | 295 | @events += WikiContent.versioned_class.find(:all, :select => select, :joins => joins, :conditions => cond.conditions) |
|
296 | 296 | end |
|
297 | 297 | |
|
298 | 298 | if @scope.include?('changesets') |
|
299 | 299 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_changesets, :project => @project, :with_subprojects => @with_subprojects)) |
|
300 | 300 | cond.add(["#{Changeset.table_name}.committed_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
301 | 301 | @events += Changeset.find(:all, :include => {:repository => :project}, :conditions => cond.conditions) |
|
302 | 302 | end |
|
303 | 303 | |
|
304 | 304 | if @scope.include?('messages') |
|
305 | 305 | cond = ARCondition.new(Project.allowed_to_condition(User.current, :view_messages, :project => @project, :with_subprojects => @with_subprojects)) |
|
306 | 306 | cond.add(["#{Message.table_name}.created_on BETWEEN ? AND ?", @date_from, @date_to]) |
|
307 | 307 | @events += Message.find(:all, :include => [{:board => :project}, :author], :conditions => cond.conditions) |
|
308 | 308 | end |
|
309 | 309 | |
|
310 | 310 | @events_by_day = @events.group_by(&:event_date) |
|
311 | 311 | |
|
312 | 312 | respond_to do |format| |
|
313 | 313 | format.html { render :layout => false if request.xhr? } |
|
314 | 314 | format.atom { render_feed(@events, :title => "#{@project || Setting.app_title}: #{l(:label_activity)}") } |
|
315 | 315 | end |
|
316 | 316 | end |
|
317 | 317 | |
|
318 | 318 | def calendar |
|
319 | 319 | @trackers = @project.rolled_up_trackers |
|
320 | 320 | retrieve_selected_tracker_ids(@trackers) |
|
321 | 321 | |
|
322 | 322 | if params[:year] and params[:year].to_i > 1900 |
|
323 | 323 | @year = params[:year].to_i |
|
324 | 324 | if params[:month] and params[:month].to_i > 0 and params[:month].to_i < 13 |
|
325 | 325 | @month = params[:month].to_i |
|
326 | 326 | end |
|
327 | 327 | end |
|
328 | 328 | @year ||= Date.today.year |
|
329 | 329 | @month ||= Date.today.month |
|
330 | 330 | @calendar = Redmine::Helpers::Calendar.new(Date.civil(@year, @month, 1), current_language, :month) |
|
331 | 331 | @with_subprojects = params[:with_subprojects].nil? ? Setting.display_subprojects_issues? : (params[:with_subprojects] == '1') |
|
332 | 332 | events = [] |
|
333 | 333 | @project.issues_with_subprojects(@with_subprojects) do |
|
334 | 334 | events += Issue.find(:all, |
|
335 | 335 | :include => [:tracker, :status, :assigned_to, :priority, :project], |
|
336 | 336 | :conditions => ["((start_date BETWEEN ? AND ?) OR (due_date BETWEEN ? AND ?)) AND #{Issue.table_name}.tracker_id IN (#{@selected_tracker_ids.join(',')})", @calendar.startdt, @calendar.enddt, @calendar.startdt, @calendar.enddt] |
|
337 | 337 | ) unless @selected_tracker_ids.empty? |
|
338 | 338 | events += Version.find(:all, :include => :project, |
|
339 | 339 | :conditions => ["effective_date BETWEEN ? AND ?", @calendar.startdt, @calendar.enddt]) |
|
340 | 340 | end |
|
341 | 341 | @calendar.events = events |
|
342 | 342 | |
|
343 | 343 | render :layout => false if request.xhr? |
|
344 | 344 | end |
|
345 | 345 | |
|
346 | 346 | def gantt |
|
347 | 347 | @trackers = @project.rolled_up_trackers |
|
348 | 348 | retrieve_selected_tracker_ids(@trackers) |
|
349 | 349 | |
|
350 | 350 | if params[:year] and params[:year].to_i >0 |
|
351 | 351 | @year_from = params[:year].to_i |
|
352 | 352 | if params[:month] and params[:month].to_i >=1 and params[:month].to_i <= 12 |
|
353 | 353 | @month_from = params[:month].to_i |
|
354 | 354 | else |
|
355 | 355 | @month_from = 1 |
|
356 | 356 | end |
|
357 | 357 | else |
|
358 | 358 | @month_from ||= Date.today.month |
|
359 | 359 | @year_from ||= Date.today.year |
|
360 | 360 | end |
|
361 | 361 | |
|
362 | 362 | zoom = (params[:zoom] || User.current.pref[:gantt_zoom]).to_i |
|
363 | 363 | @zoom = (zoom > 0 && zoom < 5) ? zoom : 2 |
|
364 | 364 | months = (params[:months] || User.current.pref[:gantt_months]).to_i |
|
365 | 365 | @months = (months > 0 && months < 25) ? months : 6 |
|
366 | 366 | |
|
367 | 367 | # Save gantt paramters as user preference (zoom and months count) |
|
368 | 368 | if (User.current.logged? && (@zoom != User.current.pref[:gantt_zoom] || @months != User.current.pref[:gantt_months])) |
|
369 | 369 | User.current.pref[:gantt_zoom], User.current.pref[:gantt_months] = @zoom, @months |
|
370 | 370 | User.current.preference.save |
|
371 | 371 | end |
|
372 | 372 | |
|
373 | 373 | @date_from = Date.civil(@year_from, @month_from, 1) |
|
374 | 374 | @date_to = (@date_from >> @months) - 1 |
|
375 | 375 | @with_subprojects = params[:with_subprojects].nil? ? Setting.display_subprojects_issues? : (params[:with_subprojects] == '1') |
|
376 | 376 | |
|
377 | 377 | @events = [] |
|
378 | 378 | @project.issues_with_subprojects(@with_subprojects) do |
|
379 | 379 | # Issues that have start and due dates |
|
380 | 380 | @events += Issue.find(:all, |
|
381 | 381 | :order => "start_date, due_date", |
|
382 | 382 | :include => [:tracker, :status, :assigned_to, :priority, :project], |
|
383 | 383 | :conditions => ["(((start_date>=? and start_date<=?) or (due_date>=? and due_date<=?) or (start_date<? and due_date>?)) and start_date is not null and due_date is not null and #{Issue.table_name}.tracker_id in (#{@selected_tracker_ids.join(',')}))", @date_from, @date_to, @date_from, @date_to, @date_from, @date_to] |
|
384 | 384 | ) unless @selected_tracker_ids.empty? |
|
385 | 385 | # Issues that don't have a due date but that are assigned to a version with a date |
|
386 | 386 | @events += Issue.find(:all, |
|
387 | 387 | :order => "start_date, effective_date", |
|
388 | 388 | :include => [:tracker, :status, :assigned_to, :priority, :project, :fixed_version], |
|
389 | 389 | :conditions => ["(((start_date>=? and start_date<=?) or (effective_date>=? and effective_date<=?) or (start_date<? and effective_date>?)) and start_date is not null and due_date is null and effective_date is not null and #{Issue.table_name}.tracker_id in (#{@selected_tracker_ids.join(',')}))", @date_from, @date_to, @date_from, @date_to, @date_from, @date_to] |
|
390 | 390 | ) unless @selected_tracker_ids.empty? |
|
391 | 391 | @events += Version.find(:all, :include => :project, |
|
392 | 392 | :conditions => ["effective_date BETWEEN ? AND ?", @date_from, @date_to]) |
|
393 | 393 | end |
|
394 | 394 | @events.sort! {|x,y| x.start_date <=> y.start_date } |
|
395 | 395 | |
|
396 | 396 | if params[:format]=='pdf' |
|
397 | 397 | @options_for_rfpdf ||= {} |
|
398 | 398 | @options_for_rfpdf[:file_name] = "#{@project.identifier}-gantt.pdf" |
|
399 | 399 | render :template => "projects/gantt.rfpdf", :layout => false |
|
400 | 400 | elsif params[:format]=='png' && respond_to?('gantt_image') |
|
401 | 401 | image = gantt_image(@events, @date_from, @months, @zoom) |
|
402 | 402 | image.format = 'PNG' |
|
403 | 403 | send_data(image.to_blob, :disposition => 'inline', :type => 'image/png', :filename => "#{@project.identifier}-gantt.png") |
|
404 | 404 | else |
|
405 | 405 | render :template => "projects/gantt.rhtml" |
|
406 | 406 | end |
|
407 | 407 | end |
|
408 | 408 | |
|
409 | 409 | private |
|
410 | 410 | # Find project of id params[:id] |
|
411 | 411 | # if not found, redirect to project list |
|
412 | 412 | # Used as a before_filter |
|
413 | 413 | def find_project |
|
414 | 414 | @project = Project.find(params[:id]) |
|
415 | 415 | rescue ActiveRecord::RecordNotFound |
|
416 | 416 | render_404 |
|
417 | 417 | end |
|
418 | 418 | |
|
419 | 419 | def find_optional_project |
|
420 | 420 | return true unless params[:id] |
|
421 | 421 | @project = Project.find(params[:id]) |
|
422 | 422 | authorize |
|
423 | 423 | rescue ActiveRecord::RecordNotFound |
|
424 | 424 | render_404 |
|
425 | 425 | end |
|
426 | 426 | |
|
427 | 427 | def retrieve_selected_tracker_ids(selectable_trackers) |
|
428 | 428 | if ids = params[:tracker_ids] |
|
429 | 429 | @selected_tracker_ids = (ids.is_a? Array) ? ids.collect { |id| id.to_i.to_s } : ids.split('/').collect { |id| id.to_i.to_s } |
|
430 | 430 | else |
|
431 | 431 | @selected_tracker_ids = selectable_trackers.collect {|t| t.id.to_s } |
|
432 | 432 | end |
|
433 | 433 | end |
|
434 | 434 | end |
@@ -1,98 +1,102 | |||
|
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 | require 'iconv' |
|
19 | 19 | |
|
20 | 20 | module RepositoriesHelper |
|
21 | 21 | def format_revision(txt) |
|
22 | 22 | txt.to_s[0,8] |
|
23 | 23 | end |
|
24 | 24 | |
|
25 | def to_path_param(path) | |
|
26 | path.to_s.split(%r{[/\\]}).select {|p| !p.blank?} | |
|
27 | end | |
|
28 | ||
|
25 | 29 | def to_utf8(str) |
|
26 | 30 | return str if /\A[\r\n\t\x20-\x7e]*\Z/n.match(str) # for us-ascii |
|
27 | 31 | @encodings ||= Setting.repositories_encodings.split(',').collect(&:strip) |
|
28 | 32 | @encodings.each do |encoding| |
|
29 | 33 | begin |
|
30 | 34 | return Iconv.conv('UTF-8', encoding, str) |
|
31 | 35 | rescue Iconv::Failure |
|
32 | 36 | # do nothing here and try the next encoding |
|
33 | 37 | end |
|
34 | 38 | end |
|
35 | 39 | str |
|
36 | 40 | end |
|
37 | 41 | |
|
38 | 42 | def repository_field_tags(form, repository) |
|
39 | 43 | method = repository.class.name.demodulize.underscore + "_field_tags" |
|
40 | 44 | send(method, form, repository) if repository.is_a?(Repository) && respond_to?(method) |
|
41 | 45 | end |
|
42 | 46 | |
|
43 | 47 | def scm_select_tag(repository) |
|
44 | 48 | scm_options = [["--- #{l(:actionview_instancetag_blank_option)} ---", '']] |
|
45 | 49 | REDMINE_SUPPORTED_SCM.each do |scm| |
|
46 | 50 | scm_options << ["Repository::#{scm}".constantize.scm_name, scm] if Setting.enabled_scm.include?(scm) || (repository && repository.class.name.demodulize == scm) |
|
47 | 51 | end |
|
48 | 52 | |
|
49 | 53 | select_tag('repository_scm', |
|
50 | 54 | options_for_select(scm_options, repository.class.name.demodulize), |
|
51 | 55 | :disabled => (repository && !repository.new_record?), |
|
52 | 56 | :onchange => remote_function(:url => { :controller => 'repositories', :action => 'edit', :id => @project }, :method => :get, :with => "Form.serialize(this.form)") |
|
53 | 57 | ) |
|
54 | 58 | end |
|
55 | 59 | |
|
56 | 60 | def with_leading_slash(path) |
|
57 | 61 | path.to_s.starts_with?('/') ? path : "/#{path}" |
|
58 | 62 | end |
|
59 | 63 | |
|
60 | 64 | def without_leading_slash(path) |
|
61 | 65 | path.gsub(%r{^/+}, '') |
|
62 | 66 | end |
|
63 | 67 | |
|
64 | 68 | def subversion_field_tags(form, repository) |
|
65 | 69 | content_tag('p', form.text_field(:url, :size => 60, :required => true, :disabled => (repository && !repository.root_url.blank?)) + |
|
66 | 70 | '<br />(http://, https://, svn://, file:///)') + |
|
67 | 71 | content_tag('p', form.text_field(:login, :size => 30)) + |
|
68 | 72 | content_tag('p', form.password_field(:password, :size => 30, :name => 'ignore', |
|
69 | 73 | :value => ((repository.new_record? || repository.password.blank?) ? '' : ('x'*15)), |
|
70 | 74 | :onfocus => "this.value=''; this.name='repository[password]';", |
|
71 | 75 | :onchange => "this.name='repository[password]';")) |
|
72 | 76 | end |
|
73 | 77 | |
|
74 | 78 | def darcs_field_tags(form, repository) |
|
75 | 79 | content_tag('p', form.text_field(:url, :label => 'Root directory', :size => 60, :required => true, :disabled => (repository && !repository.new_record?))) |
|
76 | 80 | end |
|
77 | 81 | |
|
78 | 82 | def mercurial_field_tags(form, repository) |
|
79 | 83 | content_tag('p', form.text_field(:url, :label => 'Root directory', :size => 60, :required => true, :disabled => (repository && !repository.root_url.blank?))) |
|
80 | 84 | end |
|
81 | 85 | |
|
82 | 86 | def git_field_tags(form, repository) |
|
83 | 87 | content_tag('p', form.text_field(:url, :label => 'Path to .git directory', :size => 60, :required => true, :disabled => (repository && !repository.root_url.blank?))) |
|
84 | 88 | end |
|
85 | 89 | |
|
86 | 90 | def cvs_field_tags(form, repository) |
|
87 | 91 | content_tag('p', form.text_field(:root_url, :label => 'CVSROOT', :size => 60, :required => true, :disabled => !repository.new_record?)) + |
|
88 | 92 | content_tag('p', form.text_field(:url, :label => 'Module', :size => 30, :required => true, :disabled => !repository.new_record?)) |
|
89 | 93 | end |
|
90 | 94 | |
|
91 | 95 | def bazaar_field_tags(form, repository) |
|
92 | 96 | content_tag('p', form.text_field(:url, :label => 'Root directory', :size => 60, :required => true, :disabled => (repository && !repository.new_record?))) |
|
93 | 97 | end |
|
94 | 98 | |
|
95 | 99 | def filesystem_field_tags(form, repository) |
|
96 | 100 | content_tag('p', form.text_field(:url, :label => 'Root directory', :size => 60, :required => true, :disabled => (repository && !repository.root_url.blank?))) |
|
97 | 101 | end |
|
98 | 102 | end |
@@ -1,52 +1,54 | |||
|
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 UserPreference < ActiveRecord::Base |
|
19 | 19 | belongs_to :user |
|
20 | 20 | serialize :others |
|
21 | 21 | |
|
22 | 22 | attr_protected :others |
|
23 | 23 | |
|
24 | 24 | def initialize(attributes = nil) |
|
25 | 25 | super |
|
26 | 26 | self.others ||= {} |
|
27 | 27 | end |
|
28 | 28 | |
|
29 | 29 | def before_save |
|
30 | 30 | self.others ||= {} |
|
31 | 31 | end |
|
32 | 32 | |
|
33 | 33 | def [](attr_name) |
|
34 | 34 | if attribute_present? attr_name |
|
35 | 35 | super |
|
36 | 36 | else |
|
37 | 37 | others ? others[attr_name] : nil |
|
38 | 38 | end |
|
39 | 39 | end |
|
40 | 40 | |
|
41 | 41 | def []=(attr_name, value) |
|
42 | 42 | if attribute_present? attr_name |
|
43 | 43 | super |
|
44 | 44 | else |
|
45 | self.others ||= {} | |
|
46 | self.others.store attr_name, value | |
|
45 | h = read_attribute(:others).dup || {} | |
|
46 | h.update(attr_name => value) | |
|
47 | write_attribute(:others, h) | |
|
48 | value | |
|
47 | 49 | end |
|
48 | 50 | end |
|
49 | 51 | |
|
50 | 52 | def comments_sorting; self[:comments_sorting] end |
|
51 | 53 | def comments_sorting=(order); self[:comments_sorting]=order end |
|
52 | 54 | end |
@@ -1,24 +1,24 | |||
|
1 | 1 | <% @entries.each do |entry| %> |
|
2 | 2 | <% tr_id = Digest::MD5.hexdigest(entry.path) |
|
3 | 3 | depth = params[:depth].to_i %> |
|
4 | 4 | <tr id="<%= tr_id %>" class="<%= params[:parent_id] %> entry <%= entry.kind %>"> |
|
5 | 5 | <td style="padding-left: <%=18 * depth%>px;" class="filename"> |
|
6 | 6 | <% if entry.is_dir? %> |
|
7 | <span class="expander" onclick="<%= remote_function :url => {:action => 'browse', :id => @project, :path => entry.path, :rev => @rev, :depth => (depth + 1), :parent_id => tr_id}, | |
|
7 | <span class="expander" onclick="<%= remote_function :url => {:action => 'browse', :id => @project, :path => to_path_param(entry.path), :rev => @rev, :depth => (depth + 1), :parent_id => tr_id}, | |
|
8 | 8 | :update => { :success => tr_id }, |
|
9 | 9 | :position => :after, |
|
10 | 10 | :success => "scmEntryLoaded('#{tr_id}')", |
|
11 | 11 | :condition => "scmEntryClick('#{tr_id}')"%>"> </span> |
|
12 | 12 | <% end %> |
|
13 | 13 | <%= link_to h(entry.name), |
|
14 | {:action => (entry.is_dir? ? 'browse' : 'changes'), :id => @project, :path => entry.path, :rev => @rev}, | |
|
14 | {:action => (entry.is_dir? ? 'browse' : 'changes'), :id => @project, :path => to_path_param(entry.path), :rev => @rev}, | |
|
15 | 15 | :class => (entry.is_dir? ? 'icon icon-folder' : 'icon icon-file')%> |
|
16 | 16 | </td> |
|
17 | 17 | <td class="size"><%= (entry.size ? number_to_human_size(entry.size) : "?") unless entry.is_dir? %></td> |
|
18 | 18 | <td class="revision"><%= link_to(format_revision(entry.lastrev.name), :action => 'revision', :id => @project, :rev => entry.lastrev.identifier) if entry.lastrev && entry.lastrev.identifier %></td> |
|
19 | 19 | <td class="age"><%= distance_of_time_in_words(entry.lastrev.time, Time.now) if entry.lastrev && entry.lastrev.time %></td> |
|
20 | 20 | <td class="author"><%=h(entry.lastrev.author.to_s.split('<').first) if entry.lastrev %></td> |
|
21 | 21 | <% changeset = @project.repository.changesets.find_by_revision(entry.lastrev.identifier) if entry.lastrev && entry.lastrev.identifier %> |
|
22 | 22 | <td class="comments"><%=h truncate(changeset.comments, 50) unless changeset.nil? %></td> |
|
23 | 23 | </tr> |
|
24 | 24 | <% end %> |
@@ -1,21 +1,21 | |||
|
1 | 1 | <%= link_to 'root', :action => 'browse', :id => @project, :path => '', :rev => @rev %> |
|
2 | 2 | <% |
|
3 | 3 | dirs = path.split('/') |
|
4 | 4 | if 'file' == kind |
|
5 | 5 | filename = dirs.pop |
|
6 | 6 | end |
|
7 | 7 | link_path = '' |
|
8 | 8 | dirs.each do |dir| |
|
9 | 9 | next if dir.blank? |
|
10 | 10 | link_path << '/' unless link_path.empty? |
|
11 | 11 | link_path << "#{dir}" |
|
12 | 12 | %> |
|
13 | / <%= link_to h(dir), :action => 'browse', :id => @project, :path => link_path, :rev => @rev %> | |
|
13 | / <%= link_to h(dir), :action => 'browse', :id => @project, :path => to_path_param(link_path), :rev => @rev %> | |
|
14 | 14 | <% end %> |
|
15 | 15 | <% if filename %> |
|
16 | / <%= link_to h(filename), :action => 'changes', :id => @project, :path => "#{link_path}/#{filename}", :rev => @rev %> | |
|
16 | / <%= link_to h(filename), :action => 'changes', :id => @project, :path => to_path_param("#{link_path}/#{filename}"), :rev => @rev %> | |
|
17 | 17 | <% end %> |
|
18 | 18 | |
|
19 | 19 | <%= "@ #{revision}" if revision %> |
|
20 | 20 | |
|
21 | 21 | <% html_title(with_leading_slash(path)) -%> |
@@ -1,28 +1,28 | |||
|
1 | <% form_tag({:controller => 'repositories', :action => 'diff', :id => @project, :path => path}, :method => :get) do %> | |
|
1 | <% form_tag({:controller => 'repositories', :action => 'diff', :id => @project, :path => to_path_param(path)}, :method => :get) do %> | |
|
2 | 2 | <table class="list changesets"> |
|
3 | 3 | <thead><tr> |
|
4 | 4 | <th>#</th> |
|
5 | 5 | <th></th> |
|
6 | 6 | <th></th> |
|
7 | 7 | <th><%= l(:label_date) %></th> |
|
8 | 8 | <th><%= l(:field_author) %></th> |
|
9 | 9 | <th><%= l(:field_comments) %></th> |
|
10 | 10 | </tr></thead> |
|
11 | 11 | <tbody> |
|
12 | 12 | <% show_diff = entry && entry.is_file? && revisions.size > 1 %> |
|
13 | 13 | <% line_num = 1 %> |
|
14 | 14 | <% revisions.each do |changeset| %> |
|
15 | 15 | <tr class="changeset <%= cycle 'odd', 'even' %>"> |
|
16 | 16 | <td class="id"><%= link_to format_revision(changeset.revision), :action => 'revision', :id => project, :rev => changeset.revision %></td> |
|
17 | 17 | <td class="checkbox"><%= radio_button_tag('rev', changeset.revision, (line_num==1), :id => "cb-#{line_num}", :onclick => "$('cbto-#{line_num+1}').checked=true;") if show_diff && (line_num < revisions.size) %></td> |
|
18 | 18 | <td class="checkbox"><%= radio_button_tag('rev_to', changeset.revision, (line_num==2), :id => "cbto-#{line_num}", :onclick => "if ($('cb-#{line_num}').checked==true) {$('cb-#{line_num-1}').checked=true;}") if show_diff && (line_num > 1) %></td> |
|
19 | 19 | <td class="committed_on"><%= format_time(changeset.committed_on) %></td> |
|
20 | 20 | <td class="author"><%=h changeset.committer.to_s.split('<').first %></td> |
|
21 | 21 | <td class="comments"><%= textilizable(changeset.comments) %></td> |
|
22 | 22 | </tr> |
|
23 | 23 | <% line_num += 1 %> |
|
24 | 24 | <% end %> |
|
25 | 25 | </tbody> |
|
26 | 26 | </table> |
|
27 | 27 | <%= submit_tag(l(:label_view_diff), :name => nil) if show_diff %> |
|
28 | 28 | <% end %> |
@@ -1,19 +1,19 | |||
|
1 | 1 | <h2><%= render :partial => 'navigation', :locals => { :path => @path, :kind => (@entry ? @entry.kind : nil), :revision => @rev } %></h2> |
|
2 | 2 | |
|
3 | 3 | <h3><%=h @entry.name %></h3> |
|
4 | 4 | |
|
5 | 5 | <p> |
|
6 | 6 | <% if @repository.supports_cat? %> |
|
7 | <%= link_to l(:button_view), {:action => 'entry', :id => @project, :path => @path, :rev => @rev } %> | | |
|
7 | <%= link_to l(:button_view), {:action => 'entry', :id => @project, :path => to_path_param(@path), :rev => @rev } %> | | |
|
8 | 8 | <% end %> |
|
9 | 9 | <% if @repository.supports_annotate? %> |
|
10 | <%= link_to l(:button_annotate), {:action => 'annotate', :id => @project, :path => @path, :rev => @rev } %> | | |
|
10 | <%= link_to l(:button_annotate), {:action => 'annotate', :id => @project, :path => to_path_param(@path), :rev => @rev } %> | | |
|
11 | 11 | <% end %> |
|
12 | <%= link_to(l(:button_download), {:action => 'entry', :id => @project, :path => @path, :rev => @rev, :format => 'raw' }) if @repository.supports_cat? %> | |
|
12 | <%= link_to(l(:button_download), {:action => 'entry', :id => @project, :path => to_path_param(@path), :rev => @rev, :format => 'raw' }) if @repository.supports_cat? %> | |
|
13 | 13 | <%= "(#{number_to_human_size(@entry.size)})" if @entry.size %> |
|
14 | 14 | </p> |
|
15 | 15 | |
|
16 | 16 | <%= render(:partial => 'revisions', |
|
17 | 17 | :locals => {:project => @project, :path => @path, :revisions => @changesets, :entry => @entry }) unless @changesets.empty? %> |
|
18 | 18 | |
|
19 | 19 | <% html_title(l(:label_change_plural)) -%> |
@@ -1,71 +1,71 | |||
|
1 | 1 | <div class="contextual"> |
|
2 | 2 | « |
|
3 | 3 | <% unless @changeset.previous.nil? -%> |
|
4 | 4 | <%= link_to l(:label_previous), :controller => 'repositories', :action => 'revision', :id => @project, :rev => @changeset.previous.revision %> |
|
5 | 5 | <% else -%> |
|
6 | 6 | <%= l(:label_previous) %> |
|
7 | 7 | <% end -%> |
|
8 | 8 | | |
|
9 | 9 | <% unless @changeset.next.nil? -%> |
|
10 | 10 | <%= link_to l(:label_next), :controller => 'repositories', :action => 'revision', :id => @project, :rev => @changeset.next.revision %> |
|
11 | 11 | <% else -%> |
|
12 | 12 | <%= l(:label_next) %> |
|
13 | 13 | <% end -%> |
|
14 | 14 | » |
|
15 | 15 | |
|
16 | 16 | <% form_tag({:controller => 'repositories', :action => 'revision', :id => @project, :rev => nil}, :method => :get) do %> |
|
17 | 17 | <%= text_field_tag 'rev', @rev, :size => 5 %> |
|
18 | 18 | <%= submit_tag 'OK', :name => nil %> |
|
19 | 19 | <% end %> |
|
20 | 20 | </div> |
|
21 | 21 | |
|
22 | 22 | <h2><%= l(:label_revision) %> <%= format_revision(@changeset.revision) %></h2> |
|
23 | 23 | |
|
24 | 24 | <p><% if @changeset.scmid %>ID: <%= @changeset.scmid %><br /><% end %> |
|
25 | 25 | <em><%= @changeset.committer.to_s.split('<').first %>, <%= format_time(@changeset.committed_on) %></em></p> |
|
26 | 26 | |
|
27 | 27 | <%= textilizable @changeset.comments %> |
|
28 | 28 | |
|
29 | 29 | <% if @changeset.issues.any? %> |
|
30 | 30 | <h3><%= l(:label_related_issues) %></h3> |
|
31 | 31 | <ul> |
|
32 | 32 | <% @changeset.issues.each do |issue| %> |
|
33 | 33 | <li><%= link_to_issue issue %>: <%=h issue.subject %></li> |
|
34 | 34 | <% end %> |
|
35 | 35 | </ul> |
|
36 | 36 | <% end %> |
|
37 | 37 | |
|
38 | 38 | <h3><%= l(:label_attachment_plural) %></h3> |
|
39 | 39 | <div style="float:right;"> |
|
40 | 40 | <div class="square action_A"></div> <div style="float:left;"><%= l(:label_added) %> </div> |
|
41 | 41 | <div class="square action_M"></div> <div style="float:left;"><%= l(:label_modified) %> </div> |
|
42 | 42 | <div class="square action_D"></div> <div style="float:left;"><%= l(:label_deleted) %> </div> |
|
43 | 43 | </div> |
|
44 | 44 | <p><%= link_to(l(:label_view_diff), :action => 'diff', :id => @project, :path => "", :rev => @changeset.revision) if @changeset.changes.any? %></p> |
|
45 | 45 | <table class="list"> |
|
46 | 46 | <tbody> |
|
47 | 47 | <% @changes.each do |change| %> |
|
48 | 48 | <tr class="<%= cycle 'odd', 'even' %>"> |
|
49 | 49 | <td><div class="square action_<%= change.action %>"></div> |
|
50 | 50 | <% if change.action == "D" -%> |
|
51 | 51 | <%= change.path -%> |
|
52 | 52 | <% else -%> |
|
53 | 53 | <%= link_to change.path, :action => 'entry', :id => @project, :path => without_leading_slash(change.relative_path), :rev => @changeset.revision -%> |
|
54 | 54 | <% end -%> |
|
55 | 55 | <%= "(#{change.revision})" unless change.revision.blank? %></td> |
|
56 | 56 | <td align="right"> |
|
57 | 57 | <% if change.action == "M" %> |
|
58 |
<%= link_to l(:label_view_diff), :action => 'diff', :id => @project, :path => |
|
|
58 | <%= link_to l(:label_view_diff), :action => 'diff', :id => @project, :path => to_path_param(change.relative_path), :rev => @changeset.revision %> | |
|
59 | 59 | <% end %> |
|
60 | 60 | </td> |
|
61 | 61 | </tr> |
|
62 | 62 | <% end %> |
|
63 | 63 | </tbody> |
|
64 | 64 | </table> |
|
65 | 65 | <p class="pagination"><%= pagination_links_full @changes_pages %></p> |
|
66 | 66 | |
|
67 | 67 | <% content_for :header_tags do %> |
|
68 | 68 | <%= stylesheet_link_tag "scm" %> |
|
69 | 69 | <% end %> |
|
70 | 70 | |
|
71 | 71 | <% html_title("#{l(:label_revision)} #{@changeset.revision}") -%> |
@@ -1,19 +1,109 | |||
|
1 | # Don't change this file. Configuration is done in config/environment.rb and config/environments/*.rb | |
|
1 | # Don't change this file! | |
|
2 | # Configure your app in config/environment.rb and config/environments/*.rb | |
|
2 | 3 | |
|
3 | unless defined?(RAILS_ROOT) | |
|
4 | root_path = File.join(File.dirname(__FILE__), '..') | |
|
5 | unless RUBY_PLATFORM =~ /mswin32/ | |
|
6 | require 'pathname' | |
|
7 | root_path = Pathname.new(root_path).cleanpath(true).to_s | |
|
4 | RAILS_ROOT = "#{File.dirname(__FILE__)}/.." unless defined?(RAILS_ROOT) | |
|
5 | ||
|
6 | module Rails | |
|
7 | class << self | |
|
8 | def boot! | |
|
9 | unless booted? | |
|
10 | preinitialize | |
|
11 | pick_boot.run | |
|
12 | end | |
|
13 | end | |
|
14 | ||
|
15 | def booted? | |
|
16 | defined? Rails::Initializer | |
|
17 | end | |
|
18 | ||
|
19 | def pick_boot | |
|
20 | (vendor_rails? ? VendorBoot : GemBoot).new | |
|
21 | end | |
|
22 | ||
|
23 | def vendor_rails? | |
|
24 | File.exist?("#{RAILS_ROOT}/vendor/rails") | |
|
25 | end | |
|
26 | ||
|
27 | def preinitialize | |
|
28 | load(preinitializer_path) if File.exist?(preinitializer_path) | |
|
29 | end | |
|
30 | ||
|
31 | def preinitializer_path | |
|
32 | "#{RAILS_ROOT}/config/preinitializer.rb" | |
|
33 | end | |
|
8 | 34 | end |
|
9 | RAILS_ROOT = root_path | |
|
10 | end | |
|
11 | 35 | |
|
12 | if File.directory?("#{RAILS_ROOT}/vendor/rails") | |
|
13 | require "#{RAILS_ROOT}/vendor/rails/railties/lib/initializer" | |
|
14 | else | |
|
15 | require 'rubygems' | |
|
16 | require 'initializer' | |
|
36 | class Boot | |
|
37 | def run | |
|
38 | load_initializer | |
|
39 | Rails::Initializer.run(:set_load_path) | |
|
40 | end | |
|
41 | end | |
|
42 | ||
|
43 | class VendorBoot < Boot | |
|
44 | def load_initializer | |
|
45 | require "#{RAILS_ROOT}/vendor/rails/railties/lib/initializer" | |
|
46 | Rails::Initializer.run(:install_gem_spec_stubs) | |
|
47 | end | |
|
48 | end | |
|
49 | ||
|
50 | class GemBoot < Boot | |
|
51 | def load_initializer | |
|
52 | self.class.load_rubygems | |
|
53 | load_rails_gem | |
|
54 | require 'initializer' | |
|
55 | end | |
|
56 | ||
|
57 | def load_rails_gem | |
|
58 | if version = self.class.gem_version | |
|
59 | gem 'rails', version | |
|
60 | else | |
|
61 | gem 'rails' | |
|
62 | end | |
|
63 | rescue Gem::LoadError => load_error | |
|
64 | $stderr.puts %(Missing the Rails #{version} gem. Please `gem install -v=#{version} rails`, update your RAILS_GEM_VERSION setting in config/environment.rb for the Rails version you do have installed, or comment out RAILS_GEM_VERSION to use the latest version installed.) | |
|
65 | exit 1 | |
|
66 | end | |
|
67 | ||
|
68 | class << self | |
|
69 | def rubygems_version | |
|
70 | Gem::RubyGemsVersion if defined? Gem::RubyGemsVersion | |
|
71 | end | |
|
72 | ||
|
73 | def gem_version | |
|
74 | if defined? RAILS_GEM_VERSION | |
|
75 | RAILS_GEM_VERSION | |
|
76 | elsif ENV.include?('RAILS_GEM_VERSION') | |
|
77 | ENV['RAILS_GEM_VERSION'] | |
|
78 | else | |
|
79 | parse_gem_version(read_environment_rb) | |
|
80 | end | |
|
81 | end | |
|
82 | ||
|
83 | def load_rubygems | |
|
84 | require 'rubygems' | |
|
85 | ||
|
86 | unless rubygems_version >= '0.9.4' | |
|
87 | $stderr.puts %(Rails requires RubyGems >= 0.9.4 (you have #{rubygems_version}). Please `gem update --system` and try again.) | |
|
88 | exit 1 | |
|
89 | end | |
|
90 | ||
|
91 | rescue LoadError | |
|
92 | $stderr.puts %(Rails requires RubyGems >= 0.9.4. Please install RubyGems and try again: http://rubygems.rubyforge.org) | |
|
93 | exit 1 | |
|
94 | end | |
|
95 | ||
|
96 | def parse_gem_version(text) | |
|
97 | $1 if text =~ /^[^#]*RAILS_GEM_VERSION\s*=\s*["']([!~<>=]*\s*[\d.]+)["']/ | |
|
98 | end | |
|
99 | ||
|
100 | private | |
|
101 | def read_environment_rb | |
|
102 | File.read("#{RAILS_ROOT}/config/environment.rb") | |
|
103 | end | |
|
104 | end | |
|
105 | end | |
|
17 | 106 | end |
|
18 | 107 | |
|
19 | Rails::Initializer.run(:set_load_path) | |
|
108 | # All that for this: | |
|
109 | Rails.boot! |
@@ -1,102 +1,73 | |||
|
1 | 1 | # Be sure to restart your web server when you modify this file. |
|
2 | 2 | |
|
3 | 3 | # Uncomment below to force Rails into production mode when |
|
4 | 4 | # you don't control web/app server and can't set it the proper way |
|
5 | 5 | # ENV['RAILS_ENV'] ||= 'production' |
|
6 | 6 | |
|
7 | 7 | # Specifies gem version of Rails to use when vendor/rails is not present |
|
8 |
RAILS_GEM_VERSION = '2. |
|
|
8 | RAILS_GEM_VERSION = '2.1.0' unless defined? RAILS_GEM_VERSION | |
|
9 | 9 | |
|
10 | 10 | # Bootstrap the Rails environment, frameworks, and default configuration |
|
11 | 11 | require File.join(File.dirname(__FILE__), 'boot') |
|
12 | 12 | |
|
13 | 13 | # Load Engine plugin if available |
|
14 | 14 | begin |
|
15 | 15 | require File.join(File.dirname(__FILE__), '../vendor/plugins/engines/boot') |
|
16 | 16 | rescue LoadError |
|
17 | 17 | # Not available |
|
18 | 18 | end |
|
19 | 19 | |
|
20 | 20 | Rails::Initializer.run do |config| |
|
21 | 21 | # Settings in config/environments/* take precedence those specified here |
|
22 | 22 | |
|
23 | 23 | # Skip frameworks you're not going to use |
|
24 | 24 | # config.frameworks -= [ :action_web_service, :action_mailer ] |
|
25 | 25 | |
|
26 | 26 | # Add additional load paths for sweepers |
|
27 | 27 | config.load_paths += %W( #{RAILS_ROOT}/app/sweepers ) |
|
28 | 28 | |
|
29 | 29 | # Force all environments to use the same logger level |
|
30 | 30 | # (by default production uses :info, the others :debug) |
|
31 | 31 | # config.log_level = :debug |
|
32 | 32 | |
|
33 | 33 | # Use the database for sessions instead of the file system |
|
34 | 34 | # (create the session table with 'rake db:sessions:create') |
|
35 | 35 | # config.action_controller.session_store = :active_record_store |
|
36 | 36 | config.action_controller.session_store = :PStore |
|
37 | 37 | |
|
38 | 38 | # Enable page/fragment caching by setting a file-based store |
|
39 | 39 | # (remember to create the caching directory and make it readable to the application) |
|
40 | 40 | # config.action_controller.fragment_cache_store = :file_store, "#{RAILS_ROOT}/cache" |
|
41 | 41 | |
|
42 | 42 | # Activate observers that should always be running |
|
43 | 43 | # config.active_record.observers = :cacher, :garbage_collector |
|
44 | 44 | config.active_record.observers = :message_observer |
|
45 | 45 | |
|
46 | 46 | # Make Active Record use UTC-base instead of local time |
|
47 | 47 | # config.active_record.default_timezone = :utc |
|
48 | 48 | |
|
49 | 49 | # Use Active Record's schema dumper instead of SQL when creating the test database |
|
50 | 50 | # (enables use of different database adapters for development and test environments) |
|
51 | 51 | # config.active_record.schema_format = :ruby |
|
52 | 52 | |
|
53 | 53 | # See Rails::Configuration for more options |
|
54 | 54 | |
|
55 | 55 | # SMTP server configuration |
|
56 | 56 | config.action_mailer.smtp_settings = { |
|
57 | 57 | :address => "127.0.0.1", |
|
58 | 58 | :port => 25, |
|
59 | 59 | :domain => "somenet.foo", |
|
60 | 60 | :authentication => :login, |
|
61 | 61 | :user_name => "redmine@somenet.foo", |
|
62 | 62 | :password => "redmine", |
|
63 | 63 | } |
|
64 | 64 | |
|
65 | 65 | config.action_mailer.perform_deliveries = true |
|
66 | 66 | |
|
67 | 67 | # Tell ActionMailer not to deliver emails to the real world. |
|
68 | 68 | # The :test delivery method accumulates sent emails in the |
|
69 | 69 | # ActionMailer::Base.deliveries array. |
|
70 | 70 | #config.action_mailer.delivery_method = :test |
|
71 | 71 | config.action_mailer.delivery_method = :smtp |
|
72 | 72 | |
|
73 | 73 | end |
|
74 | ||
|
75 | ActiveRecord::Errors.default_error_messages = { | |
|
76 | :inclusion => "activerecord_error_inclusion", | |
|
77 | :exclusion => "activerecord_error_exclusion", | |
|
78 | :invalid => "activerecord_error_invalid", | |
|
79 | :confirmation => "activerecord_error_confirmation", | |
|
80 | :accepted => "activerecord_error_accepted", | |
|
81 | :empty => "activerecord_error_empty", | |
|
82 | :blank => "activerecord_error_blank", | |
|
83 | :too_long => "activerecord_error_too_long", | |
|
84 | :too_short => "activerecord_error_too_short", | |
|
85 | :wrong_length => "activerecord_error_wrong_length", | |
|
86 | :taken => "activerecord_error_taken", | |
|
87 | :not_a_number => "activerecord_error_not_a_number" | |
|
88 | } | |
|
89 | ||
|
90 | ActionView::Base.field_error_proc = Proc.new{ |html_tag, instance| "#{html_tag}" } | |
|
91 | ||
|
92 | Mime::SET << Mime::CSV unless Mime::SET.include?(Mime::CSV) | |
|
93 | Mime::Type.register 'application/pdf', :pdf | |
|
94 | ||
|
95 | GLoc.set_config :default_language => :en | |
|
96 | GLoc.clear_strings | |
|
97 | GLoc.set_kcode | |
|
98 | GLoc.load_localized_strings | |
|
99 | GLoc.set_config(:raise_string_not_found_errors => false) | |
|
100 | ||
|
101 | require 'redmine' | |
|
102 |
@@ -1,15 +1,15 | |||
|
1 | 1 | class AddEnumerationsPosition < ActiveRecord::Migration |
|
2 | 2 | def self.up |
|
3 | 3 | add_column(:enumerations, :position, :integer, :default => 1) unless Enumeration.column_names.include?('position') |
|
4 |
Enumeration.find(:all).group_by(&:opt).each |
|
|
4 | Enumeration.find(:all).group_by(&:opt).each do |opt, enums| | |
|
5 | 5 | enums.each_with_index do |enum, i| |
|
6 | 6 | # do not call model callbacks |
|
7 | 7 | Enumeration.update_all "position = #{i+1}", {:id => enum.id} |
|
8 | 8 | end |
|
9 | 9 | end |
|
10 | 10 | end |
|
11 | 11 | |
|
12 | 12 | def self.down |
|
13 | 13 | remove_column :enumerations, :position |
|
14 | 14 | end |
|
15 | 15 | end |
@@ -1,15 +1,15 | |||
|
1 | 1 | class AddCustomFieldsPosition < ActiveRecord::Migration |
|
2 | 2 | def self.up |
|
3 | 3 | add_column(:custom_fields, :position, :integer, :default => 1) |
|
4 |
CustomField.find(:all).group_by(&:type).each |
|
|
4 | CustomField.find(:all).group_by(&:type).each do |t, fields| | |
|
5 | 5 | fields.each_with_index do |field, i| |
|
6 | 6 | # do not call model callbacks |
|
7 | 7 | CustomField.update_all "position = #{i+1}", {:id => field.id} |
|
8 | 8 | end |
|
9 | 9 | end |
|
10 | 10 | end |
|
11 | 11 | |
|
12 | 12 | def self.down |
|
13 | 13 | remove_column :custom_fields, :position |
|
14 | 14 | end |
|
15 | 15 | end |
@@ -1,51 +1,51 | |||
|
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 'action_view/helpers/form_helper' |
|
19 | 19 | |
|
20 | 20 | class TabularFormBuilder < ActionView::Helpers::FormBuilder |
|
21 | 21 | include GLoc |
|
22 | 22 | |
|
23 | 23 | def initialize(object_name, object, template, options, proc) |
|
24 | 24 | set_language_if_valid options.delete(:lang) |
|
25 | @object_name, @object, @template, @options, @proc = object_name, object, template, options, proc | |
|
25 | super | |
|
26 | 26 | end |
|
27 | 27 | |
|
28 | 28 | (field_helpers - %w(radio_button hidden_field) + %w(date_select)).each do |selector| |
|
29 | 29 | src = <<-END_SRC |
|
30 | 30 | def #{selector}(field, options = {}) |
|
31 | 31 | return super if options.delete :no_label |
|
32 | 32 | label_text = l(options[:label]) if options[:label] |
|
33 | 33 | label_text ||= l(("field_"+field.to_s.gsub(/\_id$/, "")).to_sym) |
|
34 | 34 | label_text << @template.content_tag("span", " *", :class => "required") if options.delete(:required) |
|
35 | 35 | label = @template.content_tag("label", label_text, |
|
36 | 36 | :class => (@object && @object.errors[field] ? "error" : nil), |
|
37 | 37 | :for => (@object_name.to_s + "_" + field.to_s)) |
|
38 | 38 | label + super |
|
39 | 39 | end |
|
40 | 40 | END_SRC |
|
41 | 41 | class_eval src, __FILE__, __LINE__ |
|
42 | 42 | end |
|
43 | 43 | |
|
44 | 44 | def select(field, choices, options = {}, html_options = {}) |
|
45 | 45 | label_text = l(("field_"+field.to_s.gsub(/\_id$/, "")).to_sym) + (options.delete(:required) ? @template.content_tag("span", " *", :class => "required"): "") |
|
46 | 46 | label = @template.content_tag("label", label_text, |
|
47 | 47 | :class => (@object && @object.errors[field] ? "error" : nil), |
|
48 | 48 | :for => (@object_name.to_s + "_" + field.to_s)) |
|
49 | 49 | label + super |
|
50 | 50 | end |
|
51 | 51 | end |
@@ -1,307 +1,307 | |||
|
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 '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 < Test::Unit::TestCase |
|
25 | 25 | fixtures :projects, :versions, :users, :roles, :members, :issues, :journals, :journal_details, |
|
26 | 26 | :trackers, :projects_trackers, :issue_statuses, :enabled_modules, :enumerations, :boards, :messages |
|
27 | 27 | |
|
28 | 28 | def setup |
|
29 | 29 | @controller = ProjectsController.new |
|
30 | 30 | @request = ActionController::TestRequest.new |
|
31 | 31 | @response = ActionController::TestResponse.new |
|
32 | 32 | @request.session[:user_id] = nil |
|
33 | 33 | end |
|
34 | 34 | |
|
35 | 35 | def test_index |
|
36 | 36 | get :index |
|
37 | 37 | assert_response :success |
|
38 | 38 | assert_template 'index' |
|
39 | 39 | assert_not_nil assigns(:project_tree) |
|
40 | 40 | # Root project as hash key |
|
41 |
assert assigns(:project_tree). |
|
|
41 | assert assigns(:project_tree).keys.include?(Project.find(1)) | |
|
42 | 42 | # Subproject in corresponding value |
|
43 | 43 | assert assigns(:project_tree)[Project.find(1)].include?(Project.find(3)) |
|
44 | 44 | end |
|
45 | 45 | |
|
46 | 46 | def test_show_by_id |
|
47 | 47 | get :show, :id => 1 |
|
48 | 48 | assert_response :success |
|
49 | 49 | assert_template 'show' |
|
50 | 50 | assert_not_nil assigns(:project) |
|
51 | 51 | end |
|
52 | 52 | |
|
53 | 53 | def test_show_by_identifier |
|
54 | 54 | get :show, :id => 'ecookbook' |
|
55 | 55 | assert_response :success |
|
56 | 56 | assert_template 'show' |
|
57 | 57 | assert_not_nil assigns(:project) |
|
58 | 58 | assert_equal Project.find_by_identifier('ecookbook'), assigns(:project) |
|
59 | 59 | end |
|
60 | 60 | |
|
61 | 61 | def test_private_subprojects_hidden |
|
62 | 62 | get :show, :id => 'ecookbook' |
|
63 | 63 | assert_response :success |
|
64 | 64 | assert_template 'show' |
|
65 | 65 | assert_no_tag :tag => 'a', :content => /Private child/ |
|
66 | 66 | end |
|
67 | 67 | |
|
68 | 68 | def test_private_subprojects_visible |
|
69 | 69 | @request.session[:user_id] = 2 # manager who is a member of the private subproject |
|
70 | 70 | get :show, :id => 'ecookbook' |
|
71 | 71 | assert_response :success |
|
72 | 72 | assert_template 'show' |
|
73 | 73 | assert_tag :tag => 'a', :content => /Private child/ |
|
74 | 74 | end |
|
75 | 75 | |
|
76 | 76 | def test_settings |
|
77 | 77 | @request.session[:user_id] = 2 # manager |
|
78 | 78 | get :settings, :id => 1 |
|
79 | 79 | assert_response :success |
|
80 | 80 | assert_template 'settings' |
|
81 | 81 | end |
|
82 | 82 | |
|
83 | 83 | def test_edit |
|
84 | 84 | @request.session[:user_id] = 2 # manager |
|
85 | 85 | post :edit, :id => 1, :project => {:name => 'Test changed name', |
|
86 | 86 | :issue_custom_field_ids => ['']} |
|
87 | 87 | assert_redirected_to 'projects/settings/ecookbook' |
|
88 | 88 | project = Project.find(1) |
|
89 | 89 | assert_equal 'Test changed name', project.name |
|
90 | 90 | end |
|
91 | 91 | |
|
92 | 92 | def test_get_destroy |
|
93 | 93 | @request.session[:user_id] = 1 # admin |
|
94 | 94 | get :destroy, :id => 1 |
|
95 | 95 | assert_response :success |
|
96 | 96 | assert_template 'destroy' |
|
97 | 97 | assert_not_nil Project.find_by_id(1) |
|
98 | 98 | end |
|
99 | 99 | |
|
100 | 100 | def test_post_destroy |
|
101 | 101 | @request.session[:user_id] = 1 # admin |
|
102 | 102 | post :destroy, :id => 1, :confirm => 1 |
|
103 | 103 | assert_redirected_to 'admin/projects' |
|
104 | 104 | assert_nil Project.find_by_id(1) |
|
105 | 105 | end |
|
106 | 106 | |
|
107 | 107 | def test_list_files |
|
108 | 108 | get :list_files, :id => 1 |
|
109 | 109 | assert_response :success |
|
110 | 110 | assert_template 'list_files' |
|
111 | 111 | assert_not_nil assigns(:versions) |
|
112 | 112 | end |
|
113 | 113 | |
|
114 | 114 | def test_changelog |
|
115 | 115 | get :changelog, :id => 1 |
|
116 | 116 | assert_response :success |
|
117 | 117 | assert_template 'changelog' |
|
118 | 118 | assert_not_nil assigns(:versions) |
|
119 | 119 | end |
|
120 | 120 | |
|
121 | 121 | def test_roadmap |
|
122 | 122 | get :roadmap, :id => 1 |
|
123 | 123 | assert_response :success |
|
124 | 124 | assert_template 'roadmap' |
|
125 | 125 | assert_not_nil assigns(:versions) |
|
126 | 126 | # Version with no date set appears |
|
127 | 127 | assert assigns(:versions).include?(Version.find(3)) |
|
128 | 128 | # Completed version doesn't appear |
|
129 | 129 | assert !assigns(:versions).include?(Version.find(1)) |
|
130 | 130 | end |
|
131 | 131 | |
|
132 | 132 | def test_roadmap_with_completed_versions |
|
133 | 133 | get :roadmap, :id => 1, :completed => 1 |
|
134 | 134 | assert_response :success |
|
135 | 135 | assert_template 'roadmap' |
|
136 | 136 | assert_not_nil assigns(:versions) |
|
137 | 137 | # Version with no date set appears |
|
138 | 138 | assert assigns(:versions).include?(Version.find(3)) |
|
139 | 139 | # Completed version appears |
|
140 | 140 | assert assigns(:versions).include?(Version.find(1)) |
|
141 | 141 | end |
|
142 | 142 | |
|
143 | 143 | def test_project_activity |
|
144 | 144 | get :activity, :id => 1, :with_subprojects => 0 |
|
145 | 145 | assert_response :success |
|
146 | 146 | assert_template 'activity' |
|
147 | 147 | assert_not_nil assigns(:events_by_day) |
|
148 | 148 | assert_not_nil assigns(:events) |
|
149 | 149 | |
|
150 | 150 | # subproject issue not included by default |
|
151 | 151 | assert !assigns(:events).include?(Issue.find(5)) |
|
152 | 152 | |
|
153 | 153 | assert_tag :tag => "h3", |
|
154 | 154 | :content => /#{2.days.ago.to_date.day}/, |
|
155 | 155 | :sibling => { :tag => "dl", |
|
156 | 156 | :child => { :tag => "dt", |
|
157 | 157 | :attributes => { :class => /issue-edit/ }, |
|
158 | 158 | :child => { :tag => "a", |
|
159 | 159 | :content => /(#{IssueStatus.find(2).name})/, |
|
160 | 160 | } |
|
161 | 161 | } |
|
162 | 162 | } |
|
163 | 163 | |
|
164 | 164 | get :activity, :id => 1, :from => 3.days.ago.to_date |
|
165 | 165 | assert_response :success |
|
166 | 166 | assert_template 'activity' |
|
167 | 167 | assert_not_nil assigns(:events_by_day) |
|
168 | 168 | |
|
169 | 169 | assert_tag :tag => "h3", |
|
170 | 170 | :content => /#{3.day.ago.to_date.day}/, |
|
171 | 171 | :sibling => { :tag => "dl", |
|
172 | 172 | :child => { :tag => "dt", |
|
173 | 173 | :attributes => { :class => /issue/ }, |
|
174 | 174 | :child => { :tag => "a", |
|
175 | 175 | :content => /#{Issue.find(1).subject}/, |
|
176 | 176 | } |
|
177 | 177 | } |
|
178 | 178 | } |
|
179 | 179 | end |
|
180 | 180 | |
|
181 | 181 | def test_activity_with_subprojects |
|
182 | 182 | get :activity, :id => 1, :with_subprojects => 1 |
|
183 | 183 | assert_response :success |
|
184 | 184 | assert_template 'activity' |
|
185 | 185 | assert_not_nil assigns(:events) |
|
186 | 186 | |
|
187 | 187 | assert assigns(:events).include?(Issue.find(1)) |
|
188 | 188 | assert !assigns(:events).include?(Issue.find(4)) |
|
189 | 189 | # subproject issue |
|
190 | 190 | assert assigns(:events).include?(Issue.find(5)) |
|
191 | 191 | end |
|
192 | 192 | |
|
193 | 193 | def test_global_activity_anonymous |
|
194 | 194 | get :activity |
|
195 | 195 | assert_response :success |
|
196 | 196 | assert_template 'activity' |
|
197 | 197 | assert_not_nil assigns(:events) |
|
198 | 198 | |
|
199 | 199 | assert assigns(:events).include?(Issue.find(1)) |
|
200 | 200 | # Issue of a private project |
|
201 | 201 | assert !assigns(:events).include?(Issue.find(4)) |
|
202 | 202 | end |
|
203 | 203 | |
|
204 | 204 | def test_global_activity_logged_user |
|
205 | 205 | @request.session[:user_id] = 2 # manager |
|
206 | 206 | get :activity |
|
207 | 207 | assert_response :success |
|
208 | 208 | assert_template 'activity' |
|
209 | 209 | assert_not_nil assigns(:events) |
|
210 | 210 | |
|
211 | 211 | assert assigns(:events).include?(Issue.find(1)) |
|
212 | 212 | # Issue of a private project the user belongs to |
|
213 | 213 | assert assigns(:events).include?(Issue.find(4)) |
|
214 | 214 | end |
|
215 | 215 | |
|
216 | 216 | |
|
217 | 217 | def test_global_activity_with_all_types |
|
218 | 218 | get :activity, :show_issues => 1, :show_news => 1, :show_files => 1, :show_documents => 1, :show_changesets => 1, :show_wiki_pages => 1, :show_messages => 1 |
|
219 | 219 | assert_response :success |
|
220 | 220 | assert_template 'activity' |
|
221 | 221 | assert_not_nil assigns(:events) |
|
222 | 222 | |
|
223 | 223 | assert assigns(:events).include?(Issue.find(1)) |
|
224 | 224 | assert !assigns(:events).include?(Issue.find(4)) |
|
225 | 225 | assert assigns(:events).include?(Message.find(5)) |
|
226 | 226 | end |
|
227 | 227 | |
|
228 | 228 | def test_calendar |
|
229 | 229 | get :calendar, :id => 1 |
|
230 | 230 | assert_response :success |
|
231 | 231 | assert_template 'calendar' |
|
232 | 232 | assert_not_nil assigns(:calendar) |
|
233 | 233 | end |
|
234 | 234 | |
|
235 | 235 | def test_calendar_with_subprojects_should_not_show_private_subprojects |
|
236 | 236 | get :calendar, :id => 1, :with_subprojects => 1, :tracker_ids => [1, 2] |
|
237 | 237 | assert_response :success |
|
238 | 238 | assert_template 'calendar' |
|
239 | 239 | assert_not_nil assigns(:calendar) |
|
240 | 240 | assert_no_tag :tag => 'a', :content => /#6/ |
|
241 | 241 | end |
|
242 | 242 | |
|
243 | 243 | def test_calendar_with_subprojects_should_show_private_subprojects |
|
244 | 244 | @request.session[:user_id] = 2 |
|
245 | 245 | get :calendar, :id => 1, :with_subprojects => 1, :tracker_ids => [1, 2] |
|
246 | 246 | assert_response :success |
|
247 | 247 | assert_template 'calendar' |
|
248 | 248 | assert_not_nil assigns(:calendar) |
|
249 | 249 | assert_tag :tag => 'a', :content => /#6/ |
|
250 | 250 | end |
|
251 | 251 | |
|
252 | 252 | def test_gantt |
|
253 | 253 | get :gantt, :id => 1 |
|
254 | 254 | assert_response :success |
|
255 | 255 | assert_template 'gantt.rhtml' |
|
256 | 256 | events = assigns(:events) |
|
257 | 257 | assert_not_nil events |
|
258 | 258 | # Issue with start and due dates |
|
259 | 259 | i = Issue.find(1) |
|
260 | 260 | assert_not_nil i.due_date |
|
261 | 261 | assert events.include?(Issue.find(1)) |
|
262 | 262 | # Issue with without due date but targeted to a version with date |
|
263 | 263 | i = Issue.find(2) |
|
264 | 264 | assert_nil i.due_date |
|
265 | 265 | assert events.include?(i) |
|
266 | 266 | end |
|
267 | 267 | |
|
268 | 268 | def test_gantt_with_subprojects_should_not_show_private_subprojects |
|
269 | 269 | get :gantt, :id => 1, :with_subprojects => 1, :tracker_ids => [1, 2] |
|
270 | 270 | assert_response :success |
|
271 | 271 | assert_template 'gantt.rhtml' |
|
272 | 272 | assert_not_nil assigns(:events) |
|
273 | 273 | assert_no_tag :tag => 'a', :content => /#6/ |
|
274 | 274 | end |
|
275 | 275 | |
|
276 | 276 | def test_gantt_with_subprojects_should_show_private_subprojects |
|
277 | 277 | @request.session[:user_id] = 2 |
|
278 | 278 | get :gantt, :id => 1, :with_subprojects => 1, :tracker_ids => [1, 2] |
|
279 | 279 | assert_response :success |
|
280 | 280 | assert_template 'gantt.rhtml' |
|
281 | 281 | assert_not_nil assigns(:events) |
|
282 | 282 | assert_tag :tag => 'a', :content => /#6/ |
|
283 | 283 | end |
|
284 | 284 | |
|
285 | 285 | def test_gantt_export_to_pdf |
|
286 | 286 | get :gantt, :id => 1, :format => 'pdf' |
|
287 | 287 | assert_response :success |
|
288 | 288 | assert_template 'gantt.rfpdf' |
|
289 | 289 | assert_equal 'application/pdf', @response.content_type |
|
290 | 290 | assert_not_nil assigns(:events) |
|
291 | 291 | end |
|
292 | 292 | |
|
293 | 293 | def test_archive |
|
294 | 294 | @request.session[:user_id] = 1 # admin |
|
295 | 295 | post :archive, :id => 1 |
|
296 | 296 | assert_redirected_to 'admin/projects' |
|
297 | 297 | assert !Project.find(1).active? |
|
298 | 298 | end |
|
299 | 299 | |
|
300 | 300 | def test_unarchive |
|
301 | 301 | @request.session[:user_id] = 1 # admin |
|
302 | 302 | Project.find(1).archive |
|
303 | 303 | post :unarchive, :id => 1 |
|
304 | 304 | assert_redirected_to 'admin/projects' |
|
305 | 305 | assert Project.find(1).active? |
|
306 | 306 | end |
|
307 | 307 | end |
@@ -1,89 +1,92 | |||
|
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 | |
|
20 | 20 | class IssuesTest < ActionController::IntegrationTest |
|
21 | 21 | fixtures :projects, |
|
22 | 22 | :users, |
|
23 | :roles, | |
|
24 | :members, | |
|
23 | 25 | :trackers, |
|
24 | 26 | :projects_trackers, |
|
27 | :enabled_modules, | |
|
25 | 28 | :issue_statuses, |
|
26 | 29 | :issues, |
|
27 | 30 | :enumerations, |
|
28 | 31 | :custom_fields, |
|
29 | 32 | :custom_values, |
|
30 | 33 | :custom_fields_trackers |
|
31 | 34 | |
|
32 | 35 | # create an issue |
|
33 | 36 | def test_add_issue |
|
34 | 37 | log_user('jsmith', 'jsmith') |
|
35 | 38 | get 'projects/1/issues/new', :tracker_id => '1' |
|
36 | 39 | assert_response :success |
|
37 | 40 | assert_template 'issues/new' |
|
38 | 41 | |
|
39 | 42 | post 'projects/1/issues/new', :tracker_id => "1", |
|
40 | 43 | :issue => { :start_date => "2006-12-26", |
|
41 | 44 | :priority_id => "3", |
|
42 | 45 | :subject => "new test issue", |
|
43 | 46 | :category_id => "", |
|
44 | 47 | :description => "new issue", |
|
45 | 48 | :done_ratio => "0", |
|
46 | 49 | :due_date => "", |
|
47 | 50 | :assigned_to_id => "" }, |
|
48 | 51 | :custom_fields => {'2' => 'Value for field 2'} |
|
49 | 52 | # find created issue |
|
50 | 53 | issue = Issue.find_by_subject("new test issue") |
|
51 | 54 | assert_kind_of Issue, issue |
|
52 | 55 | |
|
53 | 56 | # check redirection |
|
54 | 57 | assert_redirected_to "issues/show" |
|
55 | 58 | follow_redirect! |
|
56 | 59 | assert_equal issue, assigns(:issue) |
|
57 | 60 | |
|
58 | 61 | # check issue attributes |
|
59 | 62 | assert_equal 'jsmith', issue.author.login |
|
60 | 63 | assert_equal 1, issue.project.id |
|
61 | 64 | assert_equal 1, issue.status.id |
|
62 | 65 | end |
|
63 | 66 | |
|
64 | 67 | # add then remove 2 attachments to an issue |
|
65 | 68 | def test_issue_attachements |
|
66 | 69 | log_user('jsmith', 'jsmith') |
|
67 | 70 | set_tmp_attachments_directory |
|
68 | 71 | |
|
69 | 72 | post 'issues/edit/1', |
|
70 | 73 | :notes => 'Some notes', |
|
71 | 74 | :attachments => {'1' => {'file' => test_uploaded_file('testfile.txt', 'text/plain'), 'description' => 'This is an attachment'}} |
|
72 | 75 | assert_redirected_to "issues/show/1" |
|
73 | 76 | |
|
74 | 77 | # make sure attachment was saved |
|
75 | 78 | attachment = Issue.find(1).attachments.find_by_filename("testfile.txt") |
|
76 | 79 | assert_kind_of Attachment, attachment |
|
77 | 80 | assert_equal Issue.find(1), attachment.container |
|
78 | 81 | assert_equal 'This is an attachment', attachment.description |
|
79 | 82 | # verify the size of the attachment stored in db |
|
80 | 83 | #assert_equal file_data_1.length, attachment.filesize |
|
81 | 84 | # verify that the attachment was written to disk |
|
82 | 85 | assert File.exist?(attachment.diskfile) |
|
83 | 86 | |
|
84 | 87 | # remove the attachments |
|
85 | 88 | Issue.find(1).attachments.each(&:destroy) |
|
86 | 89 | assert_equal 0, Issue.find(1).attachments.length |
|
87 | 90 | end |
|
88 | 91 | |
|
89 | 92 | end |
@@ -1,84 +1,67 | |||
|
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 | ENV["RAILS_ENV"] ||= "test" |
|
19 | 19 | require File.expand_path(File.dirname(__FILE__) + "/../config/environment") |
|
20 | 20 | require 'test_help' |
|
21 | 21 | require File.expand_path(File.dirname(__FILE__) + '/helper_testcase') |
|
22 | 22 | |
|
23 | 23 | class Test::Unit::TestCase |
|
24 | 24 | # Transactional fixtures accelerate your tests by wrapping each test method |
|
25 | 25 | # in a transaction that's rolled back on completion. This ensures that the |
|
26 | 26 | # test database remains unchanged so your fixtures don't have to be reloaded |
|
27 | 27 | # between every test method. Fewer database queries means faster tests. |
|
28 | 28 | # |
|
29 | 29 | # Read Mike Clark's excellent walkthrough at |
|
30 | 30 | # http://clarkware.com/cgi/blosxom/2005/10/24#Rails10FastTesting |
|
31 | 31 | # |
|
32 | 32 | # Every Active Record database supports transactions except MyISAM tables |
|
33 | 33 | # in MySQL. Turn off transactional fixtures in this case; however, if you |
|
34 | 34 | # don't care one way or the other, switching from MyISAM to InnoDB tables |
|
35 | 35 | # is recommended. |
|
36 | 36 | self.use_transactional_fixtures = true |
|
37 | 37 | |
|
38 | 38 | # Instantiated fixtures are slow, but give you @david where otherwise you |
|
39 | 39 | # would need people(:david). If you don't want to migrate your existing |
|
40 | 40 | # test cases which use the @david style and don't mind the speed hit (each |
|
41 | 41 | # instantiated fixtures translates to a database query per test method), |
|
42 | 42 | # then set this back to true. |
|
43 | 43 | self.use_instantiated_fixtures = false |
|
44 | 44 | |
|
45 | 45 | # Add more helper methods to be used by all tests here... |
|
46 | 46 | |
|
47 | 47 | def log_user(login, password) |
|
48 | 48 | get "/account/login" |
|
49 | 49 | assert_equal nil, session[:user_id] |
|
50 | 50 | assert_response :success |
|
51 | 51 | assert_template "account/login" |
|
52 | 52 | post "/account/login", :username => login, :password => password |
|
53 | 53 | assert_redirected_to "my/page" |
|
54 | 54 | assert_equal login, User.find(session[:user_id]).login |
|
55 | 55 | end |
|
56 | 56 | |
|
57 | 57 | def test_uploaded_file(name, mime) |
|
58 | 58 | ActionController::TestUploadedFile.new(Test::Unit::TestCase.fixture_path + "/files/#{name}", mime) |
|
59 | 59 | end |
|
60 | 60 | |
|
61 | 61 | # Use a temporary directory for attachment related tests |
|
62 | 62 | def set_tmp_attachments_directory |
|
63 | 63 | Dir.mkdir "#{RAILS_ROOT}/tmp/test" unless File.directory?("#{RAILS_ROOT}/tmp/test") |
|
64 | 64 | Dir.mkdir "#{RAILS_ROOT}/tmp/test/attachments" unless File.directory?("#{RAILS_ROOT}/tmp/test/attachments") |
|
65 | 65 | Attachment.storage_path = "#{RAILS_ROOT}/tmp/test/attachments" |
|
66 | 66 | end |
|
67 | 67 | end |
|
68 | ||
|
69 | ||
|
70 | # ActionController::TestUploadedFile bug | |
|
71 | # see http://dev.rubyonrails.org/ticket/4635 | |
|
72 | class String | |
|
73 | def original_filename | |
|
74 | "testfile.txt" | |
|
75 | end | |
|
76 | ||
|
77 | def content_type | |
|
78 | "text/plain" | |
|
79 | end | |
|
80 | ||
|
81 | def read | |
|
82 | self.to_s | |
|
83 | end | |
|
84 | end |
@@ -1,33 +1,33 | |||
|
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 | |
|
20 | 20 | class RoleTest < Test::Unit::TestCase |
|
21 | 21 | fixtures :roles, :workflows |
|
22 | 22 | |
|
23 | 23 | def test_copy_workflows |
|
24 | 24 | source = Role.find(1) |
|
25 | 25 | assert_equal 90, source.workflows.size |
|
26 | 26 | |
|
27 | 27 | target = Role.new(:name => 'Target') |
|
28 | 28 | assert target.save |
|
29 |
|
|
|
29 | target.workflows.copy(source) | |
|
30 | 30 | target.reload |
|
31 | 31 | assert_equal 90, target.workflows.size |
|
32 | 32 | end |
|
33 | 33 | end |
@@ -1,33 +1,33 | |||
|
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 | |
|
20 | 20 | class TrackerTest < Test::Unit::TestCase |
|
21 | 21 | fixtures :trackers, :workflows |
|
22 | 22 | |
|
23 | 23 | def test_copy_workflows |
|
24 | 24 | source = Tracker.find(1) |
|
25 | 25 | assert_equal 89, source.workflows.size |
|
26 | 26 | |
|
27 | 27 | target = Tracker.new(:name => 'Target') |
|
28 | 28 | assert target.save |
|
29 |
|
|
|
29 | target.workflows.copy(source) | |
|
30 | 30 | target.reload |
|
31 | 31 | assert_equal 89, target.workflows.size |
|
32 | 32 | end |
|
33 | 33 | end |
@@ -1,182 +1,182 | |||
|
1 | require 'rubygems' | |
|
2 | ||
|
3 | Gem::manage_gems | |
|
4 | ||
|
5 | require 'rake/rdoctask' | |
|
6 | require 'rake/packagetask' | |
|
7 | require 'rake/gempackagetask' | |
|
8 | require 'rake/testtask' | |
|
9 | require 'rake/contrib/rubyforgepublisher' | |
|
10 | ||
|
11 | PKG_NAME = 'acts_as_versioned' | |
|
12 | PKG_VERSION = '0.3.1' | |
|
13 | PKG_FILE_NAME = "#{PKG_NAME}-#{PKG_VERSION}" | |
|
14 | PROD_HOST = "technoweenie@bidwell.textdrive.com" | |
|
15 | RUBY_FORGE_PROJECT = 'ar-versioned' | |
|
16 | RUBY_FORGE_USER = 'technoweenie' | |
|
17 | ||
|
18 | desc 'Default: run unit tests.' | |
|
19 | task :default => :test | |
|
20 | ||
|
21 | desc 'Test the calculations plugin.' | |
|
22 | Rake::TestTask.new(:test) do |t| | |
|
23 | t.libs << 'lib' | |
|
24 | t.pattern = 'test/**/*_test.rb' | |
|
25 | t.verbose = true | |
|
26 | end | |
|
27 | ||
|
28 | desc 'Generate documentation for the calculations plugin.' | |
|
29 | Rake::RDocTask.new(:rdoc) do |rdoc| | |
|
30 | rdoc.rdoc_dir = 'rdoc' | |
|
31 | rdoc.title = "#{PKG_NAME} -- Simple versioning with active record models" | |
|
32 | rdoc.options << '--line-numbers --inline-source' | |
|
33 | rdoc.rdoc_files.include('README', 'CHANGELOG', 'RUNNING_UNIT_TESTS') | |
|
34 | rdoc.rdoc_files.include('lib/**/*.rb') | |
|
35 | end | |
|
36 | ||
|
37 | spec = Gem::Specification.new do |s| | |
|
38 | s.name = PKG_NAME | |
|
39 | s.version = PKG_VERSION | |
|
40 | s.platform = Gem::Platform::RUBY | |
|
41 | s.summary = "Simple versioning with active record models" | |
|
42 | s.files = FileList["{lib,test}/**/*"].to_a + %w(README MIT-LICENSE CHANGELOG RUNNING_UNIT_TESTS) | |
|
43 | s.files.delete "acts_as_versioned_plugin.sqlite.db" | |
|
44 | s.files.delete "acts_as_versioned_plugin.sqlite3.db" | |
|
45 | s.files.delete "test/debug.log" | |
|
46 | s.require_path = 'lib' | |
|
47 | s.autorequire = 'acts_as_versioned' | |
|
48 | s.has_rdoc = true | |
|
49 | s.test_files = Dir['test/**/*_test.rb'] | |
|
50 | s.add_dependency 'activerecord', '>= 1.10.1' | |
|
51 | s.add_dependency 'activesupport', '>= 1.1.1' | |
|
52 | s.author = "Rick Olson" | |
|
53 | s.email = "technoweenie@gmail.com" | |
|
54 | s.homepage = "http://techno-weenie.net" | |
|
55 | end | |
|
56 | ||
|
57 | Rake::GemPackageTask.new(spec) do |pkg| | |
|
58 | pkg.need_tar = true | |
|
59 | end | |
|
60 | ||
|
61 | desc "Publish the API documentation" | |
|
62 | task :pdoc => [:rdoc] do | |
|
63 | Rake::RubyForgePublisher.new(RUBY_FORGE_PROJECT, RUBY_FORGE_USER).upload | |
|
64 | end | |
|
65 | ||
|
66 | desc 'Publish the gem and API docs' | |
|
67 | task :publish => [:pdoc, :rubyforge_upload] | |
|
68 | ||
|
69 | desc "Publish the release files to RubyForge." | |
|
70 | task :rubyforge_upload => :package do | |
|
71 | files = %w(gem tgz).map { |ext| "pkg/#{PKG_FILE_NAME}.#{ext}" } | |
|
72 | ||
|
73 | if RUBY_FORGE_PROJECT then | |
|
74 | require 'net/http' | |
|
75 | require 'open-uri' | |
|
76 | ||
|
77 | project_uri = "http://rubyforge.org/projects/#{RUBY_FORGE_PROJECT}/" | |
|
78 | project_data = open(project_uri) { |data| data.read } | |
|
79 | group_id = project_data[/[?&]group_id=(\d+)/, 1] | |
|
80 | raise "Couldn't get group id" unless group_id | |
|
81 | ||
|
82 | # This echos password to shell which is a bit sucky | |
|
83 | if ENV["RUBY_FORGE_PASSWORD"] | |
|
84 | password = ENV["RUBY_FORGE_PASSWORD"] | |
|
85 | else | |
|
86 | print "#{RUBY_FORGE_USER}@rubyforge.org's password: " | |
|
87 | password = STDIN.gets.chomp | |
|
88 | end | |
|
89 | ||
|
90 | login_response = Net::HTTP.start("rubyforge.org", 80) do |http| | |
|
91 | data = [ | |
|
92 | "login=1", | |
|
93 | "form_loginname=#{RUBY_FORGE_USER}", | |
|
94 | "form_pw=#{password}" | |
|
95 | ].join("&") | |
|
96 | http.post("/account/login.php", data) | |
|
97 | end | |
|
98 | ||
|
99 | cookie = login_response["set-cookie"] | |
|
100 | raise "Login failed" unless cookie | |
|
101 | headers = { "Cookie" => cookie } | |
|
102 | ||
|
103 | release_uri = "http://rubyforge.org/frs/admin/?group_id=#{group_id}" | |
|
104 | release_data = open(release_uri, headers) { |data| data.read } | |
|
105 | package_id = release_data[/[?&]package_id=(\d+)/, 1] | |
|
106 | raise "Couldn't get package id" unless package_id | |
|
107 | ||
|
108 | first_file = true | |
|
109 | release_id = "" | |
|
110 | ||
|
111 | files.each do |filename| | |
|
112 | basename = File.basename(filename) | |
|
113 | file_ext = File.extname(filename) | |
|
114 | file_data = File.open(filename, "rb") { |file| file.read } | |
|
115 | ||
|
116 | puts "Releasing #{basename}..." | |
|
117 | ||
|
118 | release_response = Net::HTTP.start("rubyforge.org", 80) do |http| | |
|
119 | release_date = Time.now.strftime("%Y-%m-%d %H:%M") | |
|
120 | type_map = { | |
|
121 | ".zip" => "3000", | |
|
122 | ".tgz" => "3110", | |
|
123 | ".gz" => "3110", | |
|
124 | ".gem" => "1400" | |
|
125 | }; type_map.default = "9999" | |
|
126 | type = type_map[file_ext] | |
|
127 | boundary = "rubyqMY6QN9bp6e4kS21H4y0zxcvoor" | |
|
128 | ||
|
129 | query_hash = if first_file then | |
|
130 | { | |
|
131 | "group_id" => group_id, | |
|
132 | "package_id" => package_id, | |
|
133 | "release_name" => PKG_FILE_NAME, | |
|
134 | "release_date" => release_date, | |
|
135 | "type_id" => type, | |
|
136 | "processor_id" => "8000", # Any | |
|
137 | "release_notes" => "", | |
|
138 | "release_changes" => "", | |
|
139 | "preformatted" => "1", | |
|
140 | "submit" => "1" | |
|
141 | } | |
|
142 | else | |
|
143 | { | |
|
144 | "group_id" => group_id, | |
|
145 | "release_id" => release_id, | |
|
146 | "package_id" => package_id, | |
|
147 | "step2" => "1", | |
|
148 | "type_id" => type, | |
|
149 | "processor_id" => "8000", # Any | |
|
150 | "submit" => "Add This File" | |
|
151 | } | |
|
152 | end | |
|
153 | ||
|
154 | query = "?" + query_hash.map do |(name, value)| | |
|
155 | [name, URI.encode(value)].join("=") | |
|
156 | end.join("&") | |
|
157 | ||
|
158 | data = [ | |
|
159 | "--" + boundary, | |
|
160 | "Content-Disposition: form-data; name=\"userfile\"; filename=\"#{basename}\"", | |
|
161 | "Content-Type: application/octet-stream", | |
|
162 | "Content-Transfer-Encoding: binary", | |
|
163 | "", file_data, "" | |
|
164 | ].join("\x0D\x0A") | |
|
165 | ||
|
166 | release_headers = headers.merge( | |
|
167 | "Content-Type" => "multipart/form-data; boundary=#{boundary}" | |
|
168 | ) | |
|
169 | ||
|
170 | target = first_file ? "/frs/admin/qrs.php" : "/frs/admin/editrelease.php" | |
|
171 | http.post(target + query, data, release_headers) | |
|
172 | end | |
|
173 | ||
|
174 | if first_file then | |
|
175 | release_id = release_response.body[/release_id=(\d+)/, 1] | |
|
176 | raise("Couldn't get release id") unless release_id | |
|
177 | end | |
|
178 | ||
|
179 | first_file = false | |
|
180 | end | |
|
181 | end | |
|
1 | require 'rubygems' | |
|
2 | ||
|
3 | Gem::manage_gems | |
|
4 | ||
|
5 | require 'rake/rdoctask' | |
|
6 | require 'rake/packagetask' | |
|
7 | require 'rake/gempackagetask' | |
|
8 | require 'rake/testtask' | |
|
9 | require 'rake/contrib/rubyforgepublisher' | |
|
10 | ||
|
11 | PKG_NAME = 'acts_as_versioned' | |
|
12 | PKG_VERSION = '0.3.1' | |
|
13 | PKG_FILE_NAME = "#{PKG_NAME}-#{PKG_VERSION}" | |
|
14 | PROD_HOST = "technoweenie@bidwell.textdrive.com" | |
|
15 | RUBY_FORGE_PROJECT = 'ar-versioned' | |
|
16 | RUBY_FORGE_USER = 'technoweenie' | |
|
17 | ||
|
18 | desc 'Default: run unit tests.' | |
|
19 | task :default => :test | |
|
20 | ||
|
21 | desc 'Test the calculations plugin.' | |
|
22 | Rake::TestTask.new(:test) do |t| | |
|
23 | t.libs << 'lib' | |
|
24 | t.pattern = 'test/**/*_test.rb' | |
|
25 | t.verbose = true | |
|
26 | end | |
|
27 | ||
|
28 | desc 'Generate documentation for the calculations plugin.' | |
|
29 | Rake::RDocTask.new(:rdoc) do |rdoc| | |
|
30 | rdoc.rdoc_dir = 'rdoc' | |
|
31 | rdoc.title = "#{PKG_NAME} -- Simple versioning with active record models" | |
|
32 | rdoc.options << '--line-numbers --inline-source' | |
|
33 | rdoc.rdoc_files.include('README', 'CHANGELOG', 'RUNNING_UNIT_TESTS') | |
|
34 | rdoc.rdoc_files.include('lib/**/*.rb') | |
|
35 | end | |
|
36 | ||
|
37 | spec = Gem::Specification.new do |s| | |
|
38 | s.name = PKG_NAME | |
|
39 | s.version = PKG_VERSION | |
|
40 | s.platform = Gem::Platform::RUBY | |
|
41 | s.summary = "Simple versioning with active record models" | |
|
42 | s.files = FileList["{lib,test}/**/*"].to_a + %w(README MIT-LICENSE CHANGELOG RUNNING_UNIT_TESTS) | |
|
43 | s.files.delete "acts_as_versioned_plugin.sqlite.db" | |
|
44 | s.files.delete "acts_as_versioned_plugin.sqlite3.db" | |
|
45 | s.files.delete "test/debug.log" | |
|
46 | s.require_path = 'lib' | |
|
47 | s.autorequire = 'acts_as_versioned' | |
|
48 | s.has_rdoc = true | |
|
49 | s.test_files = Dir['test/**/*_test.rb'] | |
|
50 | s.add_dependency 'activerecord', '>= 1.10.1' | |
|
51 | s.add_dependency 'activesupport', '>= 1.1.1' | |
|
52 | s.author = "Rick Olson" | |
|
53 | s.email = "technoweenie@gmail.com" | |
|
54 | s.homepage = "http://techno-weenie.net" | |
|
55 | end | |
|
56 | ||
|
57 | Rake::GemPackageTask.new(spec) do |pkg| | |
|
58 | pkg.need_tar = true | |
|
59 | end | |
|
60 | ||
|
61 | desc "Publish the API documentation" | |
|
62 | task :pdoc => [:rdoc] do | |
|
63 | Rake::RubyForgePublisher.new(RUBY_FORGE_PROJECT, RUBY_FORGE_USER).upload | |
|
64 | end | |
|
65 | ||
|
66 | desc 'Publish the gem and API docs' | |
|
67 | task :publish => [:pdoc, :rubyforge_upload] | |
|
68 | ||
|
69 | desc "Publish the release files to RubyForge." | |
|
70 | task :rubyforge_upload => :package do | |
|
71 | files = %w(gem tgz).map { |ext| "pkg/#{PKG_FILE_NAME}.#{ext}" } | |
|
72 | ||
|
73 | if RUBY_FORGE_PROJECT then | |
|
74 | require 'net/http' | |
|
75 | require 'open-uri' | |
|
76 | ||
|
77 | project_uri = "http://rubyforge.org/projects/#{RUBY_FORGE_PROJECT}/" | |
|
78 | project_data = open(project_uri) { |data| data.read } | |
|
79 | group_id = project_data[/[?&]group_id=(\d+)/, 1] | |
|
80 | raise "Couldn't get group id" unless group_id | |
|
81 | ||
|
82 | # This echos password to shell which is a bit sucky | |
|
83 | if ENV["RUBY_FORGE_PASSWORD"] | |
|
84 | password = ENV["RUBY_FORGE_PASSWORD"] | |
|
85 | else | |
|
86 | print "#{RUBY_FORGE_USER}@rubyforge.org's password: " | |
|
87 | password = STDIN.gets.chomp | |
|
88 | end | |
|
89 | ||
|
90 | login_response = Net::HTTP.start("rubyforge.org", 80) do |http| | |
|
91 | data = [ | |
|
92 | "login=1", | |
|
93 | "form_loginname=#{RUBY_FORGE_USER}", | |
|
94 | "form_pw=#{password}" | |
|
95 | ].join("&") | |
|
96 | http.post("/account/login.php", data) | |
|
97 | end | |
|
98 | ||
|
99 | cookie = login_response["set-cookie"] | |
|
100 | raise "Login failed" unless cookie | |
|
101 | headers = { "Cookie" => cookie } | |
|
102 | ||
|
103 | release_uri = "http://rubyforge.org/frs/admin/?group_id=#{group_id}" | |
|
104 | release_data = open(release_uri, headers) { |data| data.read } | |
|
105 | package_id = release_data[/[?&]package_id=(\d+)/, 1] | |
|
106 | raise "Couldn't get package id" unless package_id | |
|
107 | ||
|
108 | first_file = true | |
|
109 | release_id = "" | |
|
110 | ||
|
111 | files.each do |filename| | |
|
112 | basename = File.basename(filename) | |
|
113 | file_ext = File.extname(filename) | |
|
114 | file_data = File.open(filename, "rb") { |file| file.read } | |
|
115 | ||
|
116 | puts "Releasing #{basename}..." | |
|
117 | ||
|
118 | release_response = Net::HTTP.start("rubyforge.org", 80) do |http| | |
|
119 | release_date = Time.now.strftime("%Y-%m-%d %H:%M") | |
|
120 | type_map = { | |
|
121 | ".zip" => "3000", | |
|
122 | ".tgz" => "3110", | |
|
123 | ".gz" => "3110", | |
|
124 | ".gem" => "1400" | |
|
125 | }; type_map.default = "9999" | |
|
126 | type = type_map[file_ext] | |
|
127 | boundary = "rubyqMY6QN9bp6e4kS21H4y0zxcvoor" | |
|
128 | ||
|
129 | query_hash = if first_file then | |
|
130 | { | |
|
131 | "group_id" => group_id, | |
|
132 | "package_id" => package_id, | |
|
133 | "release_name" => PKG_FILE_NAME, | |
|
134 | "release_date" => release_date, | |
|
135 | "type_id" => type, | |
|
136 | "processor_id" => "8000", # Any | |
|
137 | "release_notes" => "", | |
|
138 | "release_changes" => "", | |
|
139 | "preformatted" => "1", | |
|
140 | "submit" => "1" | |
|
141 | } | |
|
142 | else | |
|
143 | { | |
|
144 | "group_id" => group_id, | |
|
145 | "release_id" => release_id, | |
|
146 | "package_id" => package_id, | |
|
147 | "step2" => "1", | |
|
148 | "type_id" => type, | |
|
149 | "processor_id" => "8000", # Any | |
|
150 | "submit" => "Add This File" | |
|
151 | } | |
|
152 | end | |
|
153 | ||
|
154 | query = "?" + query_hash.map do |(name, value)| | |
|
155 | [name, URI.encode(value)].join("=") | |
|
156 | end.join("&") | |
|
157 | ||
|
158 | data = [ | |
|
159 | "--" + boundary, | |
|
160 | "Content-Disposition: form-data; name=\"userfile\"; filename=\"#{basename}\"", | |
|
161 | "Content-Type: application/octet-stream", | |
|
162 | "Content-Transfer-Encoding: binary", | |
|
163 | "", file_data, "" | |
|
164 | ].join("\x0D\x0A") | |
|
165 | ||
|
166 | release_headers = headers.merge( | |
|
167 | "Content-Type" => "multipart/form-data; boundary=#{boundary}" | |
|
168 | ) | |
|
169 | ||
|
170 | target = first_file ? "/frs/admin/qrs.php" : "/frs/admin/editrelease.php" | |
|
171 | http.post(target + query, data, release_headers) | |
|
172 | end | |
|
173 | ||
|
174 | if first_file then | |
|
175 | release_id = release_response.body[/release_id=(\d+)/, 1] | |
|
176 | raise("Couldn't get release id") unless release_id | |
|
177 | end | |
|
178 | ||
|
179 | first_file = false | |
|
180 | end | |
|
181 | end | |
|
182 | 182 | end No newline at end of file |
@@ -1,511 +1,564 | |||
|
1 | 1 | # Copyright (c) 2005 Rick Olson |
|
2 | 2 | # |
|
3 | 3 | # Permission is hereby granted, free of charge, to any person obtaining |
|
4 | 4 | # a copy of this software and associated documentation files (the |
|
5 | 5 | # "Software"), to deal in the Software without restriction, including |
|
6 | 6 | # without limitation the rights to use, copy, modify, merge, publish, |
|
7 | 7 | # distribute, sublicense, and/or sell copies of the Software, and to |
|
8 | 8 | # permit persons to whom the Software is furnished to do so, subject to |
|
9 | 9 | # the following conditions: |
|
10 | 10 | # |
|
11 | 11 | # The above copyright notice and this permission notice shall be |
|
12 | 12 | # included in all copies or substantial portions of the Software. |
|
13 | 13 | # |
|
14 | 14 | # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
|
15 | 15 | # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |
|
16 | 16 | # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND |
|
17 | 17 | # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE |
|
18 | 18 | # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION |
|
19 | 19 | # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION |
|
20 | 20 | # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |
|
21 | 21 | |
|
22 | 22 | module ActiveRecord #:nodoc: |
|
23 | 23 | module Acts #:nodoc: |
|
24 | 24 | # Specify this act if you want to save a copy of the row in a versioned table. This assumes there is a |
|
25 | # versioned table ready and that your model has a version field. This works with optimisic locking if the lock_version | |
|
25 | # versioned table ready and that your model has a version field. This works with optimistic locking if the lock_version | |
|
26 | 26 | # column is present as well. |
|
27 | 27 | # |
|
28 | 28 | # The class for the versioned model is derived the first time it is seen. Therefore, if you change your database schema you have to restart |
|
29 | 29 | # your container for the changes to be reflected. In development mode this usually means restarting WEBrick. |
|
30 | 30 | # |
|
31 | 31 | # class Page < ActiveRecord::Base |
|
32 | 32 | # # assumes pages_versions table |
|
33 | 33 | # acts_as_versioned |
|
34 | 34 | # end |
|
35 | 35 | # |
|
36 | 36 | # Example: |
|
37 | 37 | # |
|
38 | 38 | # page = Page.create(:title => 'hello world!') |
|
39 | 39 | # page.version # => 1 |
|
40 | 40 | # |
|
41 | 41 | # page.title = 'hello world' |
|
42 | 42 | # page.save |
|
43 | 43 | # page.version # => 2 |
|
44 | 44 | # page.versions.size # => 2 |
|
45 | 45 | # |
|
46 | 46 | # page.revert_to(1) # using version number |
|
47 | 47 | # page.title # => 'hello world!' |
|
48 | 48 | # |
|
49 | 49 | # page.revert_to(page.versions.last) # using versioned instance |
|
50 | 50 | # page.title # => 'hello world' |
|
51 | 51 | # |
|
52 | # page.versions.earliest # efficient query to find the first version | |
|
53 | # page.versions.latest # efficient query to find the most recently created version | |
|
54 | # | |
|
55 | # | |
|
56 | # Simple Queries to page between versions | |
|
57 | # | |
|
58 | # page.versions.before(version) | |
|
59 | # page.versions.after(version) | |
|
60 | # | |
|
61 | # Access the previous/next versions from the versioned model itself | |
|
62 | # | |
|
63 | # version = page.versions.latest | |
|
64 | # version.previous # go back one version | |
|
65 | # version.next # go forward one version | |
|
66 | # | |
|
52 | 67 | # See ActiveRecord::Acts::Versioned::ClassMethods#acts_as_versioned for configuration options |
|
53 | 68 | module Versioned |
|
54 |
CALLBACKS = [:set_new_version, :save_version_on_create, :save_version?, :clear_ |
|
|
69 | CALLBACKS = [:set_new_version, :save_version_on_create, :save_version?, :clear_altered_attributes] | |
|
55 | 70 | def self.included(base) # :nodoc: |
|
56 | 71 | base.extend ClassMethods |
|
57 | 72 | end |
|
58 | 73 | |
|
59 | 74 | module ClassMethods |
|
60 | 75 | # == Configuration options |
|
61 | 76 | # |
|
62 | 77 | # * <tt>class_name</tt> - versioned model class name (default: PageVersion in the above example) |
|
63 | 78 | # * <tt>table_name</tt> - versioned model table name (default: page_versions in the above example) |
|
64 | 79 | # * <tt>foreign_key</tt> - foreign key used to relate the versioned model to the original model (default: page_id in the above example) |
|
65 | 80 | # * <tt>inheritance_column</tt> - name of the column to save the model's inheritance_column value for STI. (default: versioned_type) |
|
66 | 81 | # * <tt>version_column</tt> - name of the column in the model that keeps the version number (default: version) |
|
67 | 82 | # * <tt>sequence_name</tt> - name of the custom sequence to be used by the versioned model. |
|
68 | 83 | # * <tt>limit</tt> - number of revisions to keep, defaults to unlimited |
|
69 | 84 | # * <tt>if</tt> - symbol of method to check before saving a new version. If this method returns false, a new version is not saved. |
|
70 | 85 | # For finer control, pass either a Proc or modify Model#version_condition_met? |
|
71 | 86 | # |
|
72 | 87 | # acts_as_versioned :if => Proc.new { |auction| !auction.expired? } |
|
73 | 88 | # |
|
74 | 89 | # or... |
|
75 | 90 | # |
|
76 | 91 | # class Auction |
|
77 | 92 | # def version_condition_met? # totally bypasses the <tt>:if</tt> option |
|
78 | 93 | # !expired? |
|
79 | 94 | # end |
|
80 | 95 | # end |
|
81 | 96 | # |
|
82 | 97 | # * <tt>if_changed</tt> - Simple way of specifying attributes that are required to be changed before saving a model. This takes |
|
83 |
# either a symbol or array of symbols. WARNING - This will attempt to overwrite any attribute setters you may have. |
|
|
98 | # either a symbol or array of symbols. WARNING - This will attempt to overwrite any attribute setters you may have. | |
|
84 | 99 | # Use this instead if you want to write your own attribute setters (and ignore if_changed): |
|
85 | 100 | # |
|
86 | 101 | # def name=(new_name) |
|
87 | 102 | # write_changed_attribute :name, new_name |
|
88 | 103 | # end |
|
89 | 104 | # |
|
90 | 105 | # * <tt>extend</tt> - Lets you specify a module to be mixed in both the original and versioned models. You can also just pass a block |
|
91 | 106 | # to create an anonymous mixin: |
|
92 | 107 | # |
|
93 | 108 | # class Auction |
|
94 | 109 | # acts_as_versioned do |
|
95 | 110 | # def started? |
|
96 | 111 | # !started_at.nil? |
|
97 | 112 | # end |
|
98 | 113 | # end |
|
99 | 114 | # end |
|
100 | 115 | # |
|
101 | 116 | # or... |
|
102 | 117 | # |
|
103 | 118 | # module AuctionExtension |
|
104 | 119 | # def started? |
|
105 | 120 | # !started_at.nil? |
|
106 | 121 | # end |
|
107 | 122 | # end |
|
108 | 123 | # class Auction |
|
109 | 124 | # acts_as_versioned :extend => AuctionExtension |
|
110 | 125 | # end |
|
111 | 126 | # |
|
112 | 127 | # Example code: |
|
113 | 128 | # |
|
114 | 129 | # @auction = Auction.find(1) |
|
115 | 130 | # @auction.started? |
|
116 | 131 | # @auction.versions.first.started? |
|
117 | 132 | # |
|
118 | 133 | # == Database Schema |
|
119 | 134 | # |
|
120 | 135 | # The model that you're versioning needs to have a 'version' attribute. The model is versioned |
|
121 | 136 | # into a table called #{model}_versions where the model name is singlular. The _versions table should |
|
122 | 137 | # contain all the fields you want versioned, the same version column, and a #{model}_id foreign key field. |
|
123 | 138 | # |
|
124 | 139 | # A lock_version field is also accepted if your model uses Optimistic Locking. If your table uses Single Table inheritance, |
|
125 | 140 | # then that field is reflected in the versioned model as 'versioned_type' by default. |
|
126 | 141 | # |
|
127 | 142 | # Acts_as_versioned comes prepared with the ActiveRecord::Acts::Versioned::ActMethods::ClassMethods#create_versioned_table |
|
128 | 143 | # method, perfect for a migration. It will also create the version column if the main model does not already have it. |
|
129 | 144 | # |
|
130 | 145 | # class AddVersions < ActiveRecord::Migration |
|
131 | 146 | # def self.up |
|
132 | 147 | # # create_versioned_table takes the same options hash |
|
133 | 148 | # # that create_table does |
|
134 | 149 | # Post.create_versioned_table |
|
135 | 150 | # end |
|
136 |
# |
|
|
151 | # | |
|
137 | 152 | # def self.down |
|
138 | 153 | # Post.drop_versioned_table |
|
139 | 154 | # end |
|
140 | 155 | # end |
|
141 | 156 | # |
|
142 | 157 | # == Changing What Fields Are Versioned |
|
143 | 158 | # |
|
144 | 159 | # By default, acts_as_versioned will version all but these fields: |
|
145 | 160 | # |
|
146 | 161 | # [self.primary_key, inheritance_column, 'version', 'lock_version', versioned_inheritance_column] |
|
147 | 162 | # |
|
148 | 163 | # You can add or change those by modifying #non_versioned_columns. Note that this takes strings and not symbols. |
|
149 | 164 | # |
|
150 | 165 | # class Post < ActiveRecord::Base |
|
151 | 166 | # acts_as_versioned |
|
152 | 167 | # self.non_versioned_columns << 'comments_count' |
|
153 | 168 | # end |
|
154 | 169 | # |
|
155 | 170 | def acts_as_versioned(options = {}, &extension) |
|
156 | 171 | # don't allow multiple calls |
|
157 | 172 | return if self.included_modules.include?(ActiveRecord::Acts::Versioned::ActMethods) |
|
158 | 173 | |
|
159 | 174 | send :include, ActiveRecord::Acts::Versioned::ActMethods |
|
160 | ||
|
175 | ||
|
161 | 176 | cattr_accessor :versioned_class_name, :versioned_foreign_key, :versioned_table_name, :versioned_inheritance_column, |
|
162 |
:version_column, :max_version_limit, :track_ |
|
|
177 | :version_column, :max_version_limit, :track_altered_attributes, :version_condition, :version_sequence_name, :non_versioned_columns, | |
|
163 | 178 | :version_association_options |
|
164 | ||
|
179 | ||
|
165 | 180 | # legacy |
|
166 | 181 | alias_method :non_versioned_fields, :non_versioned_columns |
|
167 | 182 | alias_method :non_versioned_fields=, :non_versioned_columns= |
|
168 | 183 | |
|
169 | 184 | class << self |
|
170 | 185 | alias_method :non_versioned_fields, :non_versioned_columns |
|
171 | 186 | alias_method :non_versioned_fields=, :non_versioned_columns= |
|
172 | 187 | end |
|
173 | 188 | |
|
174 |
send :attr_accessor, : |
|
|
189 | send :attr_accessor, :altered_attributes | |
|
175 | 190 | |
|
176 | 191 | self.versioned_class_name = options[:class_name] || "Version" |
|
177 | 192 | self.versioned_foreign_key = options[:foreign_key] || self.to_s.foreign_key |
|
178 | 193 | self.versioned_table_name = options[:table_name] || "#{table_name_prefix}#{base_class.name.demodulize.underscore}_versions#{table_name_suffix}" |
|
179 | 194 | self.versioned_inheritance_column = options[:inheritance_column] || "versioned_#{inheritance_column}" |
|
180 | 195 | self.version_column = options[:version_column] || 'version' |
|
181 | 196 | self.version_sequence_name = options[:sequence_name] |
|
182 | 197 | self.max_version_limit = options[:limit].to_i |
|
183 | 198 | self.version_condition = options[:if] || true |
|
184 | 199 | self.non_versioned_columns = [self.primary_key, inheritance_column, 'version', 'lock_version', versioned_inheritance_column] |
|
185 | 200 | self.version_association_options = { |
|
186 | 201 | :class_name => "#{self.to_s}::#{versioned_class_name}", |
|
187 |
:foreign_key => |
|
|
188 | :order => 'version', | |
|
202 | :foreign_key => versioned_foreign_key, | |
|
189 | 203 | :dependent => :delete_all |
|
190 | 204 | }.merge(options[:association_options] || {}) |
|
191 | 205 | |
|
192 | 206 | if block_given? |
|
193 | 207 | extension_module_name = "#{versioned_class_name}Extension" |
|
194 | 208 | silence_warnings do |
|
195 | 209 | self.const_set(extension_module_name, Module.new(&extension)) |
|
196 | 210 | end |
|
197 | ||
|
211 | ||
|
198 | 212 | options[:extend] = self.const_get(extension_module_name) |
|
199 | 213 | end |
|
200 | 214 | |
|
201 | 215 | class_eval do |
|
202 | has_many :versions, version_association_options | |
|
216 | has_many :versions, version_association_options do | |
|
217 | # finds earliest version of this record | |
|
218 | def earliest | |
|
219 | @earliest ||= find(:first, :order => 'version') | |
|
220 | end | |
|
221 | ||
|
222 | # find latest version of this record | |
|
223 | def latest | |
|
224 | @latest ||= find(:first, :order => 'version desc') | |
|
225 | end | |
|
226 | end | |
|
203 | 227 | before_save :set_new_version |
|
204 | 228 | after_create :save_version_on_create |
|
205 | 229 | after_update :save_version |
|
206 | 230 | after_save :clear_old_versions |
|
207 |
after_save :clear_ |
|
|
208 | ||
|
231 | after_save :clear_altered_attributes | |
|
232 | ||
|
209 | 233 | unless options[:if_changed].nil? |
|
210 |
self.track_ |
|
|
234 | self.track_altered_attributes = true | |
|
211 | 235 | options[:if_changed] = [options[:if_changed]] unless options[:if_changed].is_a?(Array) |
|
212 | 236 | options[:if_changed].each do |attr_name| |
|
213 | 237 | define_method("#{attr_name}=") do |value| |
|
214 | 238 | write_changed_attribute attr_name, value |
|
215 | 239 | end |
|
216 | 240 | end |
|
217 | 241 | end |
|
218 | ||
|
242 | ||
|
219 | 243 | include options[:extend] if options[:extend].is_a?(Module) |
|
220 | 244 | end |
|
221 | 245 | |
|
222 | 246 | # create the dynamic versioned model |
|
223 | 247 | const_set(versioned_class_name, Class.new(ActiveRecord::Base)).class_eval do |
|
224 | 248 | def self.reloadable? ; false ; end |
|
249 | # find first version before the given version | |
|
250 | def self.before(version) | |
|
251 | find :first, :order => 'version desc', | |
|
252 | :conditions => ["#{original_class.versioned_foreign_key} = ? and version < ?", version.send(original_class.versioned_foreign_key), version.version] | |
|
253 | end | |
|
254 | ||
|
255 | # find first version after the given version. | |
|
256 | def self.after(version) | |
|
257 | find :first, :order => 'version', | |
|
258 | :conditions => ["#{original_class.versioned_foreign_key} = ? and version > ?", version.send(original_class.versioned_foreign_key), version.version] | |
|
259 | end | |
|
260 | ||
|
261 | def previous | |
|
262 | self.class.before(self) | |
|
263 | end | |
|
264 | ||
|
265 | def next | |
|
266 | self.class.after(self) | |
|
267 | end | |
|
268 | ||
|
269 | def versions_count | |
|
270 | page.version | |
|
271 | end | |
|
225 | 272 | end |
|
226 | ||
|
273 | ||
|
274 | versioned_class.cattr_accessor :original_class | |
|
275 | versioned_class.original_class = self | |
|
227 | 276 | versioned_class.set_table_name versioned_table_name |
|
228 | 277 | versioned_class.belongs_to self.to_s.demodulize.underscore.to_sym, |
|
229 | 278 | :class_name => "::#{self.to_s}", |
|
230 | 279 | :foreign_key => versioned_foreign_key |
|
231 | 280 | versioned_class.send :include, options[:extend] if options[:extend].is_a?(Module) |
|
232 | 281 | versioned_class.set_sequence_name version_sequence_name if version_sequence_name |
|
233 | 282 | end |
|
234 | 283 | end |
|
235 | ||
|
284 | ||
|
236 | 285 | module ActMethods |
|
237 | 286 | def self.included(base) # :nodoc: |
|
238 | 287 | base.extend ClassMethods |
|
239 | 288 | end |
|
240 | ||
|
289 | ||
|
290 | # Finds a specific version of this record | |
|
291 | def find_version(version = nil) | |
|
292 | self.class.find_version(id, version) | |
|
293 | end | |
|
294 | ||
|
241 | 295 | # Saves a version of the model if applicable |
|
242 | 296 | def save_version |
|
243 | 297 | save_version_on_create if save_version? |
|
244 | 298 | end |
|
245 | ||
|
299 | ||
|
246 | 300 | # Saves a version of the model in the versioned table. This is called in the after_save callback by default |
|
247 | 301 | def save_version_on_create |
|
248 | 302 | rev = self.class.versioned_class.new |
|
249 | 303 | self.clone_versioned_model(self, rev) |
|
250 | 304 | rev.version = send(self.class.version_column) |
|
251 | 305 | rev.send("#{self.class.versioned_foreign_key}=", self.id) |
|
252 | 306 | rev.save |
|
253 | 307 | end |
|
254 | 308 | |
|
255 | 309 | # Clears old revisions if a limit is set with the :limit option in <tt>acts_as_versioned</tt>. |
|
256 | 310 | # Override this method to set your own criteria for clearing old versions. |
|
257 | 311 | def clear_old_versions |
|
258 | 312 | return if self.class.max_version_limit == 0 |
|
259 | 313 | excess_baggage = send(self.class.version_column).to_i - self.class.max_version_limit |
|
260 | 314 | if excess_baggage > 0 |
|
261 | 315 | sql = "DELETE FROM #{self.class.versioned_table_name} WHERE version <= #{excess_baggage} AND #{self.class.versioned_foreign_key} = #{self.id}" |
|
262 | 316 | self.class.versioned_class.connection.execute sql |
|
263 | 317 | end |
|
264 | 318 | end |
|
265 | 319 | |
|
266 | # Finds a specific version of this model. | |
|
267 |
|
|
|
268 | return version if version.is_a?(self.class.versioned_class) | |
|
269 | return nil if version.is_a?(ActiveRecord::Base) | |
|
270 | find_versions(:conditions => ['version = ?', version], :limit => 1).first | |
|
271 | end | |
|
272 | ||
|
273 | # Finds versions of this model. Takes an options hash like <tt>find</tt> | |
|
274 | def find_versions(options = {}) | |
|
275 | versions.find(:all, options) | |
|
320 | def versions_count | |
|
321 | version | |
|
276 | 322 | end |
|
277 | 323 | |
|
278 | 324 | # Reverts a model to a given version. Takes either a version number or an instance of the versioned model |
|
279 | 325 | def revert_to(version) |
|
280 | 326 | if version.is_a?(self.class.versioned_class) |
|
281 | 327 | return false unless version.send(self.class.versioned_foreign_key) == self.id and !version.new_record? |
|
282 | 328 | else |
|
283 | return false unless version = find_version(version) | |
|
329 | return false unless version = versions.find_by_version(version) | |
|
284 | 330 | end |
|
285 | 331 | self.clone_versioned_model(version, self) |
|
286 | 332 | self.send("#{self.class.version_column}=", version.version) |
|
287 | 333 | true |
|
288 | 334 | end |
|
289 | 335 | |
|
290 |
# Reverts a model to a given version and saves the model. |
|
|
336 | # Reverts a model to a given version and saves the model. | |
|
291 | 337 | # Takes either a version number or an instance of the versioned model |
|
292 | 338 | def revert_to!(version) |
|
293 | 339 | revert_to(version) ? save_without_revision : false |
|
294 | 340 | end |
|
295 | 341 | |
|
296 | 342 | # Temporarily turns off Optimistic Locking while saving. Used when reverting so that a new version is not created. |
|
297 | 343 | def save_without_revision |
|
298 | 344 | save_without_revision! |
|
299 | 345 | true |
|
300 | 346 | rescue |
|
301 | 347 | false |
|
302 | 348 | end |
|
303 | 349 | |
|
304 | 350 | def save_without_revision! |
|
305 | 351 | without_locking do |
|
306 | 352 | without_revision do |
|
307 | 353 | save! |
|
308 | 354 | end |
|
309 | 355 | end |
|
310 | 356 | end |
|
311 | 357 | |
|
312 | 358 | # Returns an array of attribute keys that are versioned. See non_versioned_columns |
|
313 | 359 | def versioned_attributes |
|
314 | 360 | self.attributes.keys.select { |k| !self.class.non_versioned_columns.include?(k) } |
|
315 | 361 | end |
|
316 | ||
|
362 | ||
|
317 | 363 | # If called with no parameters, gets whether the current model has changed and needs to be versioned. |
|
318 | 364 | # If called with a single parameter, gets whether the parameter has changed. |
|
319 | 365 | def changed?(attr_name = nil) |
|
320 | 366 | attr_name.nil? ? |
|
321 |
(!self.class.track_ |
|
|
322 |
( |
|
|
367 | (!self.class.track_altered_attributes || (altered_attributes && altered_attributes.length > 0)) : | |
|
368 | (altered_attributes && altered_attributes.include?(attr_name.to_s)) | |
|
323 | 369 | end |
|
324 | ||
|
370 | ||
|
325 | 371 | # keep old dirty? method |
|
326 | 372 | alias_method :dirty?, :changed? |
|
327 | ||
|
373 | ||
|
328 | 374 | # Clones a model. Used when saving a new version or reverting a model's version. |
|
329 | 375 | def clone_versioned_model(orig_model, new_model) |
|
330 | 376 | self.versioned_attributes.each do |key| |
|
331 |
new_model.send("#{key}=", orig_model. |
|
|
377 | new_model.send("#{key}=", orig_model.send(key)) if orig_model.has_attribute?(key) | |
|
332 | 378 | end |
|
333 | ||
|
379 | ||
|
334 | 380 | if orig_model.is_a?(self.class.versioned_class) |
|
335 | 381 | new_model[new_model.class.inheritance_column] = orig_model[self.class.versioned_inheritance_column] |
|
336 | 382 | elsif new_model.is_a?(self.class.versioned_class) |
|
337 | 383 | new_model[self.class.versioned_inheritance_column] = orig_model[orig_model.class.inheritance_column] |
|
338 | 384 | end |
|
339 | 385 | end |
|
340 | ||
|
386 | ||
|
341 | 387 | # Checks whether a new version shall be saved or not. Calls <tt>version_condition_met?</tt> and <tt>changed?</tt>. |
|
342 | 388 | def save_version? |
|
343 | 389 | version_condition_met? && changed? |
|
344 | 390 | end |
|
345 | ||
|
391 | ||
|
346 | 392 | # Checks condition set in the :if option to check whether a revision should be created or not. Override this for |
|
347 | 393 | # custom version condition checking. |
|
348 | 394 | def version_condition_met? |
|
349 | 395 | case |
|
350 | 396 | when version_condition.is_a?(Symbol) |
|
351 | 397 | send(version_condition) |
|
352 | 398 | when version_condition.respond_to?(:call) && (version_condition.arity == 1 || version_condition.arity == -1) |
|
353 | 399 | version_condition.call(self) |
|
354 | 400 | else |
|
355 | 401 | version_condition |
|
356 |
end |
|
|
402 | end | |
|
357 | 403 | end |
|
358 | 404 | |
|
359 | 405 | # Executes the block with the versioning callbacks disabled. |
|
360 | 406 | # |
|
361 | 407 | # @foo.without_revision do |
|
362 | 408 | # @foo.save |
|
363 | 409 | # end |
|
364 | 410 | # |
|
365 | 411 | def without_revision(&block) |
|
366 | 412 | self.class.without_revision(&block) |
|
367 | 413 | end |
|
368 | 414 | |
|
369 | 415 | # Turns off optimistic locking for the duration of the block |
|
370 | 416 | # |
|
371 | 417 | # @foo.without_locking do |
|
372 | 418 | # @foo.save |
|
373 | 419 | # end |
|
374 | 420 | # |
|
375 | 421 | def without_locking(&block) |
|
376 | 422 | self.class.without_locking(&block) |
|
377 | 423 | end |
|
378 | 424 | |
|
379 | 425 | def empty_callback() end #:nodoc: |
|
380 | 426 | |
|
381 |
protected |
|
|
427 | protected | |
|
382 | 428 | # sets the new version before saving, unless you're using optimistic locking. In that case, let it take care of the version. |
|
383 | 429 | def set_new_version |
|
384 | 430 | self.send("#{self.class.version_column}=", self.next_version) if new_record? || (!locking_enabled? && save_version?) |
|
385 | 431 | end |
|
386 | ||
|
432 | ||
|
387 | 433 | # Gets the next available version for the current record, or 1 for a new record |
|
388 | 434 | def next_version |
|
389 | 435 | return 1 if new_record? |
|
390 | 436 | (versions.calculate(:max, :version) || 0) + 1 |
|
391 | 437 | end |
|
392 | ||
|
438 | ||
|
393 | 439 | # clears current changed attributes. Called after save. |
|
394 |
def clear_ |
|
|
395 |
self. |
|
|
440 | def clear_altered_attributes | |
|
441 | self.altered_attributes = [] | |
|
396 | 442 | end |
|
397 | 443 | |
|
398 | 444 | def write_changed_attribute(attr_name, attr_value) |
|
399 | 445 | # Convert to db type for comparison. Avoids failing Float<=>String comparisons. |
|
400 | 446 | attr_value_for_db = self.class.columns_hash[attr_name.to_s].type_cast(attr_value) |
|
401 |
(self. |
|
|
447 | (self.altered_attributes ||= []) << attr_name.to_s unless self.changed?(attr_name) || self.send(attr_name) == attr_value_for_db | |
|
402 | 448 | write_attribute(attr_name, attr_value_for_db) |
|
403 | 449 | end |
|
404 | 450 | |
|
405 | private | |
|
406 | CALLBACKS.each do |attr_name| | |
|
407 | alias_method "orig_#{attr_name}".to_sym, attr_name | |
|
408 | end | |
|
409 | ||
|
410 | 451 | module ClassMethods |
|
411 | 452 | # Finds a specific version of a specific row of this model |
|
412 | def find_version(id, version) | |
|
413 |
|
|
|
414 | :conditions => ["#{versioned_foreign_key} = ? AND version = ?", id, version], | |
|
415 | :limit => 1).first | |
|
453 | def find_version(id, version = nil) | |
|
454 | return find(id) unless version | |
|
455 | ||
|
456 | conditions = ["#{versioned_foreign_key} = ? AND version = ?", id, version] | |
|
457 | options = { :conditions => conditions, :limit => 1 } | |
|
458 | ||
|
459 | if result = find_versions(id, options).first | |
|
460 | result | |
|
461 | else | |
|
462 | raise RecordNotFound, "Couldn't find #{name} with ID=#{id} and VERSION=#{version}" | |
|
463 | end | |
|
416 | 464 | end |
|
417 | ||
|
465 | ||
|
418 | 466 | # Finds versions of a specific model. Takes an options hash like <tt>find</tt> |
|
419 | 467 | def find_versions(id, options = {}) |
|
420 | 468 | versioned_class.find :all, { |
|
421 | 469 | :conditions => ["#{versioned_foreign_key} = ?", id], |
|
422 | 470 | :order => 'version' }.merge(options) |
|
423 | 471 | end |
|
424 | 472 | |
|
425 | 473 | # Returns an array of columns that are versioned. See non_versioned_columns |
|
426 | 474 | def versioned_columns |
|
427 | 475 | self.columns.select { |c| !non_versioned_columns.include?(c.name) } |
|
428 | 476 | end |
|
429 | ||
|
477 | ||
|
430 | 478 | # Returns an instance of the dynamic versioned model |
|
431 | 479 | def versioned_class |
|
432 | 480 | const_get versioned_class_name |
|
433 | 481 | end |
|
434 | 482 | |
|
435 | 483 | # Rake migration task to create the versioned table using options passed to acts_as_versioned |
|
436 | 484 | def create_versioned_table(create_table_options = {}) |
|
437 | 485 | # create version column in main table if it does not exist |
|
438 | 486 | if !self.content_columns.find { |c| %w(version lock_version).include? c.name } |
|
439 | 487 | self.connection.add_column table_name, :version, :integer |
|
440 | 488 | end |
|
441 | ||
|
489 | ||
|
442 | 490 | self.connection.create_table(versioned_table_name, create_table_options) do |t| |
|
443 | 491 | t.column versioned_foreign_key, :integer |
|
444 | 492 | t.column :version, :integer |
|
445 | 493 | end |
|
446 | ||
|
494 | ||
|
447 | 495 | updated_col = nil |
|
448 | 496 | self.versioned_columns.each do |col| |
|
449 | 497 | updated_col = col if !updated_col && %(updated_at updated_on).include?(col.name) |
|
450 | 498 | self.connection.add_column versioned_table_name, col.name, col.type, |
|
451 | 499 | :limit => col.limit, |
|
452 | :default => col.default | |
|
500 | :default => col.default, | |
|
501 | :scale => col.scale, | |
|
502 | :precision => col.precision | |
|
453 | 503 | end |
|
454 | ||
|
504 | ||
|
455 | 505 | if type_col = self.columns_hash[inheritance_column] |
|
456 | 506 | self.connection.add_column versioned_table_name, versioned_inheritance_column, type_col.type, |
|
457 | 507 | :limit => type_col.limit, |
|
458 | :default => type_col.default | |
|
508 | :default => type_col.default, | |
|
509 | :scale => type_col.scale, | |
|
510 | :precision => type_col.precision | |
|
459 | 511 | end |
|
460 | ||
|
512 | ||
|
461 | 513 | if updated_col.nil? |
|
462 | 514 | self.connection.add_column versioned_table_name, :updated_at, :timestamp |
|
463 | 515 | end |
|
464 | 516 | end |
|
465 | ||
|
517 | ||
|
466 | 518 | # Rake migration task to drop the versioned table |
|
467 | 519 | def drop_versioned_table |
|
468 | 520 | self.connection.drop_table versioned_table_name |
|
469 | 521 | end |
|
470 | ||
|
522 | ||
|
471 | 523 | # Executes the block with the versioning callbacks disabled. |
|
472 | 524 | # |
|
473 | 525 | # Foo.without_revision do |
|
474 | 526 | # @foo.save |
|
475 | 527 | # end |
|
476 | 528 | # |
|
477 | 529 | def without_revision(&block) |
|
478 | 530 | class_eval do |
|
479 |
CALLBACKS.each do |attr_name| |
|
|
531 | CALLBACKS.each do |attr_name| | |
|
532 | alias_method "orig_#{attr_name}".to_sym, attr_name | |
|
480 | 533 | alias_method attr_name, :empty_callback |
|
481 | 534 | end |
|
482 | 535 | end |
|
483 |
|
|
|
536 | block.call | |
|
537 | ensure | |
|
484 | 538 | class_eval do |
|
485 | 539 | CALLBACKS.each do |attr_name| |
|
486 | 540 | alias_method attr_name, "orig_#{attr_name}".to_sym |
|
487 | 541 | end |
|
488 | 542 | end |
|
489 | result | |
|
490 | 543 | end |
|
491 | 544 | |
|
492 | 545 | # Turns off optimistic locking for the duration of the block |
|
493 | 546 | # |
|
494 | 547 | # Foo.without_locking do |
|
495 | 548 | # @foo.save |
|
496 | 549 | # end |
|
497 | 550 | # |
|
498 | 551 | def without_locking(&block) |
|
499 | 552 | current = ActiveRecord::Base.lock_optimistically |
|
500 | 553 | ActiveRecord::Base.lock_optimistically = false if current |
|
501 | 554 | result = block.call |
|
502 | 555 | ActiveRecord::Base.lock_optimistically = true if current |
|
503 | 556 | result |
|
504 |
end |
|
|
557 | end | |
|
505 | 558 | end |
|
506 | 559 | end |
|
507 | 560 | end |
|
508 | 561 | end |
|
509 | 562 | end |
|
510 | 563 | |
|
511 | 564 | ActiveRecord::Base.send :include, ActiveRecord::Acts::Versioned No newline at end of file |
@@ -1,40 +1,41 | |||
|
1 | $:.unshift(File.dirname(__FILE__) + '/../../../rails/activesupport/lib') | |
|
2 | $:.unshift(File.dirname(__FILE__) + '/../../../rails/activerecord/lib') | |
|
1 | 3 | $:.unshift(File.dirname(__FILE__) + '/../lib') |
|
2 | ||
|
3 | 4 | require 'test/unit' |
|
4 | require File.expand_path(File.join(File.dirname(__FILE__), '../../../../config/environment.rb')) | |
|
5 |
require 'active_ |
|
|
5 | begin | |
|
6 | require 'active_support' | |
|
7 | require 'active_record' | |
|
8 | require 'active_record/fixtures' | |
|
9 | rescue LoadError | |
|
10 | require 'rubygems' | |
|
11 | retry | |
|
12 | end | |
|
13 | require 'acts_as_versioned' | |
|
6 | 14 | |
|
7 | 15 | config = YAML::load(IO.read(File.dirname(__FILE__) + '/database.yml')) |
|
8 | 16 | ActiveRecord::Base.logger = Logger.new(File.dirname(__FILE__) + "/debug.log") |
|
9 |
ActiveRecord::Base. |
|
|
17 | ActiveRecord::Base.configurations = {'test' => config[ENV['DB'] || 'sqlite3']} | |
|
18 | ActiveRecord::Base.establish_connection(ActiveRecord::Base.configurations['test']) | |
|
10 | 19 | |
|
11 | 20 | load(File.dirname(__FILE__) + "/schema.rb") |
|
12 | 21 | |
|
13 | 22 | # set up custom sequence on widget_versions for DBs that support sequences |
|
14 | 23 | if ENV['DB'] == 'postgresql' |
|
15 | 24 | ActiveRecord::Base.connection.execute "DROP SEQUENCE widgets_seq;" rescue nil |
|
16 | 25 | ActiveRecord::Base.connection.remove_column :widget_versions, :id |
|
17 | 26 | ActiveRecord::Base.connection.execute "CREATE SEQUENCE widgets_seq START 101;" |
|
18 | 27 | ActiveRecord::Base.connection.execute "ALTER TABLE widget_versions ADD COLUMN id INTEGER PRIMARY KEY DEFAULT nextval('widgets_seq');" |
|
19 | 28 | end |
|
20 | 29 | |
|
21 | 30 | Test::Unit::TestCase.fixture_path = File.dirname(__FILE__) + "/fixtures/" |
|
22 |
$ |
|
|
31 | $:.unshift(Test::Unit::TestCase.fixture_path) | |
|
23 | 32 | |
|
24 | 33 | class Test::Unit::TestCase #:nodoc: |
|
25 | def create_fixtures(*table_names) | |
|
26 | if block_given? | |
|
27 | Fixtures.create_fixtures(Test::Unit::TestCase.fixture_path, table_names) { yield } | |
|
28 | else | |
|
29 | Fixtures.create_fixtures(Test::Unit::TestCase.fixture_path, table_names) | |
|
30 | end | |
|
31 | end | |
|
32 | ||
|
33 | 34 | # Turn off transactional fixtures if you're working with MyISAM tables in MySQL |
|
34 | 35 | self.use_transactional_fixtures = true |
|
35 | 36 | |
|
36 | 37 | # Instantiated fixtures are slow, but give you @david where you otherwise would need people(:david) |
|
37 | 38 | self.use_instantiated_fixtures = false |
|
38 | 39 | |
|
39 | 40 | # Add more helper methods to be used by all tests here... |
|
40 | 41 | end No newline at end of file |
@@ -1,6 +1,6 | |||
|
1 | 1 | class Widget < ActiveRecord::Base |
|
2 | 2 | acts_as_versioned :sequence_name => 'widgets_seq', :association_options => { |
|
3 |
:dependent => |
|
|
3 | :dependent => :nullify, :order => 'version desc' | |
|
4 | 4 | } |
|
5 | 5 | non_versioned_columns << 'foo' |
|
6 | 6 | end No newline at end of file |
@@ -1,32 +1,46 | |||
|
1 | 1 | require File.join(File.dirname(__FILE__), 'abstract_unit') |
|
2 | 2 | |
|
3 | 3 | if ActiveRecord::Base.connection.supports_migrations? |
|
4 | 4 | class Thing < ActiveRecord::Base |
|
5 | 5 | attr_accessor :version |
|
6 | 6 | acts_as_versioned |
|
7 | 7 | end |
|
8 | 8 | |
|
9 | 9 | class MigrationTest < Test::Unit::TestCase |
|
10 | 10 | self.use_transactional_fixtures = false |
|
11 | 11 | def teardown |
|
12 | ActiveRecord::Base.connection.initialize_schema_information | |
|
13 |
ActiveRecord::Base.connection. |
|
|
14 | ||
|
12 | if ActiveRecord::Base.connection.respond_to?(:initialize_schema_information) | |
|
13 | ActiveRecord::Base.connection.initialize_schema_information | |
|
14 | ActiveRecord::Base.connection.update "UPDATE schema_info SET version = 0" | |
|
15 | else | |
|
16 | ActiveRecord::Base.connection.initialize_schema_migrations_table | |
|
17 | ActiveRecord::Base.connection.assume_migrated_upto_version(0) | |
|
18 | end | |
|
19 | ||
|
15 | 20 | Thing.connection.drop_table "things" rescue nil |
|
16 | 21 | Thing.connection.drop_table "thing_versions" rescue nil |
|
17 | 22 | Thing.reset_column_information |
|
18 | 23 | end |
|
19 | 24 | |
|
20 | 25 | def test_versioned_migration |
|
21 | 26 | assert_raises(ActiveRecord::StatementInvalid) { Thing.create :title => 'blah blah' } |
|
22 | 27 | # take 'er up |
|
23 | 28 | ActiveRecord::Migrator.up(File.dirname(__FILE__) + '/fixtures/migrations/') |
|
24 | t = Thing.create :title => 'blah blah' | |
|
29 | t = Thing.create :title => 'blah blah', :price => 123.45, :type => 'Thing' | |
|
25 | 30 | assert_equal 1, t.versions.size |
|
31 | ||
|
32 | # check that the price column has remembered its value correctly | |
|
33 | assert_equal t.price, t.versions.first.price | |
|
34 | assert_equal t.title, t.versions.first.title | |
|
35 | assert_equal t[:type], t.versions.first[:type] | |
|
36 | ||
|
37 | # make sure that the precision of the price column has been preserved | |
|
38 | assert_equal 7, Thing::Version.columns.find{|c| c.name == "price"}.precision | |
|
39 | assert_equal 2, Thing::Version.columns.find{|c| c.name == "price"}.scale | |
|
26 | 40 | |
|
27 | 41 | # now lets take 'er back down |
|
28 | 42 | ActiveRecord::Migrator.down(File.dirname(__FILE__) + '/fixtures/migrations/') |
|
29 | 43 | assert_raises(ActiveRecord::StatementInvalid) { Thing.create :title => 'blah blah' } |
|
30 | 44 | end |
|
31 | 45 | end |
|
32 | 46 | end |
@@ -1,313 +1,347 | |||
|
1 | 1 | require File.join(File.dirname(__FILE__), 'abstract_unit') |
|
2 | 2 | require File.join(File.dirname(__FILE__), 'fixtures/page') |
|
3 | 3 | require File.join(File.dirname(__FILE__), 'fixtures/widget') |
|
4 | 4 | |
|
5 | 5 | class VersionedTest < Test::Unit::TestCase |
|
6 | 6 | fixtures :pages, :page_versions, :locked_pages, :locked_pages_revisions, :authors, :landmarks, :landmark_versions |
|
7 | set_fixture_class :page_versions => Page::Version | |
|
7 | 8 | |
|
8 | 9 | def test_saves_versioned_copy |
|
9 | p = Page.create :title => 'first title', :body => 'first body' | |
|
10 | p = Page.create! :title => 'first title', :body => 'first body' | |
|
10 | 11 | assert !p.new_record? |
|
11 | 12 | assert_equal 1, p.versions.size |
|
12 | 13 | assert_equal 1, p.version |
|
13 | 14 | assert_instance_of Page.versioned_class, p.versions.first |
|
14 | 15 | end |
|
15 | 16 | |
|
16 | 17 | def test_saves_without_revision |
|
17 | 18 | p = pages(:welcome) |
|
18 | 19 | old_versions = p.versions.count |
|
19 | ||
|
20 | ||
|
20 | 21 | p.save_without_revision |
|
21 | ||
|
22 | ||
|
22 | 23 | p.without_revision do |
|
23 | 24 | p.update_attributes :title => 'changed' |
|
24 | 25 | end |
|
25 | ||
|
26 | ||
|
26 | 27 | assert_equal old_versions, p.versions.count |
|
27 | 28 | end |
|
28 | 29 | |
|
29 | 30 | def test_rollback_with_version_number |
|
30 | 31 | p = pages(:welcome) |
|
31 | 32 | assert_equal 24, p.version |
|
32 | 33 | assert_equal 'Welcome to the weblog', p.title |
|
33 | ||
|
34 | ||
|
34 | 35 | assert p.revert_to!(p.versions.first.version), "Couldn't revert to 23" |
|
35 | 36 | assert_equal 23, p.version |
|
36 | 37 | assert_equal 'Welcome to the weblg', p.title |
|
37 | 38 | end |
|
38 | 39 | |
|
39 | 40 | def test_versioned_class_name |
|
40 | 41 | assert_equal 'Version', Page.versioned_class_name |
|
41 | 42 | assert_equal 'LockedPageRevision', LockedPage.versioned_class_name |
|
42 | 43 | end |
|
43 | 44 | |
|
44 | 45 | def test_versioned_class |
|
45 | 46 | assert_equal Page::Version, Page.versioned_class |
|
46 | 47 | assert_equal LockedPage::LockedPageRevision, LockedPage.versioned_class |
|
47 | 48 | end |
|
48 | 49 | |
|
49 | 50 | def test_special_methods |
|
50 | 51 | assert_nothing_raised { pages(:welcome).feeling_good? } |
|
51 | 52 | assert_nothing_raised { pages(:welcome).versions.first.feeling_good? } |
|
52 | 53 | assert_nothing_raised { locked_pages(:welcome).hello_world } |
|
53 | 54 | assert_nothing_raised { locked_pages(:welcome).versions.first.hello_world } |
|
54 | 55 | end |
|
55 | 56 | |
|
56 | 57 | def test_rollback_with_version_class |
|
57 | 58 | p = pages(:welcome) |
|
58 | 59 | assert_equal 24, p.version |
|
59 | 60 | assert_equal 'Welcome to the weblog', p.title |
|
60 | ||
|
61 | ||
|
61 | 62 | assert p.revert_to!(p.versions.first), "Couldn't revert to 23" |
|
62 | 63 | assert_equal 23, p.version |
|
63 | 64 | assert_equal 'Welcome to the weblg', p.title |
|
64 | 65 | end |
|
65 | ||
|
66 | ||
|
66 | 67 | def test_rollback_fails_with_invalid_revision |
|
67 | 68 | p = locked_pages(:welcome) |
|
68 | 69 | assert !p.revert_to!(locked_pages(:thinking)) |
|
69 | 70 | end |
|
70 | 71 | |
|
71 | 72 | def test_saves_versioned_copy_with_options |
|
72 | p = LockedPage.create :title => 'first title' | |
|
73 | p = LockedPage.create! :title => 'first title' | |
|
73 | 74 | assert !p.new_record? |
|
74 | 75 | assert_equal 1, p.versions.size |
|
75 | 76 | assert_instance_of LockedPage.versioned_class, p.versions.first |
|
76 | 77 | end |
|
77 | ||
|
78 | ||
|
78 | 79 | def test_rollback_with_version_number_with_options |
|
79 | 80 | p = locked_pages(:welcome) |
|
80 | 81 | assert_equal 'Welcome to the weblog', p.title |
|
81 | 82 | assert_equal 'LockedPage', p.versions.first.version_type |
|
82 | ||
|
83 | ||
|
83 | 84 | assert p.revert_to!(p.versions.first.version), "Couldn't revert to 23" |
|
84 | 85 | assert_equal 'Welcome to the weblg', p.title |
|
85 | 86 | assert_equal 'LockedPage', p.versions.first.version_type |
|
86 | 87 | end |
|
87 | ||
|
88 | ||
|
88 | 89 | def test_rollback_with_version_class_with_options |
|
89 | 90 | p = locked_pages(:welcome) |
|
90 | 91 | assert_equal 'Welcome to the weblog', p.title |
|
91 | 92 | assert_equal 'LockedPage', p.versions.first.version_type |
|
92 | ||
|
93 | ||
|
93 | 94 | assert p.revert_to!(p.versions.first), "Couldn't revert to 1" |
|
94 | 95 | assert_equal 'Welcome to the weblg', p.title |
|
95 | 96 | assert_equal 'LockedPage', p.versions.first.version_type |
|
96 | 97 | end |
|
97 | ||
|
98 | ||
|
98 | 99 | def test_saves_versioned_copy_with_sti |
|
99 | p = SpecialLockedPage.create :title => 'first title' | |
|
100 | p = SpecialLockedPage.create! :title => 'first title' | |
|
100 | 101 | assert !p.new_record? |
|
101 | 102 | assert_equal 1, p.versions.size |
|
102 | 103 | assert_instance_of LockedPage.versioned_class, p.versions.first |
|
103 | 104 | assert_equal 'SpecialLockedPage', p.versions.first.version_type |
|
104 | 105 | end |
|
105 | ||
|
106 | ||
|
106 | 107 | def test_rollback_with_version_number_with_sti |
|
107 | 108 | p = locked_pages(:thinking) |
|
108 | 109 | assert_equal 'So I was thinking', p.title |
|
109 | ||
|
110 | ||
|
110 | 111 | assert p.revert_to!(p.versions.first.version), "Couldn't revert to 1" |
|
111 | 112 | assert_equal 'So I was thinking!!!', p.title |
|
112 | 113 | assert_equal 'SpecialLockedPage', p.versions.first.version_type |
|
113 | 114 | end |
|
114 | 115 | |
|
115 | 116 | def test_lock_version_works_with_versioning |
|
116 | 117 | p = locked_pages(:thinking) |
|
117 | 118 | p2 = LockedPage.find(p.id) |
|
118 | ||
|
119 | ||
|
119 | 120 | p.title = 'fresh title' |
|
120 | 121 | p.save |
|
121 | 122 | assert_equal 2, p.versions.size # limit! |
|
122 | ||
|
123 | ||
|
123 | 124 | assert_raises(ActiveRecord::StaleObjectError) do |
|
124 | 125 | p2.title = 'stale title' |
|
125 | 126 | p2.save |
|
126 | 127 | end |
|
127 | 128 | end |
|
128 | 129 | |
|
129 | 130 | def test_version_if_condition |
|
130 | p = Page.create :title => "title" | |
|
131 | p = Page.create! :title => "title" | |
|
131 | 132 | assert_equal 1, p.version |
|
132 | ||
|
133 | ||
|
133 | 134 | Page.feeling_good = false |
|
134 | 135 | p.save |
|
135 | 136 | assert_equal 1, p.version |
|
136 | 137 | Page.feeling_good = true |
|
137 | 138 | end |
|
138 | ||
|
139 | ||
|
139 | 140 | def test_version_if_condition2 |
|
140 | 141 | # set new if condition |
|
141 | 142 | Page.class_eval do |
|
142 | 143 | def new_feeling_good() title[0..0] == 'a'; end |
|
143 | 144 | alias_method :old_feeling_good, :feeling_good? |
|
144 | 145 | alias_method :feeling_good?, :new_feeling_good |
|
145 | 146 | end |
|
146 | ||
|
147 | p = Page.create :title => "title" | |
|
147 | ||
|
148 | p = Page.create! :title => "title" | |
|
148 | 149 | assert_equal 1, p.version # version does not increment |
|
149 | 150 | assert_equal 1, p.versions(true).size |
|
150 | ||
|
151 | ||
|
151 | 152 | p.update_attributes(:title => 'new title') |
|
152 | 153 | assert_equal 1, p.version # version does not increment |
|
153 | 154 | assert_equal 1, p.versions(true).size |
|
154 | ||
|
155 | ||
|
155 | 156 | p.update_attributes(:title => 'a title') |
|
156 | 157 | assert_equal 2, p.version |
|
157 | 158 | assert_equal 2, p.versions(true).size |
|
158 | ||
|
159 | ||
|
159 | 160 | # reset original if condition |
|
160 | 161 | Page.class_eval { alias_method :feeling_good?, :old_feeling_good } |
|
161 | 162 | end |
|
162 | ||
|
163 | ||
|
163 | 164 | def test_version_if_condition_with_block |
|
164 | 165 | # set new if condition |
|
165 | 166 | old_condition = Page.version_condition |
|
166 | 167 | Page.version_condition = Proc.new { |page| page.title[0..0] == 'b' } |
|
167 | ||
|
168 | p = Page.create :title => "title" | |
|
168 | ||
|
169 | p = Page.create! :title => "title" | |
|
169 | 170 | assert_equal 1, p.version # version does not increment |
|
170 | 171 | assert_equal 1, p.versions(true).size |
|
171 | ||
|
172 | ||
|
172 | 173 | p.update_attributes(:title => 'a title') |
|
173 | 174 | assert_equal 1, p.version # version does not increment |
|
174 | 175 | assert_equal 1, p.versions(true).size |
|
175 | ||
|
176 | ||
|
176 | 177 | p.update_attributes(:title => 'b title') |
|
177 | 178 | assert_equal 2, p.version |
|
178 | 179 | assert_equal 2, p.versions(true).size |
|
179 | ||
|
180 | ||
|
180 | 181 | # reset original if condition |
|
181 | 182 | Page.version_condition = old_condition |
|
182 | 183 | end |
|
183 | 184 | |
|
184 | 185 | def test_version_no_limit |
|
185 | p = Page.create :title => "title", :body => 'first body' | |
|
186 | p = Page.create! :title => "title", :body => 'first body' | |
|
186 | 187 | p.save |
|
187 | 188 | p.save |
|
188 | 189 | 5.times do |i| |
|
189 | 190 | assert_page_title p, i |
|
190 | 191 | end |
|
191 | 192 | end |
|
192 | 193 | |
|
193 | 194 | def test_version_max_limit |
|
194 | p = LockedPage.create :title => "title" | |
|
195 | p = LockedPage.create! :title => "title" | |
|
195 | 196 | p.update_attributes(:title => "title1") |
|
196 | 197 | p.update_attributes(:title => "title2") |
|
197 | 198 | 5.times do |i| |
|
198 | 199 | assert_page_title p, i, :lock_version |
|
199 | 200 | assert p.versions(true).size <= 2, "locked version can only store 2 versions" |
|
200 | 201 | end |
|
201 | 202 | end |
|
202 | ||
|
203 |
def test_track_ |
|
|
204 |
assert !Page.track_ |
|
|
205 |
assert LockedPage.track_ |
|
|
206 |
assert SpecialLockedPage.track_ |
|
|
203 | ||
|
204 | def test_track_altered_attributes_default_value | |
|
205 | assert !Page.track_altered_attributes | |
|
206 | assert LockedPage.track_altered_attributes | |
|
207 | assert SpecialLockedPage.track_altered_attributes | |
|
207 | 208 | end |
|
208 | ||
|
209 | ||
|
209 | 210 | def test_version_order |
|
210 | 211 | assert_equal 23, pages(:welcome).versions.first.version |
|
211 | 212 | assert_equal 24, pages(:welcome).versions.last.version |
|
212 | assert_equal 23, pages(:welcome).find_versions.first.version | |
|
213 | assert_equal 24, pages(:welcome).find_versions.last.version | |
|
214 | 213 | end |
|
215 | ||
|
216 |
def test_track_ |
|
|
217 | p = LockedPage.create :title => "title" | |
|
214 | ||
|
215 | def test_track_altered_attributes | |
|
216 | p = LockedPage.create! :title => "title" | |
|
218 | 217 | assert_equal 1, p.lock_version |
|
219 | 218 | assert_equal 1, p.versions(true).size |
|
220 | ||
|
219 | ||
|
221 | 220 | p.title = 'title' |
|
222 | 221 | assert !p.save_version? |
|
223 | 222 | p.save |
|
224 | 223 | assert_equal 2, p.lock_version # still increments version because of optimistic locking |
|
225 | 224 | assert_equal 1, p.versions(true).size |
|
226 | ||
|
225 | ||
|
227 | 226 | p.title = 'updated title' |
|
228 | 227 | assert p.save_version? |
|
229 | 228 | p.save |
|
230 | 229 | assert_equal 3, p.lock_version |
|
231 | 230 | assert_equal 1, p.versions(true).size # version 1 deleted |
|
232 | 231 | |
|
233 | 232 | p.title = 'updated title!' |
|
234 | 233 | assert p.save_version? |
|
235 | 234 | p.save |
|
236 | 235 | assert_equal 4, p.lock_version |
|
237 | 236 | assert_equal 2, p.versions(true).size # version 1 deleted |
|
238 | 237 | end |
|
239 | ||
|
238 | ||
|
240 | 239 | def assert_page_title(p, i, version_field = :version) |
|
241 | 240 | p.title = "title#{i}" |
|
242 | 241 | p.save |
|
243 | 242 | assert_equal "title#{i}", p.title |
|
244 | 243 | assert_equal (i+4), p.send(version_field) |
|
245 | 244 | end |
|
246 | ||
|
245 | ||
|
247 | 246 | def test_find_versions |
|
248 | 247 | assert_equal 2, locked_pages(:welcome).versions.size |
|
249 |
assert_equal 1, locked_pages(:welcome). |
|
|
250 |
assert_equal 2, locked_pages(:welcome). |
|
|
251 |
assert_equal 0, locked_pages(:thinking). |
|
|
252 |
assert_equal 2, locked_pages(:welcome). |
|
|
248 | assert_equal 1, locked_pages(:welcome).versions.find(:all, :conditions => ['title LIKE ?', '%weblog%']).length | |
|
249 | assert_equal 2, locked_pages(:welcome).versions.find(:all, :conditions => ['title LIKE ?', '%web%']).length | |
|
250 | assert_equal 0, locked_pages(:thinking).versions.find(:all, :conditions => ['title LIKE ?', '%web%']).length | |
|
251 | assert_equal 2, locked_pages(:welcome).versions.length | |
|
252 | end | |
|
253 | ||
|
254 | def test_find_version | |
|
255 | assert_equal page_versions(:welcome_1), Page.find_version(pages(:welcome).id, 23) | |
|
256 | assert_equal page_versions(:welcome_2), Page.find_version(pages(:welcome).id, 24) | |
|
257 | assert_equal pages(:welcome), Page.find_version(pages(:welcome).id) | |
|
258 | ||
|
259 | assert_equal page_versions(:welcome_1), pages(:welcome).find_version(23) | |
|
260 | assert_equal page_versions(:welcome_2), pages(:welcome).find_version(24) | |
|
261 | assert_equal pages(:welcome), pages(:welcome).find_version | |
|
262 | ||
|
263 | assert_raise(ActiveRecord::RecordNotFound) { Page.find_version(pages(:welcome).id, 1) } | |
|
264 | assert_raise(ActiveRecord::RecordNotFound) { Page.find_version(0, 23) } | |
|
253 | 265 | end |
|
254 | ||
|
266 | ||
|
255 | 267 | def test_with_sequence |
|
256 | 268 | assert_equal 'widgets_seq', Widget.versioned_class.sequence_name |
|
257 | Widget.create :name => 'new widget' | |
|
258 | Widget.create :name => 'new widget' | |
|
259 | Widget.create :name => 'new widget' | |
|
269 | 3.times { Widget.create! :name => 'new widget' } | |
|
260 | 270 | assert_equal 3, Widget.count |
|
261 | 271 | assert_equal 3, Widget.versioned_class.count |
|
262 | 272 | end |
|
263 | 273 | |
|
264 | 274 | def test_has_many_through |
|
265 | 275 | assert_equal [authors(:caged), authors(:mly)], pages(:welcome).authors |
|
266 | 276 | end |
|
267 | 277 | |
|
268 | 278 | def test_has_many_through_with_custom_association |
|
269 | 279 | assert_equal [authors(:caged), authors(:mly)], pages(:welcome).revisors |
|
270 | 280 | end |
|
271 | ||
|
281 | ||
|
272 | 282 | def test_referential_integrity |
|
273 | 283 | pages(:welcome).destroy |
|
274 | 284 | assert_equal 0, Page.count |
|
275 | 285 | assert_equal 0, Page::Version.count |
|
276 | 286 | end |
|
277 | ||
|
287 | ||
|
278 | 288 | def test_association_options |
|
279 | 289 | association = Page.reflect_on_association(:versions) |
|
280 | 290 | options = association.options |
|
281 | 291 | assert_equal :delete_all, options[:dependent] |
|
282 | 292 | assert_equal 'version', options[:order] |
|
283 | ||
|
293 | ||
|
284 | 294 | association = Widget.reflect_on_association(:versions) |
|
285 | 295 | options = association.options |
|
286 |
assert_ |
|
|
296 | assert_equal :nullify, options[:dependent] | |
|
287 | 297 | assert_equal 'version desc', options[:order] |
|
288 | 298 | assert_equal 'widget_id', options[:foreign_key] |
|
289 | ||
|
290 | widget = Widget.create :name => 'new widget' | |
|
299 | ||
|
300 | widget = Widget.create! :name => 'new widget' | |
|
291 | 301 | assert_equal 1, Widget.count |
|
292 | 302 | assert_equal 1, Widget.versioned_class.count |
|
293 | 303 | widget.destroy |
|
294 | 304 | assert_equal 0, Widget.count |
|
295 | 305 | assert_equal 1, Widget.versioned_class.count |
|
296 | 306 | end |
|
297 | 307 | |
|
298 | 308 | def test_versioned_records_should_belong_to_parent |
|
299 | 309 | page = pages(:welcome) |
|
300 | 310 | page_version = page.versions.last |
|
301 | 311 | assert_equal page, page_version.page |
|
302 | 312 | end |
|
303 | ||
|
304 |
def test_un |
|
|
305 | landmarks(:washington).attributes = landmarks(:washington).attributes | |
|
313 | ||
|
314 | def test_unaltered_attributes | |
|
315 | landmarks(:washington).attributes = landmarks(:washington).attributes.except("id") | |
|
306 | 316 | assert !landmarks(:washington).changed? |
|
307 | 317 | end |
|
308 | ||
|
318 | ||
|
309 | 319 | def test_unchanged_string_attributes |
|
310 |
landmarks(:washington).attributes = landmarks(:washington).attributes.inject({}) { |params, (key, value)| params.update |
|
|
320 | landmarks(:washington).attributes = landmarks(:washington).attributes.except("id").inject({}) { |params, (key, value)| params.update(key => value.to_s) } | |
|
311 | 321 | assert !landmarks(:washington).changed? |
|
312 | 322 | end |
|
313 | end | |
|
323 | ||
|
324 | def test_should_find_earliest_version | |
|
325 | assert_equal page_versions(:welcome_1), pages(:welcome).versions.earliest | |
|
326 | end | |
|
327 | ||
|
328 | def test_should_find_latest_version | |
|
329 | assert_equal page_versions(:welcome_2), pages(:welcome).versions.latest | |
|
330 | end | |
|
331 | ||
|
332 | def test_should_find_previous_version | |
|
333 | assert_equal page_versions(:welcome_1), page_versions(:welcome_2).previous | |
|
334 | assert_equal page_versions(:welcome_1), pages(:welcome).versions.before(page_versions(:welcome_2)) | |
|
335 | end | |
|
336 | ||
|
337 | def test_should_find_next_version | |
|
338 | assert_equal page_versions(:welcome_2), page_versions(:welcome_1).next | |
|
339 | assert_equal page_versions(:welcome_2), pages(:welcome).versions.after(page_versions(:welcome_1)) | |
|
340 | end | |
|
341 | ||
|
342 | def test_should_find_version_count | |
|
343 | assert_equal 24, pages(:welcome).versions_count | |
|
344 | assert_equal 24, page_versions(:welcome_1).versions_count | |
|
345 | assert_equal 24, page_versions(:welcome_2).versions_count | |
|
346 | end | |
|
347 | end No newline at end of file |
@@ -1,3 +1,9 | |||
|
1 | 1 | require 'rfpdf' |
|
2 | 2 | |
|
3 | ActionView::Base::register_template_handler 'rfpdf', RFPDF::View No newline at end of file | |
|
3 | begin | |
|
4 | ActionView::Template::register_template_handler 'rfpdf', RFPDF::View | |
|
5 | rescue NameError | |
|
6 | # Rails < 2.1 | |
|
7 | RFPDF::View.backward_compatibility_mode = true | |
|
8 | ActionView::Base::register_template_handler 'rfpdf', RFPDF::View | |
|
9 | end |
@@ -1,75 +1,85 | |||
|
1 | 1 | # Copyright (c) 2006 4ssoM LLC <www.4ssoM.com> |
|
2 | 2 | # |
|
3 | 3 | # The MIT License |
|
4 | 4 | # |
|
5 | 5 | # Permission is hereby granted, free of charge, to any person obtaining a copy |
|
6 | 6 | # of this software and associated documentation files (the "Software"), to deal |
|
7 | 7 | # in the Software without restriction, including without limitation the rights |
|
8 | 8 | # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
|
9 | 9 | # copies of the Software, and to permit persons to whom the Software is |
|
10 | 10 | # furnished to do so, subject to the following conditions: |
|
11 | 11 | # |
|
12 | 12 | # The above copyright notice and this permission notice shall be included in |
|
13 | 13 | # all copies or substantial portions of the Software. |
|
14 | 14 | # |
|
15 | 15 | # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
|
16 | 16 | # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
|
17 | 17 | # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
|
18 | 18 | # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
|
19 | 19 | # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
|
20 | 20 | # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
|
21 | 21 | # THE SOFTWARE. |
|
22 | 22 | # |
|
23 | 23 | # Thanks go out to Bruce Williams of codefluency who created RTex. This |
|
24 | 24 | # template handler is modification of his work. |
|
25 | 25 | # |
|
26 | 26 | # Example Registration |
|
27 | 27 | # |
|
28 | 28 | # ActionView::Base::register_template_handler 'rfpdf', RFpdfView |
|
29 | 29 | |
|
30 | 30 | module RFPDF |
|
31 | 31 | |
|
32 | 32 | class View |
|
33 | @@backward_compatibility_mode = false | |
|
34 | cattr_accessor :backward_compatibility_mode | |
|
33 | 35 | |
|
34 | 36 | def initialize(action_view) |
|
35 | 37 | @action_view = action_view |
|
36 | 38 | # Override with @options_for_rfpdf Hash in your controller |
|
37 | 39 | @options = { |
|
38 | 40 | # Run through latex first? (for table of contents, etc) |
|
39 | 41 | :pre_process => false, |
|
40 | 42 | # Debugging mode; raises exception |
|
41 | 43 | :debug => false, |
|
42 | 44 | # Filename of pdf to generate |
|
43 | 45 | :file_name => "#{@action_view.controller.action_name}.pdf", |
|
44 | 46 | # Temporary Directory |
|
45 | 47 | :temp_dir => "#{File.expand_path(RAILS_ROOT)}/tmp" |
|
46 | 48 | }.merge(@action_view.controller.instance_eval{ @options_for_rfpdf } || {}).with_indifferent_access |
|
47 | 49 | end |
|
50 | ||
|
51 | def self.compilable? | |
|
52 | false | |
|
53 | end | |
|
54 | ||
|
55 | def compilable? | |
|
56 | self.class.compilable? | |
|
57 | end | |
|
48 | 58 | |
|
49 | 59 | def render(template, local_assigns = {}) |
|
50 | 60 | @pdf_name = "Default.pdf" if @pdf_name.nil? |
|
51 | 61 | unless @action_view.controller.headers["Content-Type"] == 'application/pdf' |
|
52 | 62 | @generate = true |
|
53 | 63 | @action_view.controller.headers["Content-Type"] = 'application/pdf' |
|
54 | 64 | @action_view.controller.headers["Content-disposition:"] = "inline; filename=\"#{@options[:file_name]}\"" |
|
55 | 65 | end |
|
56 | 66 | assigns = @action_view.assigns.dup |
|
57 | 67 | |
|
58 | 68 | if content_for_layout = @action_view.instance_variable_get("@content_for_layout") |
|
59 | 69 | assigns['content_for_layout'] = content_for_layout |
|
60 | 70 | end |
|
61 | 71 | |
|
62 | 72 | result = @action_view.instance_eval do |
|
63 | 73 | assigns.each do |key,val| |
|
64 | 74 | instance_variable_set "@#{key}", val |
|
65 | 75 | end |
|
66 | 76 | local_assigns.each do |key,val| |
|
67 | 77 | class << self; self; end.send(:define_method,key){ val } |
|
68 | 78 | end |
|
69 | ERB.new(template).result(binding) | |
|
79 | ERB.new(@@backward_compatibility_mode == true ? template : template.source).result(binding) | |
|
70 | 80 | end |
|
71 | 81 | end |
|
72 | 82 | |
|
73 | 83 | end |
|
74 | 84 | |
|
75 | 85 | end No newline at end of file |
General Comments 0
You need to be logged in to leave comments.
Login now