lazylead 0.5.2 → 0.6.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: aa4ef974cb7b4799fb96acf04912513098f53c9adbb7f2bff4ea5bd8b498fb7e
4
- data.tar.gz: e9d8aea4a9fe5da7a63e35062cd7c0ff6c7670e1645d031dc27b6cf599d71608
3
+ metadata.gz: 50916d6916b46d8ea4c92fc6c4a02b75d6f9e35bbaad260057dd61bb0e25b650
4
+ data.tar.gz: d536e0d99c760a853d3a24f47d2b1a8fbece654790af65ec02fe7c3bf69af085
5
5
  SHA512:
6
- metadata.gz: 897ee5e71b3eb970246cdeca00179b27d0d906e8e457dc384a2b79e25ed1a746f785ebb5981a7deb458aaef78428c6ae63025e42579daea160f3463eb611b455
7
- data.tar.gz: 39850bcbb8e33239d8a5ab74a291f4ec4332067573ee3287775ad3c33c0210faf801358472ff9fc7b1dd531e6aedafc90e5393cf57ee04b32d5a6af120c02bc0
6
+ metadata.gz: da6131b9e02115b3b61c70c3a38665f563f4ff7d37f20b0939073f11c05549c061403e9fa1711897b26917b88f324e89dd42c64128a27b72b6ebbf0b134e1d20
7
+ data.tar.gz: bcc6287781baced0f0c0e0d4506fc1fa18439c98ff66a9254704620b537e9eb448daabdef646aa145ddf2d744c2f07f805ef717612bba0e4b86901be8ef33a85
@@ -32,7 +32,7 @@ Gem::Specification.new do |s|
32
32
  s.rubygems_version = "2.2"
33
33
  s.required_ruby_version = ">=2.6.5"
34
34
  s.name = "lazylead"
35
- s.version = "0.5.2"
35
+ s.version = "0.6.0"
36
36
  s.license = "MIT"
37
37
  s.summary = "Eliminate the annoying work within bug-trackers."
38
38
  s.description = "Ticketing systems (Github, Jira, etc.) are strongly
@@ -45,7 +45,7 @@ tasks instead of solving technical problems."
45
45
  s.authors = ["Yurii Dubinka"]
46
46
  s.email = "yurii.dubinka@gmail.com"
47
47
  s.homepage = "http://github.com/dgroup/lazylead"
48
- s.post_install_message = "Thanks for installing Lazylead v0.5.2!
48
+ s.post_install_message = "Thanks for installing Lazylead v0.6.0!
49
49
  Read our blog posts: https://lazylead.org
50
50
  Stay in touch with the community in Telegram: https://t.me/lazylead
51
51
  Follow us on Twitter: https://twitter.com/lazylead
@@ -71,6 +71,7 @@ tasks instead of solving technical problems."
71
71
  s.add_runtime_dependency "slop", "4.4"
72
72
  s.add_runtime_dependency "sqlite3", "1.4.2"
73
73
  s.add_runtime_dependency "tilt", "2.0.10"
74
+ s.add_runtime_dependency "tmpdir", "0.1.0"
74
75
  s.add_runtime_dependency "tzinfo", "1.1"
75
76
  s.add_runtime_dependency "tzinfo-data", "1.2019.3"
76
77
  s.add_runtime_dependency "vcs4sql", "0.1.0"
@@ -42,6 +42,7 @@ module Lazylead
42
42
 
43
43
  # Split text value by delimiter, trim all spaces and reject blank items
44
44
  def slice(key, delim)
45
+ return [] unless to_h.key? key
45
46
  to_h[key].split(delim).map(&:chomp).map(&:strip).reject(&:blank?)
46
47
  end
47
48
 
@@ -75,9 +75,10 @@ module Lazylead
75
75
  end
76
76
 
77
77
  def add_attachments(mail, opts)
78
- return unless opts.key? "attachments"
79
- opts["attachments"].select { |a| File.file? a }
80
- .each { |a| mail.add_file a }
78
+ return unless opts.key?("attachments") || opts.key?(:attachments)
79
+ attach = opts["attachments"] || opts[:attachments]
80
+ return if attach.nil?
81
+ attach.select { |a| File.file? a }.each { |a| mail.add_file a }
81
82
  end
82
83
  end
83
84
  end
@@ -25,6 +25,7 @@
25
25
  require "jira-ruby"
26
26
  require "forwardable"
27
27
  require_relative "../salt"
28
+ require_relative "../opts"
28
29
 
29
30
  module Lazylead
30
31
  # Jira system for manipulation with issues.
@@ -45,16 +46,32 @@ module Lazylead
45
46
  " and salt #{@salt.id} (found=#{@salt.specified?})"
46
47
  end
47
48
 
48
- def issues(jql, opts = {})
49
+ # Find the jira issues by 'JQL'
50
+ # @param 'jql' - Jira search query
51
+ # @param 'opts' - Parameters for Jira search query.
52
+ # :max_results => maximum number of tickets per one iteration.
53
+ # :fields => ticket fields to be fetched like assignee, summary, etc.
54
+ def issues(jql, opts = { max_results: 50, fields: nil, expand: nil })
49
55
  raw do |jira|
50
- jira.Issue.jql(jql, opts).map { |i| Lazylead::Issue.new(i, jira) }
56
+ start = 0
57
+ tickets = []
58
+ total = jira.Issue.jql(jql, max_results: 0)
59
+ @log.debug "Found #{total} ticket(s) in '#{jql}'"
60
+ loop do
61
+ tickets.concat(jira.Issue.jql(jql, opts.merge(start_at: start))
62
+ .map { |i| Lazylead::Issue.new(i, jira) })
63
+ @log.debug "Fetched #{tickets.size}"
64
+ start += opts.fetch(:max_results, 50).to_i
65
+ break if start > total
66
+ end
67
+ tickets
51
68
  end
52
69
  end
53
70
 
54
71
  # Execute request to the ticketing system using raw client.
55
72
  # For Jira the raw client is 'jira-ruby' gem.
56
73
  def raw
57
- raise "ll-06: No block given to method" unless block_given?
74
+ raise "ll-009: No block given to method" unless block_given?
58
75
  yield client
59
76
  end
60
77
 
@@ -325,7 +342,7 @@ module Lazylead
325
342
 
326
343
  # Execute request to the ticketing system using raw client.
327
344
  def raw
328
- raise "ll-08: No block given to method" unless block_given?
345
+ raise "ll-008: No block given to method" unless block_given?
329
346
  yield(OpenStruct.new(Issue: self))
330
347
  end
331
348
 
@@ -81,7 +81,7 @@ module Lazylead
81
81
  def post
82
82
  return if @opts.key? "silent"
83
83
  @issue.post comment
84
- @issue.add_label "LL.accuracy", grade(@accuracy)
84
+ @issue.add_label "LL.accuracy", "#{grade(@accuracy)}%"
85
85
  end
86
86
 
87
87
  # The jira comment in markdown format
@@ -31,10 +31,12 @@ module Lazylead
31
31
  super("Log files", 2, "Attachments")
32
32
  end
33
33
 
34
- # Ensure that ticket has a '*.log' file more '10KB'
34
+ # Ensure that ticket has a '*.log' file more '5KB'
35
35
  def matching(attachment)
36
- attachment.attrs["size"].to_i > 10_240 &&
37
- File.extname(attachment.attrs["filename"]).downcase.start_with?(".log")
36
+ name = attachment.attrs["filename"].downcase
37
+ return false unless attachment.attrs["size"].to_i > 5120
38
+ return true if File.extname(name).start_with?(".log", ".txt")
39
+ %w[.log.zip .log.gz .log.tar.gz].any? { |l| name.end_with? l }
38
40
  end
39
41
  end
40
42
  end
@@ -43,7 +43,7 @@ module Lazylead
43
43
  .reject(&:blank?)
44
44
  .map(&:strip)
45
45
  .flat_map { |l| l.split(" ").map(&:strip) }
46
- .select { |w| w.start_with?("http://", "https://") }
46
+ .select { |w| w.include?("http://") || w.include?("https://") }
47
47
  .any? { |u| @envs.any? { |e| u.match? e } }
48
48
  end
49
49
  end
@@ -34,24 +34,64 @@ module Lazylead
34
34
 
35
35
  def passed(issue)
36
36
  return false if issue.description.nil?
37
- !frames(issue.description).select { |f| oracle?(f) || java?(f) }.empty?
37
+ frames(issue.description).any? { |f| oracle?(f) || java?(f) }
38
38
  end
39
39
 
40
- # Detect all {noformat} frames in description field
40
+ # Detect all {noformat}, {code} frames in ticket description
41
41
  def frames(description)
42
- description.enum_for(:scan, /(?={noformat})/)
43
- .map { Regexp.last_match.offset(0).first }
44
- .each_slice(2).map do |f|
45
- description[f.first, f.last - f.first + "{noformat}".size]
42
+ noformat(description).concat(code(description))
43
+ end
44
+
45
+ # Detect all noformat blocks and give all text snippets in array
46
+ # @param desc The jira ticket description
47
+ def noformat(desc)
48
+ return [] unless desc.match?(/{(n|N)(o|O)(f|F)(o|O)(r|R)(m|M)(a|A)(t|T)}/)
49
+ desc.enum_for(:scan, /(?=\{(n|N)(o|O)(f|F)(o|O)(r|R)(m|M)(a|A)(t|T)})/)
50
+ .map { Regexp.last_match.offset(0).first }
51
+ .each_slice(2).map do |f|
52
+ desc[f.first, f.last - f.first + "{noformat}".size]
53
+ end
54
+ end
55
+
56
+ # Detect all {code:*} blocks and give all text snippets in array
57
+ # @param desc The jira ticket description
58
+ def code(desc)
59
+ return [] unless desc.match?(/{(c|C)(o|O)(d|D)(e|E)(:\S+)?}/)
60
+ words = desc.gsub(/{(c|C)(o|O)(d|D)(e|E)/, " {code")
61
+ .gsub("}", "} ")
62
+ .gsub("Caused by:", "Caused_by:")
63
+ .split(" ")
64
+ .map(&:strip)
65
+ .reject(&:blank?)
66
+ pairs(words, "{code").map { |s| words[s.first..s.last].join("\n") }
67
+ end
68
+
69
+ # Detect indexes of pairs which are starting from particular text
70
+ # @param words is array of words
71
+ # @param text is a label for pairs
72
+ #
73
+ # paris([aa,tag,bb,cc,tag,dd], "tag") => [[1, 4]]
74
+ # paris([aa,tag,bb,cc,tag,dd,tag,ee], "tag") => [[1, 4]] # non closed
75
+ #
76
+ def pairs(words, text)
77
+ snippets = [[]]
78
+ words.each_with_index do |e, i|
79
+ next unless e.start_with? text
80
+ pair = snippets.last
81
+ pair << i if pair.size.zero? || pair.size == 1
82
+ snippets[-1] = pair
83
+ snippets << [] if pair.size == 2
46
84
  end
85
+ snippets.select { |s| s.size == 2 }
47
86
  end
48
87
 
49
88
  # @return true if frame has few lines with java stack frames
50
89
  def java?(frame)
51
- allowed = ["at ", "Caused by:"]
52
- frame.split("\n")
53
- .map(&:strip)
54
- .count { |l| allowed.any? { |a| l.start_with? a } } > 3
90
+ allowed = ["at", "Caused by:", "Caused_by:"]
91
+ frame.match?(/\s\S+\.\S+Exception:\s/) ||
92
+ frame.split("\n")
93
+ .map(&:strip)
94
+ .count { |l| allowed.any? { |a| l.start_with? a } } > 3
55
95
  end
56
96
 
57
97
  # @return true if frame has Oracle error
@@ -38,7 +38,7 @@ module Lazylead
38
38
  return false if issue.description.nil?
39
39
  @tc = @ar = @er = -1
40
40
  issue.description.split("\n").reject(&:blank?).each_with_index do |l, i|
41
- line = escape(l.strip.downcase)
41
+ line = escape(l.downcase.gsub(/\s+/, ""))
42
42
  detect_tc(line, i)
43
43
  detect_ar(line, i)
44
44
  detect_er(line, i)
@@ -65,21 +65,27 @@ module Lazylead
65
65
  # Detect index of line with test case
66
66
  def detect_tc(line, index)
67
67
  return unless @tc.negative?
68
- @tc = index if %w[testcase: tc: teststeps: teststeps steps:].any? do |e|
69
- e.eql? line
70
- end
68
+ @tc = index if eql?(line, %w[testcase: tc: teststeps: teststeps steps:])
71
69
  end
72
70
 
73
71
  # Detect index of line with actual result
74
72
  def detect_ar(line, index)
75
73
  return unless @ar.negative? && index > @tc
76
- @ar = index if %w[ar: actualresult: ar=].any? { |e| line.start_with? e }
74
+ @ar = index if starts?(line, %w[ar: actualresult: ar= *ar*= *ar*:])
77
75
  end
78
76
 
79
77
  # Detect index of line with expected result
80
78
  def detect_er(line, index)
81
79
  return unless @er.negative? && index > @tc
82
- @er = index if %w[er: expectedresult: er=].any? { |e| line.start_with? e }
80
+ @er = index if starts?(line, %w[er: expectedresult: er= *er*= *er*:])
81
+ end
82
+
83
+ def starts?(line, text)
84
+ text.any? { |t| line.start_with? t }
85
+ end
86
+
87
+ def eql?(line, text)
88
+ text.any? { |t| t.eql? line }
83
89
  end
84
90
  end
85
91
  end
@@ -0,0 +1,96 @@
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 "date"
26
+ require_relative "../log"
27
+ require_relative "../opts"
28
+ require_relative "../system/jira"
29
+
30
+ module Lazylead
31
+ module Task
32
+ #
33
+ # Email alerts about illegal issue assignment(s).
34
+ #
35
+ # Author:: Yurii Dubinka (yurii.dubinka@gmail.com)
36
+ # Copyright:: Copyright (c) 2019-2020 Yurii Dubinka
37
+ # License:: MIT
38
+ class Assignment
39
+ def initialize(log = Log.new)
40
+ @log = log
41
+ end
42
+
43
+ def run(sys, postman, opts)
44
+ allowed = opts.slice "allowed", ","
45
+ silent = opts.key? "silent"
46
+ issues = sys.issues opts["jql"],
47
+ opts.jira_defaults.merge(expand: "changelog")
48
+ return if issues.empty?
49
+ postman.send opts.merge(
50
+ assignees: issues.map { |i| Assignee.new(i, allowed, silent) }
51
+ .select(&:illegal?)
52
+ .each(&:add_label)
53
+ )
54
+ end
55
+ end
56
+
57
+ # Instance of "Assignee" history item for the particular ticket.
58
+ class Assignee
59
+ attr_reader :issue
60
+
61
+ def initialize(issue, allowed, silent)
62
+ @issue = issue
63
+ @allowed = allowed
64
+ @silent = silent
65
+ end
66
+
67
+ # Gives true when last change of "Assignee" field was done
68
+ # by not authorized person.
69
+ def illegal?
70
+ @allowed.none? do |a|
71
+ return false if last.nil?
72
+ a == last["author"]["name"]
73
+ end
74
+ end
75
+
76
+ # Detect details about last change of "Assignee" to non-null value
77
+ def last
78
+ @last ||= issue.history.reverse.find do |h|
79
+ h["items"].any? do |i|
80
+ i["field"] == "assignee"
81
+ end
82
+ end
83
+ end
84
+
85
+ # Mark ticket with particular label
86
+ def add_label(name = "LL.IllegalChangeOfAssignee")
87
+ @issue.add_label(name) unless @silent
88
+ end
89
+
90
+ # The name of current assignee for ticket
91
+ def to
92
+ @issue.fields["assignee"]["name"]
93
+ end
94
+ end
95
+ end
96
+ end
@@ -85,6 +85,10 @@ module Lazylead
85
85
  def add_label
86
86
  @issue.add_label("LL.IllegalChangeOfFixVersion") unless @silent
87
87
  end
88
+
89
+ def current
90
+ @issue.fields["fixVersions"].first["name"]
91
+ end
88
92
  end
89
93
  end
90
94
  end
@@ -0,0 +1,76 @@
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 "tmpdir"
26
+ require "nokogiri"
27
+ require "active_support/core_ext/hash/conversions"
28
+ require_relative "../../salt"
29
+ require_relative "../../opts"
30
+
31
+ module Lazylead
32
+ module Task
33
+ module Svn
34
+ #
35
+ # Send notification about modification of svn files since particular
36
+ # revision.
37
+ #
38
+ class Diff
39
+ def initialize(log = Log.new)
40
+ @log = log
41
+ end
42
+
43
+ def run(_, postman, opts)
44
+ cmd = [
45
+ "svn log --diff --no-auth-cache",
46
+ "--username #{opts.decrypt('svn_user', 'svn_salt')}",
47
+ "--password #{opts.decrypt('svn_password', 'svn_salt')}",
48
+ "-r#{opts['since_rev']}:HEAD #{opts['svn_url']}"
49
+ ]
50
+ stdout = `#{cmd.join(" ")}`
51
+ send_email stdout, postman, opts unless stdout.blank?
52
+ end
53
+
54
+ # Send email with svn log as an attachment.
55
+ # The attachment won't be stored locally and we'll be removed once
56
+ # mail sent.
57
+ def send_email(stdout, postman, opts)
58
+ Dir.mktmpdir do |dir|
59
+ name = "svn-log-#{Date.today.strftime('%d-%b-%Y')}.html"
60
+ f = File.open(File.join(dir, name), "w")
61
+ f.write(
62
+ Email.new(
63
+ opts["template-attachment"],
64
+ opts.merge(stdout: stdout, version: Lazylead::VERSION)
65
+ ).render
66
+ )
67
+ postman.send opts.merge(stdout: stdout, attachments: [f.path])
68
+ rescue StandardError => e
69
+ @log.error "ll-010: Can't send an email for #{opts} based on "\
70
+ "'#{stdout}'", e
71
+ end
72
+ end
73
+ end
74
+ end
75
+ end
76
+ end