lazylead 0.3.1 → 0.5.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.0pdd.yml +4 -1
- data/.docs/accuracy.md +107 -0
- data/.docs/accuracy_email.jpg +0 -0
- data/.docs/accuracy_jira_comment.jpg +0 -0
- data/.docs/duedate_expired.md +3 -3
- data/.docs/propagate_down.md +4 -4
- data/.gitattributes +1 -0
- data/.github/dependabot.yml +6 -0
- data/.pdd +1 -1
- data/.rubocop.yml +6 -0
- data/Rakefile +2 -0
- data/bin/lazylead +7 -4
- data/lazylead.gemspec +5 -4
- data/lib/lazylead/exchange.rb +16 -9
- data/lib/lazylead/log.rb +30 -8
- data/lib/lazylead/model.rb +78 -22
- data/lib/lazylead/opts.rb +80 -0
- data/lib/lazylead/postman.rb +1 -1
- data/lib/lazylead/schedule.rb +18 -17
- data/lib/lazylead/smtp.rb +1 -1
- data/lib/lazylead/system/jira.rb +55 -14
- data/lib/lazylead/system/synced.rb +2 -1
- data/lib/lazylead/task/accuracy/accuracy.rb +136 -0
- data/lib/lazylead/task/accuracy/affected_build.rb +39 -0
- data/lib/lazylead/task/accuracy/attachment.rb +44 -0
- data/lib/lazylead/task/accuracy/environment.rb +39 -0
- data/lib/lazylead/task/accuracy/logs.rb +40 -0
- data/lib/lazylead/task/accuracy/records.rb +45 -0
- data/lib/lazylead/task/accuracy/requirement.rb +49 -0
- data/lib/lazylead/task/accuracy/servers.rb +50 -0
- data/lib/lazylead/task/accuracy/stacktrace.rb +63 -0
- data/lib/lazylead/task/accuracy/testcase.rb +75 -0
- data/lib/lazylead/task/accuracy/wiki.rb +41 -0
- data/lib/lazylead/task/alert.rb +8 -6
- data/lib/lazylead/task/confluence_ref.rb +4 -3
- data/lib/lazylead/task/echo.rb +22 -0
- data/lib/lazylead/task/fix_version.rb +18 -7
- data/lib/lazylead/task/missing_comment.rb +7 -5
- data/lib/lazylead/task/propagate_down.rb +11 -3
- data/lib/lazylead/task/savepoint.rb +1 -1
- data/lib/lazylead/task/touch.rb +119 -0
- data/lib/lazylead/version.rb +1 -1
- data/lib/messages/accuracy.erb +118 -0
- data/lib/messages/svn_log.erb +117 -0
- data/lib/messages/svn_touch.erb +147 -0
- data/license.txt +1 -1
- data/readme.md +20 -19
- data/test/lazylead/cc_test.rb +2 -2
- data/test/lazylead/cli/app_test.rb +12 -12
- data/test/lazylead/exchange_test.rb +3 -3
- data/test/lazylead/model_test.rb +4 -4
- data/test/lazylead/opts_test.rb +70 -0
- data/test/lazylead/postman_test.rb +1 -1
- data/test/lazylead/smtp_test.rb +1 -1
- data/test/lazylead/system/jira_test.rb +65 -1
- data/test/lazylead/task/accuracy/accuracy_test.rb +73 -0
- data/test/lazylead/task/accuracy/affected_build_test.rb +42 -0
- data/test/lazylead/task/accuracy/attachment_test.rb +50 -0
- data/test/lazylead/task/accuracy/environment_test.rb +42 -0
- data/test/lazylead/task/accuracy/logs_test.rb +78 -0
- data/test/lazylead/task/accuracy/records_test.rb +60 -0
- data/test/lazylead/task/accuracy/servers_test.rb +66 -0
- data/test/lazylead/task/accuracy/stacktrace_test.rb +113 -0
- data/test/lazylead/task/accuracy/testcase_test.rb +205 -0
- data/test/lazylead/task/accuracy/wiki_test.rb +40 -0
- data/test/lazylead/task/assignee_alert_test.rb +2 -2
- data/test/lazylead/task/duedate_test.rb +36 -26
- data/test/lazylead/task/fix_version_test.rb +9 -6
- data/test/lazylead/task/missing_comment_test.rb +11 -9
- data/test/lazylead/task/propagate_down_test.rb +4 -2
- data/test/lazylead/task/touch_test.rb +88 -0
- data/test/test.rb +25 -0
- data/upgrades/sqlite/001-install-main-lazylead-tables.sql +1 -5
- data/upgrades/sqlite/999.testdata.sql +12 -16
- metadata +65 -8
- data/.travis.yml +0 -16
data/lib/lazylead/task/alert.rb
CHANGED
@@ -23,6 +23,7 @@
|
|
23
23
|
# OR OTHER DEALINGS IN THE SOFTWARE.
|
24
24
|
|
25
25
|
require_relative "../log"
|
26
|
+
require_relative "../opts"
|
26
27
|
require_relative "../email"
|
27
28
|
require_relative "../version"
|
28
29
|
require_relative "../postman"
|
@@ -43,12 +44,13 @@ module Lazylead
|
|
43
44
|
# - prepare email based on predefined template (*.erb)
|
44
45
|
# - send the required notifications pre-defined "addressee".
|
45
46
|
class Alert
|
46
|
-
def initialize(log = Log
|
47
|
+
def initialize(log = Log.new)
|
47
48
|
@log = log
|
48
49
|
end
|
49
50
|
|
50
51
|
def run(sys, postman, opts)
|
51
|
-
|
52
|
+
tickets = sys.issues(opts["sql"], opts.jira_defaults)
|
53
|
+
postman.send opts.merge(tickets: tickets) unless tickets.empty?
|
52
54
|
end
|
53
55
|
end
|
54
56
|
|
@@ -64,12 +66,12 @@ module Lazylead
|
|
64
66
|
# The email message is sending to the assignee regarding all his/her issues,
|
65
67
|
# not like one email per each issue.
|
66
68
|
class AssigneeAlert
|
67
|
-
def initialize(log = Log
|
69
|
+
def initialize(log = Log.new)
|
68
70
|
@log = log
|
69
71
|
end
|
70
72
|
|
71
73
|
def run(sys, postman, opts)
|
72
|
-
sys.issues(opts["sql"])
|
74
|
+
sys.issues(opts["sql"], opts.jira_defaults)
|
73
75
|
.group_by(&:assignee)
|
74
76
|
.each do |a, t|
|
75
77
|
postman.send opts.merge(to: a.email, addressee: a.name, tickets: t)
|
@@ -89,12 +91,12 @@ module Lazylead
|
|
89
91
|
# The email message is sending to the assignee regarding all his/her issues,
|
90
92
|
# not like one email per each issue.
|
91
93
|
class ReporterAlert
|
92
|
-
def initialize(log = Log
|
94
|
+
def initialize(log = Log.new)
|
93
95
|
@log = log
|
94
96
|
end
|
95
97
|
|
96
98
|
def run(sys, postman, opts)
|
97
|
-
sys.issues(opts["sql"])
|
99
|
+
sys.issues(opts["sql"], opts.jira_defaults)
|
98
100
|
.group_by(&:reporter)
|
99
101
|
.each do |a, t|
|
100
102
|
postman.send opts.merge(to: a.email, addressee: a.name, tickets: t)
|
@@ -26,6 +26,7 @@ require "json"
|
|
26
26
|
require "faraday"
|
27
27
|
require_relative "../system/jira"
|
28
28
|
require_relative "../log"
|
29
|
+
require_relative "../opts"
|
29
30
|
require_relative "../confluence"
|
30
31
|
|
31
32
|
module Lazylead
|
@@ -35,14 +36,14 @@ module Lazylead
|
|
35
36
|
# @todo #/DEV Support sub-task for link search. Potentially, the issue
|
36
37
|
# might have sub-tasks where discussion ongoing.
|
37
38
|
class ConfluenceRef
|
38
|
-
def initialize(log = Log
|
39
|
+
def initialize(log = Log.new)
|
39
40
|
@log = log
|
40
41
|
end
|
41
42
|
|
42
43
|
def run(sys, _, opts)
|
43
44
|
confluences = confluences(opts)
|
44
45
|
return if confluences.empty?
|
45
|
-
sys.issues(opts["jql"])
|
46
|
+
sys.issues(opts["jql"], opts.jira_defaults)
|
46
47
|
.map { |i| Link.new(i, sys, confluences) }
|
47
48
|
.each(&:fetch_links)
|
48
49
|
.select(&:need_link?)
|
@@ -50,7 +51,7 @@ module Lazylead
|
|
50
51
|
end
|
51
52
|
|
52
53
|
def confluences(opts)
|
53
|
-
return [] if opts
|
54
|
+
return [] if opts.blank? "confluences"
|
54
55
|
JSON.parse(opts["confluences"], object_class: OpenStruct)
|
55
56
|
.map { |c| Confluence.new(c) }
|
56
57
|
end
|
data/lib/lazylead/task/echo.rb
CHANGED
@@ -22,6 +22,8 @@
|
|
22
22
|
# ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE
|
23
23
|
# OR OTHER DEALINGS IN THE SOFTWARE.
|
24
24
|
|
25
|
+
require_relative "../log"
|
26
|
+
|
25
27
|
module Lazylead
|
26
28
|
module Task
|
27
29
|
# Lazylead task which prints to STDOUT the current class name and team.
|
@@ -30,9 +32,29 @@ module Lazylead
|
|
30
32
|
# Copyright:: Copyright (c) 2019-2020 Yurii Dubinka
|
31
33
|
# License:: MIT
|
32
34
|
class Echo
|
35
|
+
def initialize(log = Log.new)
|
36
|
+
@log = log
|
37
|
+
end
|
38
|
+
|
33
39
|
def run(_, _, _)
|
34
40
|
self.class.to_s
|
35
41
|
end
|
36
42
|
end
|
43
|
+
|
44
|
+
# Lazylead task which prints the current time to a file.
|
45
|
+
#
|
46
|
+
# Author:: Yurii Dubinka (yurii.dubinka@gmail.com)
|
47
|
+
# Copyright:: Copyright (c) 2019-2020 Yurii Dubinka
|
48
|
+
# License:: MIT
|
49
|
+
class EchoIO
|
50
|
+
def initialize(log = Log.new, path = "test/resources/echo.txt")
|
51
|
+
@log = log
|
52
|
+
@path = path
|
53
|
+
end
|
54
|
+
|
55
|
+
def run(_, _, _)
|
56
|
+
File.open(@path, "w") { |f| f.write Time.now }
|
57
|
+
end
|
58
|
+
end
|
37
59
|
end
|
38
60
|
end
|
@@ -23,24 +23,30 @@
|
|
23
23
|
# OR OTHER DEALINGS IN THE SOFTWARE.
|
24
24
|
|
25
25
|
require "date"
|
26
|
-
require_relative "../system/jira"
|
27
26
|
require_relative "../log"
|
27
|
+
require_relative "../opts"
|
28
|
+
require_relative "../system/jira"
|
28
29
|
|
29
30
|
module Lazylead
|
30
31
|
module Task
|
31
32
|
# @todo #/DEV Each task should verify input arguments.
|
32
33
|
# The common API should be provided for each task.
|
33
34
|
class FixVersion
|
34
|
-
def initialize(log = Log
|
35
|
+
def initialize(log = Log.new)
|
35
36
|
@log = log
|
36
37
|
end
|
37
38
|
|
38
39
|
def run(sys, postman, opts)
|
39
|
-
allowed = opts
|
40
|
+
allowed = opts.slice("allowed", ",")
|
41
|
+
silent = opts.key? "silent"
|
42
|
+
issues = sys.issues(
|
43
|
+
opts["jql"], opts.jira_defaults.merge(expand: "changelog")
|
44
|
+
)
|
45
|
+
return if issues.empty?
|
40
46
|
postman.send opts.merge(
|
41
|
-
versions:
|
42
|
-
|
43
|
-
|
47
|
+
versions: issues.map { |i| Version.new(i, allowed, silent) }
|
48
|
+
.select(&:changed?)
|
49
|
+
.each(&:add_label)
|
44
50
|
)
|
45
51
|
end
|
46
52
|
end
|
@@ -49,9 +55,10 @@ module Lazylead
|
|
49
55
|
class Version
|
50
56
|
attr_reader :issue
|
51
57
|
|
52
|
-
def initialize(issue, allowed)
|
58
|
+
def initialize(issue, allowed, silent)
|
53
59
|
@issue = issue
|
54
60
|
@allowed = allowed
|
61
|
+
@silent = silent
|
55
62
|
end
|
56
63
|
|
57
64
|
# Gives true when last change of "Fix Version" field was done
|
@@ -74,6 +81,10 @@ module Lazylead
|
|
74
81
|
end
|
75
82
|
end
|
76
83
|
end
|
84
|
+
|
85
|
+
def add_label
|
86
|
+
@issue.add_label("LL.IllegalChangeOfFixVersion") unless @silent
|
87
|
+
end
|
77
88
|
end
|
78
89
|
end
|
79
90
|
end
|
@@ -23,6 +23,7 @@
|
|
23
23
|
# OR OTHER DEALINGS IN THE SOFTWARE.
|
24
24
|
|
25
25
|
require_relative "../log"
|
26
|
+
require_relative "../opts"
|
26
27
|
require_relative "../postman"
|
27
28
|
|
28
29
|
module Lazylead
|
@@ -36,16 +37,17 @@ module Lazylead
|
|
36
37
|
# nobody mentioned in comment the ftp location for recorded session.
|
37
38
|
# Such cases needs to be reported.
|
38
39
|
class MissingComment
|
39
|
-
def initialize(log = Log
|
40
|
+
def initialize(log = Log.new)
|
40
41
|
@log = log
|
41
42
|
end
|
42
43
|
|
43
44
|
def run(sys, postman, opts)
|
44
|
-
opts["details"] = "text '#{opts['text']}'" if opts
|
45
|
+
opts["details"] = "text '#{opts['text']}'" if opts.blank? "details"
|
46
|
+
issues = sys.issues(opts["jql"], opts.jira_defaults)
|
47
|
+
return if issues.empty?
|
45
48
|
postman.send opts.merge(
|
46
|
-
comments:
|
47
|
-
|
48
|
-
.reject { |c| c.body? opts["text"] }
|
49
|
+
comments: issues.map { |i| Comments.new(i, sys) }
|
50
|
+
.reject { |c| c.body? opts["text"] }
|
49
51
|
)
|
50
52
|
end
|
51
53
|
end
|
@@ -23,6 +23,7 @@
|
|
23
23
|
# OR OTHER DEALINGS IN THE SOFTWARE.
|
24
24
|
|
25
25
|
require_relative "../log"
|
26
|
+
require_relative "../opts"
|
26
27
|
require_relative "../version"
|
27
28
|
require_relative "../system/jira"
|
28
29
|
|
@@ -39,15 +40,22 @@ module Lazylead
|
|
39
40
|
# - apply diff to sub-tasks
|
40
41
|
# - make a comment to sub-task with clarification.
|
41
42
|
class PropagateDown
|
42
|
-
def initialize(log = Log
|
43
|
+
def initialize(log = Log.new)
|
43
44
|
@log = log
|
44
45
|
end
|
45
46
|
|
46
47
|
# @todo #/DEV Define a new module Lazylead::Task with basic methods like
|
47
48
|
# split, groupBy(assignee, reporter, etc), blank?
|
48
49
|
def run(sys, _, opts)
|
49
|
-
fields = opts
|
50
|
-
sys.issues(
|
50
|
+
fields = opts.slice("propagate", ",")
|
51
|
+
sys.issues(
|
52
|
+
opts["jql"],
|
53
|
+
{
|
54
|
+
expand: "changelog",
|
55
|
+
max_results: opts.fetch("max_results", 50),
|
56
|
+
fields: ["subtasks"] + opts.jira_fields
|
57
|
+
}
|
58
|
+
)
|
51
59
|
.map { |i| Parent.new(i, sys, fields) }
|
52
60
|
.select(&:subtasks?)
|
53
61
|
.each(&:fetch)
|
@@ -0,0 +1,119 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# The MIT License
|
4
|
+
#
|
5
|
+
# Copyright (c) 2019-2020 Yurii Dubinka
|
6
|
+
#
|
7
|
+
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
8
|
+
# of this software and associated documentation files (the "Software"),
|
9
|
+
# to deal in the Software without restriction, including without limitation
|
10
|
+
# the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
11
|
+
# and/or sell copies of the Software, and to permit persons to whom
|
12
|
+
# the Software is furnished to do so, subject to the following conditions:
|
13
|
+
#
|
14
|
+
# The above copyright notice and this permission notice shall be included
|
15
|
+
# in all copies or substantial portions of the Software.
|
16
|
+
#
|
17
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
18
|
+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
19
|
+
# FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
|
20
|
+
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
21
|
+
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
22
|
+
# ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE
|
23
|
+
# OR OTHER DEALINGS IN THE SOFTWARE.
|
24
|
+
|
25
|
+
require "nokogiri"
|
26
|
+
require "active_support/core_ext/hash/conversions"
|
27
|
+
require_relative "../salt"
|
28
|
+
require_relative "../opts"
|
29
|
+
|
30
|
+
module Lazylead
|
31
|
+
module Task
|
32
|
+
#
|
33
|
+
# Send notification about modification of critical files in svn repo.
|
34
|
+
#
|
35
|
+
class SvnTouch
|
36
|
+
def initialize(log = Log.new)
|
37
|
+
@log = log
|
38
|
+
end
|
39
|
+
|
40
|
+
def run(_, postman, opts)
|
41
|
+
files = opts.slice("files", ",")
|
42
|
+
commits = touch(files, opts)
|
43
|
+
postman.send(opts.merge(entries: commits)) unless commits.empty?
|
44
|
+
end
|
45
|
+
|
46
|
+
# Return all svn commits for a particular date range, which are touching
|
47
|
+
# somehow the critical files within the svn repo.
|
48
|
+
def touch(files, opts)
|
49
|
+
xpath = files.map { |f| "contains(text(),\"#{f}\")" }.join(" or ")
|
50
|
+
svn_log(opts).xpath("//logentry[paths/path[#{xpath}]]")
|
51
|
+
.map(&method(:to_entry))
|
52
|
+
.each do |e|
|
53
|
+
if e.paths.path.respond_to? :delete_if
|
54
|
+
e.paths.path.delete_if { |p| files.none? { |f| p.include? f } }
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
# Return all svn commits for particular date range in repo
|
60
|
+
def svn_log(opts)
|
61
|
+
now = if opts.key? "now"
|
62
|
+
DateTime.parse(opts["now"])
|
63
|
+
else
|
64
|
+
DateTime.now
|
65
|
+
end
|
66
|
+
start = (now.to_time - opts["period"].to_i).to_datetime
|
67
|
+
cmd = [
|
68
|
+
"svn log --no-auth-cache",
|
69
|
+
"--username #{opts.decrypt('svn_user', 'svn_salt')}",
|
70
|
+
"--password #{opts.decrypt('svn_password', 'svn_salt')}",
|
71
|
+
"--xml -v -r {#{start}}:{#{now}} #{opts['svn_url']}"
|
72
|
+
]
|
73
|
+
raw = `#{cmd.join(" ")}`
|
74
|
+
Nokogiri.XML(raw, nil, "UTF-8")
|
75
|
+
end
|
76
|
+
|
77
|
+
# Convert single revision(XML text) to entry object.
|
78
|
+
# Entry object is a simple ruby struct object.
|
79
|
+
def to_entry(xml)
|
80
|
+
e = to_struct(Hash.from_xml(xml.to_s.strip)).logentry
|
81
|
+
if e.paths.path.respond_to? :each
|
82
|
+
e.paths.path.each(&:strip!)
|
83
|
+
else
|
84
|
+
e.paths.path.strip!
|
85
|
+
end
|
86
|
+
e
|
87
|
+
end
|
88
|
+
|
89
|
+
# Make a simple ruby struct object from hash hierarchically,
|
90
|
+
# considering nested hash(es) (if applicable)
|
91
|
+
def to_struct(hsh)
|
92
|
+
OpenStruct.new(
|
93
|
+
hsh.transform_values { |v| v.is_a?(Hash) ? to_struct(v) : v }
|
94
|
+
)
|
95
|
+
end
|
96
|
+
end
|
97
|
+
|
98
|
+
#
|
99
|
+
# Send notification about modification of svn files since particular
|
100
|
+
# revision.
|
101
|
+
#
|
102
|
+
class SvnLog
|
103
|
+
def initialize(log = Log.new)
|
104
|
+
@log = log
|
105
|
+
end
|
106
|
+
|
107
|
+
def run(_, postman, opts)
|
108
|
+
cmd = [
|
109
|
+
"svn log --diff --no-auth-cache",
|
110
|
+
"--username #{opts.decrypt('svn_user', 'svn_salt')}",
|
111
|
+
"--password #{opts.decrypt('svn_password', 'svn_salt')}",
|
112
|
+
"-r#{opts['since_rev']}:HEAD #{opts['svn_url']}"
|
113
|
+
]
|
114
|
+
stdout = `#{cmd.join(" ")}`
|
115
|
+
postman.send(opts.merge(stdout: stdout)) unless stdout.blank?
|
116
|
+
end
|
117
|
+
end
|
118
|
+
end
|
119
|
+
end
|
data/lib/lazylead/version.rb
CHANGED
@@ -0,0 +1,118 @@
|
|
1
|
+
<!DOCTYPE html>
|
2
|
+
<html lang="en">
|
3
|
+
<head>
|
4
|
+
<style> /* CSS styles taken from https://github.com/yegor256/tacit */
|
5
|
+
th {
|
6
|
+
font-weight: 600
|
7
|
+
}
|
8
|
+
|
9
|
+
table tr {
|
10
|
+
border-bottom-width: 2.16px
|
11
|
+
}
|
12
|
+
|
13
|
+
table tr th {
|
14
|
+
border-bottom-width: 2.16px
|
15
|
+
}
|
16
|
+
|
17
|
+
table tr td, table tr th {
|
18
|
+
overflow: hidden;
|
19
|
+
padding: 5.4px 3.6px;
|
20
|
+
line-height: 14px;
|
21
|
+
}
|
22
|
+
|
23
|
+
#summary {
|
24
|
+
text-align: left;
|
25
|
+
}
|
26
|
+
|
27
|
+
.auto {
|
28
|
+
min-width: auto;
|
29
|
+
white-space: nowrap;
|
30
|
+
}
|
31
|
+
|
32
|
+
a {
|
33
|
+
color: #275a90;
|
34
|
+
text-decoration: none
|
35
|
+
}
|
36
|
+
|
37
|
+
a:hover {
|
38
|
+
text-decoration: underline
|
39
|
+
}
|
40
|
+
|
41
|
+
* {
|
42
|
+
border: 0;
|
43
|
+
border-collapse: separate;
|
44
|
+
border-spacing: 0;
|
45
|
+
box-sizing: border-box;
|
46
|
+
margin: 0;
|
47
|
+
max-width: 100%;
|
48
|
+
padding: 0;
|
49
|
+
vertical-align: baseline;
|
50
|
+
font-family: system-ui, "Helvetica Neue", Helvetica, Arial, sans-serif;
|
51
|
+
font-size: 13px;
|
52
|
+
font-stretch: normal;
|
53
|
+
font-style: normal;
|
54
|
+
font-weight: 400;
|
55
|
+
line-height: 29.7px
|
56
|
+
}
|
57
|
+
|
58
|
+
html, body {
|
59
|
+
width: 100%
|
60
|
+
}
|
61
|
+
|
62
|
+
html {
|
63
|
+
height: 100%
|
64
|
+
}
|
65
|
+
|
66
|
+
body {
|
67
|
+
background: #fff;
|
68
|
+
color: #1a1919;
|
69
|
+
padding: 36px
|
70
|
+
}
|
71
|
+
</style>
|
72
|
+
<title>Accuracy</title>
|
73
|
+
</head>
|
74
|
+
<body>
|
75
|
+
<p>Hi,</p>
|
76
|
+
<p>The triage score and accuracy posted to the following tickets::</p>
|
77
|
+
<table summary="table with tickets triage score">
|
78
|
+
<tr>
|
79
|
+
<th id="key">Key</th>
|
80
|
+
<th id="duedate">Due date</th>
|
81
|
+
<th id="priority">Priority</th>
|
82
|
+
<th id="score">Score</th>
|
83
|
+
<th id="accuracy">Accuracy</th>
|
84
|
+
<th id="reporter">Reporter</th>
|
85
|
+
<th id="summary">Summary</th>
|
86
|
+
</tr>
|
87
|
+
<% tickets.sort_by { |s| s.issue.fields["priority"]["id"].to_i }
|
88
|
+
.each do |score| %>
|
89
|
+
<tr>
|
90
|
+
<td>
|
91
|
+
<div class="auto">
|
92
|
+
<a href="<%= score.issue.url %>"><%= score.issue.key %></a>
|
93
|
+
</div>
|
94
|
+
</td>
|
95
|
+
<td>
|
96
|
+
<div class="auto"><%= score.issue.duedate %></div>
|
97
|
+
</td>
|
98
|
+
<td><%= score.issue.priority %></td>
|
99
|
+
<td>
|
100
|
+
<span style="color: <%= score.color %>"><%= score.score %></span>
|
101
|
+
</td>
|
102
|
+
<td>
|
103
|
+
<span style="color: <%= score.color %>"><%= score.accuracy %>%</span>
|
104
|
+
</td>
|
105
|
+
<td>
|
106
|
+
<div class="auto">
|
107
|
+
<%= score.issue.reporter.name %>
|
108
|
+
</div>
|
109
|
+
</td>
|
110
|
+
<td><%= score.issue.summary %></td>
|
111
|
+
</tr>
|
112
|
+
<% end %>
|
113
|
+
</table>
|
114
|
+
<p>Posted by
|
115
|
+
<a href="https://github.com/dgroup/lazylead">lazylead v<%= version %></a>.
|
116
|
+
</p>
|
117
|
+
</body>
|
118
|
+
</html>
|