jirametrics 2.12pre1 → 2.12pre2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/lib/jirametrics/board_movement_calculator.rb +6 -0
- data/lib/jirametrics/estimate_accuracy_chart.rb +33 -11
- data/lib/jirametrics/estimation_configuration.rb +0 -2
- data/lib/jirametrics/examples/standard_project.rb +6 -2
- data/lib/jirametrics/project_config.rb +2 -0
- data/lib/jirametrics/sprint_burndown.rb +32 -32
- data/lib/jirametrics/sprint_issue_change_data.rb +3 -3
- metadata +2 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 499b5a712159d628f0c19b54d826bcfff4ed52a7a74563f171505df6f9bb9059
|
4
|
+
data.tar.gz: 694f1973f07bbd88f062a2d1250690df2d43dc326d4368492ac58d89af5184c7
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 1b349c33e08ca97f135fb0d1a2625a238b35948ca16aa2e421887a79cc18c6f2537790e8093ce47b9e1cdc5b1e694f533efa2663b8739e00e5518b61908bb08e
|
7
|
+
data.tar.gz: f0da1809b457e37e473e7893c86dfc3009738ba98e5a9eba913f9a61f7ccd533a9b09deb1f3fa0126bfadbe3909cd811d3f590293c9fd9128d811ba3bf8aa56a
|
@@ -124,6 +124,12 @@ class BoardMovementCalculator
|
|
124
124
|
column_name, entry_time = find_current_column_and_entry_time_in_column issue
|
125
125
|
return [nil, 'This issue is not visible on the board. No way to predict when it will be done.'] if column_name.nil?
|
126
126
|
|
127
|
+
if entry_time.nil?
|
128
|
+
message = "Couldn't find the time issue #{issue.key} entered column #{column_name.inspect}"
|
129
|
+
puts message, issue.dump
|
130
|
+
return [nil, message]
|
131
|
+
end
|
132
|
+
|
127
133
|
age_in_column = (today - entry_time.to_date).to_i + 1
|
128
134
|
|
129
135
|
message = nil
|
@@ -22,15 +22,18 @@ class EstimateAccuracyChart < ChartBase
|
|
22
22
|
</div>
|
23
23
|
HTML
|
24
24
|
|
25
|
-
@y_axis_label = 'Story Point Estimates'
|
26
25
|
@y_axis_type = 'linear'
|
27
|
-
@y_axis_block = ->(issue, start_time) {
|
26
|
+
@y_axis_block = ->(issue, start_time) { estimate_at(issue: issue, start_time: start_time)&.to_f }
|
28
27
|
@y_axis_sort_order = nil
|
29
28
|
|
30
29
|
instance_eval(&configuration_block)
|
31
30
|
end
|
32
31
|
|
33
32
|
def run
|
33
|
+
if @y_axis_label.nil?
|
34
|
+
text = current_board.estimation_configuration.units == :story_points ? 'Story Points' : 'Days'
|
35
|
+
@y_axis_label = "Estimated #{text}"
|
36
|
+
end
|
34
37
|
data_sets = scan_issues
|
35
38
|
|
36
39
|
return '' if data_sets.empty?
|
@@ -41,6 +44,7 @@ class EstimateAccuracyChart < ChartBase
|
|
41
44
|
def scan_issues
|
42
45
|
completed_hash, aging_hash = split_into_completed_and_aging issues: issues
|
43
46
|
|
47
|
+
estimation_units = current_board.estimation_configuration.units
|
44
48
|
@has_aging_data = !aging_hash.empty?
|
45
49
|
|
46
50
|
[
|
@@ -53,9 +57,13 @@ class EstimateAccuracyChart < ChartBase
|
|
53
57
|
# We sort so that the smaller circles are in front of the bigger circles.
|
54
58
|
data = hash.sort(&hash_sorter).collect do |key, values|
|
55
59
|
estimate, cycle_time = *key
|
56
|
-
|
57
|
-
title = [
|
58
|
-
|
60
|
+
|
61
|
+
title = [
|
62
|
+
"Estimate: #{estimate_label(estimate: estimate, estimation_units: estimation_units)}, " \
|
63
|
+
"Cycletime: #{label_days(cycle_time)}, " \
|
64
|
+
"#{values.size} issues"
|
65
|
+
] + values.collect { |issue| "#{issue.key}: #{issue.summary}" }
|
66
|
+
|
59
67
|
{
|
60
68
|
'x' => cycle_time,
|
61
69
|
'y' => estimate,
|
@@ -77,6 +85,18 @@ class EstimateAccuracyChart < ChartBase
|
|
77
85
|
end
|
78
86
|
end
|
79
87
|
|
88
|
+
def estimate_label estimate:, estimation_units:
|
89
|
+
if @y_axis_type == 'linear'
|
90
|
+
if estimation_units == :story_points
|
91
|
+
estimate_label = "#{estimate}pts"
|
92
|
+
elsif estimation_units == :seconds
|
93
|
+
estimate_label = label_days estimate
|
94
|
+
end
|
95
|
+
end
|
96
|
+
estimate_label = estimate.to_s if estimate_label.nil?
|
97
|
+
estimate_label
|
98
|
+
end
|
99
|
+
|
80
100
|
def split_into_completed_and_aging issues:
|
81
101
|
aging_hash = {}
|
82
102
|
completed_hash = {}
|
@@ -126,16 +146,18 @@ class EstimateAccuracyChart < ChartBase
|
|
126
146
|
end
|
127
147
|
end
|
128
148
|
|
129
|
-
def
|
130
|
-
|
131
|
-
estimate_display_name = current_board.estimation_configuration.display_name
|
149
|
+
def estimate_at issue:, start_time:, estimation_configuration: current_board.estimation_configuration
|
150
|
+
estimate = nil
|
132
151
|
|
133
152
|
issue.changes.each do |change|
|
134
|
-
return
|
153
|
+
return estimate if change.time >= start_time
|
135
154
|
|
136
|
-
|
155
|
+
if change.field == estimation_configuration.display_name || change.field == estimation_configuration.field_id
|
156
|
+
estimate = change.value
|
157
|
+
estimate = estimate.to_f / (24 * 60 * 60) if estimation_configuration.units == :seconds
|
158
|
+
end
|
137
159
|
end
|
138
|
-
|
160
|
+
estimate
|
139
161
|
end
|
140
162
|
|
141
163
|
def y_axis label:, sort_order: nil, &block
|
@@ -72,8 +72,12 @@ class Exporter
|
|
72
72
|
grouping_rules do |issue, rules|
|
73
73
|
if issue.resolution
|
74
74
|
rules.label = "#{issue.status.name}:#{issue.resolution}"
|
75
|
-
|
76
|
-
|
75
|
+
if rules.label.start_with? 'Completed'
|
76
|
+
rules.label = 'Cancelled'
|
77
|
+
elsif rules.label.start_with? 'Closed'
|
78
|
+
rules.label = 'Done'
|
79
|
+
# rules.label = issue.status.name
|
80
|
+
end
|
77
81
|
end
|
78
82
|
end
|
79
83
|
end
|
@@ -151,6 +151,8 @@ class ProjectConfig
|
|
151
151
|
end
|
152
152
|
|
153
153
|
def status_category_mapping status:, category:
|
154
|
+
return if @exporter.downloading?
|
155
|
+
|
154
156
|
status, status_id = possible_statuses.parse_name_id status
|
155
157
|
category, category_id = possible_statuses.parse_name_id category
|
156
158
|
|
@@ -121,7 +121,7 @@ class SprintBurndown < ChartBase
|
|
121
121
|
|
122
122
|
# select all the changes that are relevant for the sprint. If this issue never appears in this sprint then return [].
|
123
123
|
def changes_for_one_issue issue:, sprint:
|
124
|
-
|
124
|
+
estimate = 0.0
|
125
125
|
ever_in_sprint = false
|
126
126
|
currently_in_sprint = false
|
127
127
|
change_data = []
|
@@ -142,26 +142,26 @@ class SprintBurndown < ChartBase
|
|
142
142
|
if currently_in_sprint == false && in_change_item
|
143
143
|
action = :enter_sprint
|
144
144
|
ever_in_sprint = true
|
145
|
-
value =
|
145
|
+
value = estimate
|
146
146
|
elsif currently_in_sprint && in_change_item == false
|
147
147
|
action = :leave_sprint
|
148
|
-
value = -
|
148
|
+
value = -estimate
|
149
149
|
end
|
150
150
|
currently_in_sprint = in_change_item
|
151
151
|
elsif change.field == estimate_display_name && (issue_completed_time.nil? || change.time < issue_completed_time)
|
152
152
|
action = :story_points
|
153
|
-
|
154
|
-
value =
|
153
|
+
estimate = change.value.to_f
|
154
|
+
value = estimate - change.old_value.to_f
|
155
155
|
elsif completed_has_been_tracked == false && change.time == issue_completed_time
|
156
156
|
completed_has_been_tracked = true
|
157
157
|
action = :issue_stopped
|
158
|
-
value = -
|
158
|
+
value = -estimate
|
159
159
|
end
|
160
160
|
|
161
161
|
next unless action
|
162
162
|
|
163
163
|
change_data << SprintIssueChangeData.new(
|
164
|
-
time: change.time, issue: issue, action: action, value: value,
|
164
|
+
time: change.time, issue: issue, action: action, value: value, estimate: estimate
|
165
165
|
)
|
166
166
|
end
|
167
167
|
|
@@ -178,7 +178,7 @@ class SprintBurndown < ChartBase
|
|
178
178
|
summary_stats = SprintSummaryStats.new
|
179
179
|
summary_stats.completed = 0.0
|
180
180
|
|
181
|
-
|
181
|
+
estimate = 0.0
|
182
182
|
start_data_written = false
|
183
183
|
data_set = []
|
184
184
|
|
@@ -187,11 +187,11 @@ class SprintBurndown < ChartBase
|
|
187
187
|
change_data_for_sprint.each do |change_data|
|
188
188
|
if start_data_written == false && change_data.time >= sprint.start_time
|
189
189
|
data_set << {
|
190
|
-
y:
|
190
|
+
y: estimate,
|
191
191
|
x: chart_format(sprint.start_time),
|
192
|
-
title: "Sprint started with #{
|
192
|
+
title: "Sprint started with #{estimate} points"
|
193
193
|
}
|
194
|
-
summary_stats.started =
|
194
|
+
summary_stats.started = estimate
|
195
195
|
start_data_written = true
|
196
196
|
end
|
197
197
|
|
@@ -200,12 +200,12 @@ class SprintBurndown < ChartBase
|
|
200
200
|
case change_data.action
|
201
201
|
when :enter_sprint
|
202
202
|
issues_currently_in_sprint << change_data.issue.key
|
203
|
-
|
203
|
+
estimate += change_data.estimate
|
204
204
|
when :leave_sprint
|
205
205
|
issues_currently_in_sprint.delete change_data.issue.key
|
206
|
-
|
206
|
+
estimate -= change_data.estimate
|
207
207
|
when :story_points
|
208
|
-
|
208
|
+
estimate += change_data.value if issues_currently_in_sprint.include? change_data.issue.key
|
209
209
|
end
|
210
210
|
|
211
211
|
next unless change_data.time >= sprint.start_time
|
@@ -215,26 +215,26 @@ class SprintBurndown < ChartBase
|
|
215
215
|
when :story_points
|
216
216
|
next unless issues_currently_in_sprint.include? change_data.issue.key
|
217
217
|
|
218
|
-
|
219
|
-
message = "Story points changed from #{
|
218
|
+
old_estimate = change_data.estimate - change_data.value
|
219
|
+
message = "Story points changed from #{old_estimate} points to #{change_data.estimate} points"
|
220
220
|
summary_stats.points_values_changed = true
|
221
221
|
when :enter_sprint
|
222
|
-
message = "Added to sprint with #{change_data.
|
223
|
-
summary_stats.added += change_data.
|
222
|
+
message = "Added to sprint with #{change_data.estimate || 'no'} points"
|
223
|
+
summary_stats.added += change_data.estimate
|
224
224
|
when :issue_stopped
|
225
|
-
|
226
|
-
message = "Completed with #{change_data.
|
225
|
+
estimate -= change_data.estimate
|
226
|
+
message = "Completed with #{change_data.estimate || 'no'} points"
|
227
227
|
issues_currently_in_sprint.delete change_data.issue.key
|
228
|
-
summary_stats.completed += change_data.
|
228
|
+
summary_stats.completed += change_data.estimate
|
229
229
|
when :leave_sprint
|
230
|
-
message = "Removed from sprint with #{change_data.
|
231
|
-
summary_stats.removed += change_data.
|
230
|
+
message = "Removed from sprint with #{change_data.estimate || 'no'} points"
|
231
|
+
summary_stats.removed += change_data.estimate
|
232
232
|
else
|
233
233
|
raise "Unexpected action: #{change_data.action}"
|
234
234
|
end
|
235
235
|
|
236
236
|
data_set << {
|
237
|
-
y:
|
237
|
+
y: estimate,
|
238
238
|
x: chart_format(change_data.time),
|
239
239
|
title: "#{change_data.issue.key} #{message}"
|
240
240
|
}
|
@@ -243,27 +243,27 @@ class SprintBurndown < ChartBase
|
|
243
243
|
unless start_data_written
|
244
244
|
# There was nothing that triggered us to write the sprint started block so do it now.
|
245
245
|
data_set << {
|
246
|
-
y:
|
246
|
+
y: estimate,
|
247
247
|
x: chart_format(sprint.start_time),
|
248
|
-
title: "Sprint started with #{
|
248
|
+
title: "Sprint started with #{estimate} points"
|
249
249
|
}
|
250
|
-
summary_stats.started =
|
250
|
+
summary_stats.started = estimate
|
251
251
|
end
|
252
252
|
|
253
253
|
if sprint.completed_time
|
254
254
|
data_set << {
|
255
|
-
y:
|
255
|
+
y: estimate,
|
256
256
|
x: chart_format(sprint.completed_time),
|
257
|
-
title: "Sprint ended with #{
|
257
|
+
title: "Sprint ended with #{estimate} points unfinished"
|
258
258
|
}
|
259
|
-
summary_stats.remaining =
|
259
|
+
summary_stats.remaining = estimate
|
260
260
|
end
|
261
261
|
|
262
262
|
unless sprint.completed_at?(time_range.end)
|
263
263
|
data_set << {
|
264
|
-
y:
|
264
|
+
y: estimate,
|
265
265
|
x: chart_format(time_range.end),
|
266
|
-
title: "Sprint still active. #{
|
266
|
+
title: "Sprint still active. #{estimate} points still in progress."
|
267
267
|
}
|
268
268
|
end
|
269
269
|
|
@@ -4,14 +4,14 @@ require 'jirametrics/value_equality'
|
|
4
4
|
|
5
5
|
class SprintIssueChangeData
|
6
6
|
include ValueEquality
|
7
|
-
attr_reader :time, :action, :value, :issue, :
|
7
|
+
attr_reader :time, :action, :value, :issue, :estimate
|
8
8
|
|
9
|
-
def initialize time:, action:, value:, issue:,
|
9
|
+
def initialize time:, action:, value:, issue:, estimate:
|
10
10
|
@time = time
|
11
11
|
@action = action
|
12
12
|
@value = value
|
13
13
|
@issue = issue
|
14
|
-
@
|
14
|
+
@estimate = estimate
|
15
15
|
end
|
16
16
|
|
17
17
|
def inspect
|
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: jirametrics
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.
|
4
|
+
version: 2.12pre2
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Mike Bowler
|
8
8
|
bindir: bin
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-03
|
10
|
+
date: 2025-04-03 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: random-word
|