lazylead 0.3.1 → 0.5.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (77) hide show
  1. checksums.yaml +4 -4
  2. data/.0pdd.yml +4 -1
  3. data/.docs/accuracy.md +107 -0
  4. data/.docs/accuracy_email.jpg +0 -0
  5. data/.docs/accuracy_jira_comment.jpg +0 -0
  6. data/.docs/duedate_expired.md +3 -3
  7. data/.docs/propagate_down.md +4 -4
  8. data/.gitattributes +1 -0
  9. data/.github/dependabot.yml +6 -0
  10. data/.pdd +1 -1
  11. data/.rubocop.yml +6 -0
  12. data/Rakefile +2 -0
  13. data/bin/lazylead +7 -4
  14. data/lazylead.gemspec +5 -4
  15. data/lib/lazylead/exchange.rb +16 -9
  16. data/lib/lazylead/log.rb +30 -8
  17. data/lib/lazylead/model.rb +78 -22
  18. data/lib/lazylead/opts.rb +80 -0
  19. data/lib/lazylead/postman.rb +1 -1
  20. data/lib/lazylead/schedule.rb +18 -17
  21. data/lib/lazylead/smtp.rb +1 -1
  22. data/lib/lazylead/system/jira.rb +55 -14
  23. data/lib/lazylead/system/synced.rb +2 -1
  24. data/lib/lazylead/task/accuracy/accuracy.rb +136 -0
  25. data/lib/lazylead/task/accuracy/affected_build.rb +39 -0
  26. data/lib/lazylead/task/accuracy/attachment.rb +44 -0
  27. data/lib/lazylead/task/accuracy/environment.rb +39 -0
  28. data/lib/lazylead/task/accuracy/logs.rb +40 -0
  29. data/lib/lazylead/task/accuracy/records.rb +45 -0
  30. data/lib/lazylead/task/accuracy/requirement.rb +49 -0
  31. data/lib/lazylead/task/accuracy/servers.rb +50 -0
  32. data/lib/lazylead/task/accuracy/stacktrace.rb +63 -0
  33. data/lib/lazylead/task/accuracy/testcase.rb +75 -0
  34. data/lib/lazylead/task/accuracy/wiki.rb +41 -0
  35. data/lib/lazylead/task/alert.rb +8 -6
  36. data/lib/lazylead/task/confluence_ref.rb +4 -3
  37. data/lib/lazylead/task/echo.rb +22 -0
  38. data/lib/lazylead/task/fix_version.rb +18 -7
  39. data/lib/lazylead/task/missing_comment.rb +7 -5
  40. data/lib/lazylead/task/propagate_down.rb +11 -3
  41. data/lib/lazylead/task/savepoint.rb +1 -1
  42. data/lib/lazylead/task/touch.rb +119 -0
  43. data/lib/lazylead/version.rb +1 -1
  44. data/lib/messages/accuracy.erb +118 -0
  45. data/lib/messages/svn_log.erb +117 -0
  46. data/lib/messages/svn_touch.erb +147 -0
  47. data/license.txt +1 -1
  48. data/readme.md +20 -19
  49. data/test/lazylead/cc_test.rb +2 -2
  50. data/test/lazylead/cli/app_test.rb +12 -12
  51. data/test/lazylead/exchange_test.rb +3 -3
  52. data/test/lazylead/model_test.rb +4 -4
  53. data/test/lazylead/opts_test.rb +70 -0
  54. data/test/lazylead/postman_test.rb +1 -1
  55. data/test/lazylead/smtp_test.rb +1 -1
  56. data/test/lazylead/system/jira_test.rb +65 -1
  57. data/test/lazylead/task/accuracy/accuracy_test.rb +73 -0
  58. data/test/lazylead/task/accuracy/affected_build_test.rb +42 -0
  59. data/test/lazylead/task/accuracy/attachment_test.rb +50 -0
  60. data/test/lazylead/task/accuracy/environment_test.rb +42 -0
  61. data/test/lazylead/task/accuracy/logs_test.rb +78 -0
  62. data/test/lazylead/task/accuracy/records_test.rb +60 -0
  63. data/test/lazylead/task/accuracy/servers_test.rb +66 -0
  64. data/test/lazylead/task/accuracy/stacktrace_test.rb +113 -0
  65. data/test/lazylead/task/accuracy/testcase_test.rb +205 -0
  66. data/test/lazylead/task/accuracy/wiki_test.rb +40 -0
  67. data/test/lazylead/task/assignee_alert_test.rb +2 -2
  68. data/test/lazylead/task/duedate_test.rb +36 -26
  69. data/test/lazylead/task/fix_version_test.rb +9 -6
  70. data/test/lazylead/task/missing_comment_test.rb +11 -9
  71. data/test/lazylead/task/propagate_down_test.rb +4 -2
  72. data/test/lazylead/task/touch_test.rb +88 -0
  73. data/test/test.rb +25 -0
  74. data/upgrades/sqlite/001-install-main-lazylead-tables.sql +1 -5
  75. data/upgrades/sqlite/999.testdata.sql +12 -16
  76. metadata +65 -8
  77. data/.travis.yml +0 -16
@@ -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::NOTHING)
47
+ def initialize(log = Log.new)
47
48
  @log = log
48
49
  end
49
50
 
50
51
  def run(sys, postman, opts)
51
- postman.send opts.merge(tickets: sys.issues(opts["sql"]))
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::NOTHING)
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::NOTHING)
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::NOTHING)
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["confluences"].nil? || opts["confluences"].blank?
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
@@ -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::NOTHING)
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["allowed"].split(",").map(&:strip).reject(&:blank?)
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: sys.issues(opts["jql"], expand: "changelog")
42
- .map { |i| Version.new(i, allowed) }
43
- .select(&:changed?)
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::NOTHING)
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["details"].blank?
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: sys.issues(opts["jql"])
47
- .map { |i| Comments.new(i, sys) }
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::NOTHING)
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["fields"].split(",").map(&:strip).reject(&:blank?)
50
- sys.issues(opts["jql"], fields: ["subtasks"] + fields)
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)
@@ -34,7 +34,7 @@ module Lazylead
34
34
  # Send current configuration to admin user.
35
35
  #
36
36
  class Savepoint
37
- def initialize(log = Log::NOTHING)
37
+ def initialize(log = Log.new)
38
38
  @log = log
39
39
  end
40
40
 
@@ -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
@@ -23,5 +23,5 @@
23
23
  # OR OTHER DEALINGS IN THE SOFTWARE.
24
24
 
25
25
  module Lazylead
26
- VERSION = "0.3.1"
26
+ VERSION = "0.5.0"
27
27
  end
@@ -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>