changeset.rb
127 lines
| 5.0 KiB
| text/x-ruby
|
RubyLexer
|
r374 | # redMine - project management software | ||
# Copyright (C) 2006-2007 Jean-Philippe Lang | ||||
# | ||||
# This program is free software; you can redistribute it and/or | ||||
# modify it under the terms of the GNU General Public License | ||||
# as published by the Free Software Foundation; either version 2 | ||||
# of the License, or (at your option) any later version. | ||||
# | ||||
# This program is distributed in the hope that it will be useful, | ||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
# | ||||
# You should have received a copy of the GNU General Public License | ||||
# along with this program; if not, write to the Free Software | ||||
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | ||||
class Changeset < ActiveRecord::Base | ||||
belongs_to :repository | ||||
has_many :changes, :dependent => :delete_all | ||||
|
r470 | has_and_belongs_to_many :issues | ||
|
r663 | |||
acts_as_event :title => Proc.new {|o| "#{l(:label_revision)} #{o.revision}" + (o.comments.blank? ? '' : (': ' + o.comments))}, | ||||
:description => :comments, | ||||
:datetime => :committed_on, | ||||
:author => :committer, | ||||
:url => Proc.new {|o| {:controller => 'repositories', :action => 'revision', :id => o.repository.project_id, :rev => o.revision}} | ||||
|
r755 | |||
acts_as_searchable :columns => 'comments', | ||||
:include => :repository, | ||||
:project_key => "#{Repository.table_name}.project_id", | ||||
:date_column => 'committed_on' | ||||
|
r374 | |||
|
r377 | validates_presence_of :repository_id, :revision, :committed_on, :commit_date | ||
|
r374 | validates_uniqueness_of :revision, :scope => :repository_id | ||
|
r570 | validates_uniqueness_of :scmid, :scope => :repository_id, :allow_nil => true | ||
|
r377 | |||
|
r651 | def comments=(comment) | ||
write_attribute(:comments, comment.strip) | ||||
end | ||||
|
r377 | def committed_on=(date) | ||
self.commit_date = date | ||||
super | ||||
end | ||||
|
r470 | |||
|
r1213 | def project | ||
repository.project | ||||
end | ||||
|
r470 | def after_create | ||
scan_comment_for_issue_ids | ||||
end | ||||
|
r1112 | require 'pp' | ||
|
r470 | |||
def scan_comment_for_issue_ids | ||||
|
r476 | return if comments.blank? | ||
|
r470 | # keywords used to reference issues | ||
|
r848 | ref_keywords = Setting.commit_ref_keywords.downcase.split(",").collect(&:strip) | ||
|
r470 | # keywords used to fix issues | ||
|
r848 | fix_keywords = Setting.commit_fix_keywords.downcase.split(",").collect(&:strip) | ||
|
r810 | # status and optional done ratio applied | ||
|
r470 | fix_status = IssueStatus.find_by_id(Setting.commit_fix_status_id) | ||
|
r810 | done_ratio = Setting.commit_fix_done_ratio.blank? ? nil : Setting.commit_fix_done_ratio.to_i | ||
|
r470 | |||
|
r848 | kw_regexp = (ref_keywords + fix_keywords).collect{|kw| Regexp.escape(kw)}.join("|") | ||
|
r470 | return if kw_regexp.blank? | ||
|
r848 | referenced_issues = [] | ||
|
r905 | |||
if ref_keywords.delete('*') | ||||
# find any issue ID in the comments | ||||
target_issue_ids = [] | ||||
comments.scan(%r{([\s\(,-^])#(\d+)(?=[[:punct:]]|\s|<|$)}).each { |m| target_issue_ids << m[1] } | ||||
referenced_issues += repository.project.issues.find_all_by_id(target_issue_ids) | ||||
end | ||||
|
r476 | comments.scan(Regexp.new("(#{kw_regexp})[\s:]+(([\s,;&]*#?\\d+)+)", Regexp::IGNORECASE)).each do |match| | ||
|
r470 | action = match[0] | ||
target_issue_ids = match[1].scan(/\d+/) | ||||
target_issues = repository.project.issues.find_all_by_id(target_issue_ids) | ||||
if fix_status && fix_keywords.include?(action.downcase) | ||||
# update status of issues | ||||
logger.debug "Issues fixed by changeset #{self.revision}: #{issue_ids.join(', ')}." if logger && logger.debug? | ||||
target_issues.each do |issue| | ||||
|
r1169 | # the issue may have been updated by the closure of another one (eg. duplicate) | ||
issue.reload | ||||
|
r1112 | # don't change the status is the issue is closed | ||
|
r470 | next if issue.status.is_closed? | ||
|
r1112 | user = committer_user || User.anonymous | ||
|
r1222 | csettext = "r#{self.revision}" | ||
if self.scmid && (! (csettext =~ /^r[0-9]+$/)) | ||||
csettext = "commit:\"#{self.scmid}\"" | ||||
end | ||||
journal = issue.init_journal(user, l(:text_status_changed_by_changeset, csettext)) | ||||
|
r470 | issue.status = fix_status | ||
|
r810 | issue.done_ratio = done_ratio if done_ratio | ||
|
r470 | issue.save | ||
|
r1112 | Mailer.deliver_issue_edit(journal) if Setting.notified_events.include?('issue_updated') | ||
|
r470 | end | ||
end | ||||
|
r848 | referenced_issues += target_issues | ||
|
r470 | end | ||
|
r905 | |||
|
r848 | self.issues = referenced_issues.uniq | ||
|
r470 | end | ||
|
r925 | |||
|
r1112 | # Returns the Redmine User corresponding to the committer | ||
def committer_user | ||||
if committer && committer.strip =~ /^([^<]+)(<(.*)>)?$/ | ||||
username, email = $1.strip, $3 | ||||
u = User.find_by_login(username) | ||||
u ||= User.find_by_mail(email) unless email.blank? | ||||
u | ||||
end | ||||
end | ||||
|
r925 | # Returns the previous changeset | ||
def previous | ||||
|
r1222 | @previous ||= Changeset.find(:first, :conditions => ['id < ? AND repository_id = ?', self.id, self.repository_id], :order => 'id DESC') | ||
|
r925 | end | ||
# Returns the next changeset | ||||
def next | ||||
|
r1222 | @next ||= Changeset.find(:first, :conditions => ['id > ? AND repository_id = ?', self.id, self.repository_id], :order => 'id ASC') | ||
|
r925 | end | ||
|
r374 | end | ||