##// END OF EJS Templates
Use estimated hours to weight issues in version completion calculation (#2182)....
Jean-Philippe Lang -
r2347:d3b2049851c8
parent child
Show More
@@ -1,107 +1,143
1 # redMine - project management software
1 # redMine - project management software
2 # Copyright (C) 2006 Jean-Philippe Lang
2 # Copyright (C) 2006 Jean-Philippe Lang
3 #
3 #
4 # This program is free software; you can redistribute it and/or
4 # This program is free software; you can redistribute it and/or
5 # modify it under the terms of the GNU General Public License
5 # modify it under the terms of the GNU General Public License
6 # as published by the Free Software Foundation; either version 2
6 # as published by the Free Software Foundation; either version 2
7 # of the License, or (at your option) any later version.
7 # of the License, or (at your option) any later version.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU General Public License
14 # You should have received a copy of the GNU General Public License
15 # along with this program; if not, write to the Free Software
15 # along with this program; if not, write to the Free Software
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17
17
18 class Version < ActiveRecord::Base
18 class Version < ActiveRecord::Base
19 before_destroy :check_integrity
19 before_destroy :check_integrity
20 belongs_to :project
20 belongs_to :project
21 has_many :fixed_issues, :class_name => 'Issue', :foreign_key => 'fixed_version_id'
21 has_many :fixed_issues, :class_name => 'Issue', :foreign_key => 'fixed_version_id'
22 acts_as_attachable :view_permission => :view_files,
22 acts_as_attachable :view_permission => :view_files,
23 :delete_permission => :manage_files
23 :delete_permission => :manage_files
24
24
25 validates_presence_of :name
25 validates_presence_of :name
26 validates_uniqueness_of :name, :scope => [:project_id]
26 validates_uniqueness_of :name, :scope => [:project_id]
27 validates_length_of :name, :maximum => 60
27 validates_length_of :name, :maximum => 60
28 validates_format_of :effective_date, :with => /^\d{4}-\d{2}-\d{2}$/, :message => 'activerecord_error_not_a_date', :allow_nil => true
28 validates_format_of :effective_date, :with => /^\d{4}-\d{2}-\d{2}$/, :message => 'activerecord_error_not_a_date', :allow_nil => true
29
29
30 def start_date
30 def start_date
31 effective_date
31 effective_date
32 end
32 end
33
33
34 def due_date
34 def due_date
35 effective_date
35 effective_date
36 end
36 end
37
37
38 # Returns the total estimated time for this version
38 # Returns the total estimated time for this version
39 def estimated_hours
39 def estimated_hours
40 @estimated_hours ||= fixed_issues.sum(:estimated_hours).to_f
40 @estimated_hours ||= fixed_issues.sum(:estimated_hours).to_f
41 end
41 end
42
42
43 # Returns the total reported time for this version
43 # Returns the total reported time for this version
44 def spent_hours
44 def spent_hours
45 @spent_hours ||= TimeEntry.sum(:hours, :include => :issue, :conditions => ["#{Issue.table_name}.fixed_version_id = ?", id]).to_f
45 @spent_hours ||= TimeEntry.sum(:hours, :include => :issue, :conditions => ["#{Issue.table_name}.fixed_version_id = ?", id]).to_f
46 end
46 end
47
47
48 # Returns true if the version is completed: due date reached and no open issues
48 # Returns true if the version is completed: due date reached and no open issues
49 def completed?
49 def completed?
50 effective_date && (effective_date <= Date.today) && (open_issues_count == 0)
50 effective_date && (effective_date <= Date.today) && (open_issues_count == 0)
51 end
51 end
52
52
53 def completed_pourcent
53 def completed_pourcent
54 if fixed_issues.count == 0
54 if issues_count == 0
55 0
55 0
56 elsif open_issues_count == 0
56 elsif open_issues_count == 0
57 100
57 100
58 else
58 else
59 (closed_issues_count * 100 + Issue.sum('done_ratio', :include => 'status', :conditions => ["fixed_version_id = ? AND is_closed = ?", id, false]).to_f) / fixed_issues.count
59 issues_progress(false) + issues_progress(true)
60 end
60 end
61 end
61 end
62
62
63 def closed_pourcent
63 def closed_pourcent
64 if fixed_issues.count == 0
64 if issues_count == 0
65 0
65 0
66 else
66 else
67 closed_issues_count * 100.0 / fixed_issues.count
67 issues_progress(false)
68 end
68 end
69 end
69 end
70
70
71 # Returns true if the version is overdue: due date reached and some open issues
71 # Returns true if the version is overdue: due date reached and some open issues
72 def overdue?
72 def overdue?
73 effective_date && (effective_date < Date.today) && (open_issues_count > 0)
73 effective_date && (effective_date < Date.today) && (open_issues_count > 0)
74 end
74 end
75
75
76 # Returns assigned issues count
77 def issues_count
78 @issue_count ||= fixed_issues.count
79 end
80
76 def open_issues_count
81 def open_issues_count
77 @open_issues_count ||= Issue.count(:all, :conditions => ["fixed_version_id = ? AND is_closed = ?", self.id, false], :include => :status)
82 @open_issues_count ||= Issue.count(:all, :conditions => ["fixed_version_id = ? AND is_closed = ?", self.id, false], :include => :status)
78 end
83 end
79
84
80 def closed_issues_count
85 def closed_issues_count
81 @closed_issues_count ||= Issue.count(:all, :conditions => ["fixed_version_id = ? AND is_closed = ?", self.id, true], :include => :status)
86 @closed_issues_count ||= Issue.count(:all, :conditions => ["fixed_version_id = ? AND is_closed = ?", self.id, true], :include => :status)
82 end
87 end
83
88
84 def wiki_page
89 def wiki_page
85 if project.wiki && !wiki_page_title.blank?
90 if project.wiki && !wiki_page_title.blank?
86 @wiki_page ||= project.wiki.find_page(wiki_page_title)
91 @wiki_page ||= project.wiki.find_page(wiki_page_title)
87 end
92 end
88 @wiki_page
93 @wiki_page
89 end
94 end
90
95
91 def to_s; name end
96 def to_s; name end
92
97
93 # Versions are sorted by effective_date and name
98 # Versions are sorted by effective_date and name
94 # Those with no effective_date are at the end, sorted by name
99 # Those with no effective_date are at the end, sorted by name
95 def <=>(version)
100 def <=>(version)
96 if self.effective_date
101 if self.effective_date
97 version.effective_date ? (self.effective_date == version.effective_date ? self.name <=> version.name : self.effective_date <=> version.effective_date) : -1
102 version.effective_date ? (self.effective_date == version.effective_date ? self.name <=> version.name : self.effective_date <=> version.effective_date) : -1
98 else
103 else
99 version.effective_date ? 1 : (self.name <=> version.name)
104 version.effective_date ? 1 : (self.name <=> version.name)
100 end
105 end
101 end
106 end
102
107
103 private
108 private
104 def check_integrity
109 def check_integrity
105 raise "Can't delete version" if self.fixed_issues.find(:first)
110 raise "Can't delete version" if self.fixed_issues.find(:first)
106 end
111 end
112
113 # Returns the average estimated time of assigned issues
114 # or 1 if no issue has an estimated time
115 # Used to weigth unestimated issues in progress calculation
116 def estimated_average
117 if @estimated_average.nil?
118 average = fixed_issues.average(:estimated_hours).to_f
119 if average == 0
120 average = 1
121 end
122 @estimated_average = average
123 end
124 @estimated_average
125 end
126
127 # Returns the total progress of open or closed issues
128 def issues_progress(open)
129 @issues_progress ||= {}
130 @issues_progress[open] ||= begin
131 progress = 0
132 if issues_count > 0
133 ratio = open ? 'done_ratio' : 100
134 done = fixed_issues.sum("COALESCE(estimated_hours, #{estimated_average}) * #{ratio}",
135 :include => :status,
136 :conditions => ["is_closed = ?", !open]).to_f
137
138 progress = done / (estimated_average * issues_count)
139 end
140 progress
141 end
142 end
107 end
143 end
@@ -1,36 +1,120
1 # Redmine - project management software
1 # Redmine - project management software
2 # Copyright (C) 2006-2008 Jean-Philippe Lang
2 # Copyright (C) 2006-2008 Jean-Philippe Lang
3 #
3 #
4 # This program is free software; you can redistribute it and/or
4 # This program is free software; you can redistribute it and/or
5 # modify it under the terms of the GNU General Public License
5 # modify it under the terms of the GNU General Public License
6 # as published by the Free Software Foundation; either version 2
6 # as published by the Free Software Foundation; either version 2
7 # of the License, or (at your option) any later version.
7 # of the License, or (at your option) any later version.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU General Public License
14 # You should have received a copy of the GNU General Public License
15 # along with this program; if not, write to the Free Software
15 # along with this program; if not, write to the Free Software
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17
17
18 require File.dirname(__FILE__) + '/../test_helper'
18 require File.dirname(__FILE__) + '/../test_helper'
19
19
20 class VersionTest < Test::Unit::TestCase
20 class VersionTest < Test::Unit::TestCase
21 fixtures :projects, :issues, :issue_statuses, :versions
21 fixtures :projects, :users, :issues, :issue_statuses, :trackers, :enumerations, :versions
22
22
23 def setup
23 def setup
24 end
24 end
25
25
26 def test_create
26 def test_create
27 v = Version.new(:project => Project.find(1), :name => '1.1', :effective_date => '2011-03-25')
27 v = Version.new(:project => Project.find(1), :name => '1.1', :effective_date => '2011-03-25')
28 assert v.save
28 assert v.save
29 end
29 end
30
30
31 def test_invalid_effective_date_validation
31 def test_invalid_effective_date_validation
32 v = Version.new(:project => Project.find(1), :name => '1.1', :effective_date => '99999-01-01')
32 v = Version.new(:project => Project.find(1), :name => '1.1', :effective_date => '99999-01-01')
33 assert !v.save
33 assert !v.save
34 assert_equal 'activerecord_error_not_a_date', v.errors.on(:effective_date)
34 assert_equal 'activerecord_error_not_a_date', v.errors.on(:effective_date)
35 end
35 end
36
37 def test_progress_should_be_0_with_no_assigned_issues
38 project = Project.find(1)
39 v = Version.create!(:project => project, :name => 'Progress')
40 assert_equal 0, v.completed_pourcent
41 assert_equal 0, v.closed_pourcent
42 end
43
44 def test_progress_should_be_0_with_unbegun_assigned_issues
45 project = Project.find(1)
46 v = Version.create!(:project => project, :name => 'Progress')
47 add_issue(v)
48 add_issue(v, :done_ratio => 0)
49 assert_progress_equal 0, v.completed_pourcent
50 assert_progress_equal 0, v.closed_pourcent
51 end
52
53 def test_progress_should_be_100_with_closed_assigned_issues
54 project = Project.find(1)
55 status = IssueStatus.find(:first, :conditions => {:is_closed => true})
56 v = Version.create!(:project => project, :name => 'Progress')
57 add_issue(v, :status => status)
58 add_issue(v, :status => status, :done_ratio => 20)
59 add_issue(v, :status => status, :done_ratio => 70, :estimated_hours => 25)
60 add_issue(v, :status => status, :estimated_hours => 15)
61 assert_progress_equal 100.0, v.completed_pourcent
62 assert_progress_equal 100.0, v.closed_pourcent
63 end
64
65 def test_progress_should_consider_done_ratio_of_open_assigned_issues
66 project = Project.find(1)
67 v = Version.create!(:project => project, :name => 'Progress')
68 add_issue(v)
69 add_issue(v, :done_ratio => 20)
70 add_issue(v, :done_ratio => 70)
71 assert_progress_equal (0.0 + 20.0 + 70.0)/3, v.completed_pourcent
72 assert_progress_equal 0, v.closed_pourcent
73 end
74
75 def test_progress_should_consider_closed_issues_as_completed
76 project = Project.find(1)
77 v = Version.create!(:project => project, :name => 'Progress')
78 add_issue(v)
79 add_issue(v, :done_ratio => 20)
80 add_issue(v, :status => IssueStatus.find(:first, :conditions => {:is_closed => true}))
81 assert_progress_equal (0.0 + 20.0 + 100.0)/3, v.completed_pourcent
82 assert_progress_equal (100.0)/3, v.closed_pourcent
83 end
84
85 def test_progress_should_consider_estimated_hours_to_weigth_issues
86 project = Project.find(1)
87 v = Version.create!(:project => project, :name => 'Progress')
88 add_issue(v, :estimated_hours => 10)
89 add_issue(v, :estimated_hours => 20, :done_ratio => 30)
90 add_issue(v, :estimated_hours => 40, :done_ratio => 10)
91 add_issue(v, :estimated_hours => 25, :status => IssueStatus.find(:first, :conditions => {:is_closed => true}))
92 assert_progress_equal (10.0*0 + 20.0*0.3 + 40*0.1 + 25.0*1)/95.0*100, v.completed_pourcent
93 assert_progress_equal 25.0/95.0*100, v.closed_pourcent
94 end
95
96 def test_progress_should_consider_average_estimated_hours_to_weigth_unestimated_issues
97 project = Project.find(1)
98 v = Version.create!(:project => project, :name => 'Progress')
99 add_issue(v, :done_ratio => 20)
100 add_issue(v, :status => IssueStatus.find(:first, :conditions => {:is_closed => true}))
101 add_issue(v, :estimated_hours => 10, :done_ratio => 30)
102 add_issue(v, :estimated_hours => 40, :done_ratio => 10)
103 assert_progress_equal (25.0*0.2 + 25.0*1 + 10.0*0.3 + 40.0*0.1)/100.0*100, v.completed_pourcent
104 assert_progress_equal 25.0/100.0*100, v.closed_pourcent
105 end
106
107 private
108
109 def add_issue(version, attributes={})
110 Issue.create!({:project => version.project,
111 :fixed_version => version,
112 :subject => 'Test',
113 :author => User.find(:first),
114 :tracker => version.project.trackers.find(:first)}.merge(attributes))
115 end
116
117 def assert_progress_equal(expected_float, actual_float, message="")
118 assert_in_delta(expected_float, actual_float, 0.000001, message="")
119 end
36 end
120 end
General Comments 0
You need to be logged in to leave comments. Login now