2015-01-20 19:46:27 -05:00
|
|
|
class JiraService < IssueTrackerService
|
2015-12-17 17:08:14 -05:00
|
|
|
include HTTParty
|
2016-03-24 12:00:26 -04:00
|
|
|
include Gitlab::Routing.url_helpers
|
2015-01-20 19:46:27 -05:00
|
|
|
|
2015-12-17 17:08:14 -05:00
|
|
|
DEFAULT_API_VERSION = 2
|
|
|
|
|
|
|
|
prop_accessor :username, :password, :api_url, :jira_issue_transition_id,
|
|
|
|
:title, :description, :project_url, :issues_url, :new_issue_url
|
|
|
|
|
2016-04-21 11:13:14 -04:00
|
|
|
validates :api_url, presence: true, url: true, if: :activated?
|
|
|
|
|
2015-12-17 17:08:14 -05:00
|
|
|
before_validation :set_api_url, :set_jira_issue_transition_id
|
|
|
|
|
|
|
|
before_update :reset_password
|
|
|
|
|
|
|
|
def reset_password
|
|
|
|
# don't reset the password if a new one is provided
|
|
|
|
if api_url_changed? && !password_touched?
|
|
|
|
self.password = nil
|
|
|
|
end
|
|
|
|
end
|
2015-01-20 19:46:27 -05:00
|
|
|
|
2015-02-12 20:06:55 -05:00
|
|
|
def help
|
2016-01-08 09:22:42 -05:00
|
|
|
'Setting `project_url`, `issues_url` and `new_issue_url` will '\
|
2015-03-27 15:57:53 -04:00
|
|
|
'allow a user to easily navigate to the Jira issue tracker. See the '\
|
|
|
|
'[integration doc](http://doc.gitlab.com/ce/integration/external-issue-tracker.html) '\
|
|
|
|
'for details.'
|
2015-02-12 20:06:55 -05:00
|
|
|
end
|
|
|
|
|
2015-01-20 19:46:27 -05:00
|
|
|
def title
|
|
|
|
if self.properties && self.properties['title'].present?
|
|
|
|
self.properties['title']
|
|
|
|
else
|
|
|
|
'JIRA'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def description
|
|
|
|
if self.properties && self.properties['description'].present?
|
|
|
|
self.properties['description']
|
|
|
|
else
|
|
|
|
'Jira issue tracker'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_param
|
|
|
|
'jira'
|
|
|
|
end
|
2015-12-17 17:08:14 -05:00
|
|
|
|
|
|
|
def fields
|
|
|
|
super.push(
|
|
|
|
{ type: 'text', name: 'api_url', placeholder: 'https://jira.example.com/rest/api/2' },
|
|
|
|
{ type: 'text', name: 'username', placeholder: '' },
|
|
|
|
{ type: 'password', name: 'password', placeholder: '' },
|
|
|
|
{ type: 'text', name: 'jira_issue_transition_id', placeholder: '2' }
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def execute(push, issue = nil)
|
|
|
|
if issue.nil?
|
|
|
|
# No specific issue, that means
|
|
|
|
# we just want to test settings
|
|
|
|
test_settings
|
|
|
|
else
|
|
|
|
close_issue(push, issue)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_cross_reference_note(mentioned, noteable, author)
|
|
|
|
issue_name = mentioned.id
|
|
|
|
project = self.project
|
|
|
|
noteable_name = noteable.class.name.underscore.downcase
|
|
|
|
noteable_id = if noteable.is_a?(Commit)
|
|
|
|
noteable.id
|
|
|
|
else
|
|
|
|
noteable.iid
|
|
|
|
end
|
|
|
|
|
|
|
|
entity_url = build_entity_url(noteable_name.to_sym, noteable_id)
|
|
|
|
|
|
|
|
data = {
|
|
|
|
user: {
|
|
|
|
name: author.name,
|
|
|
|
url: resource_url(user_path(author)),
|
|
|
|
},
|
|
|
|
project: {
|
|
|
|
name: project.path_with_namespace,
|
|
|
|
url: resource_url(namespace_project_path(project.namespace, project))
|
|
|
|
},
|
|
|
|
entity: {
|
|
|
|
name: noteable_name.humanize.downcase,
|
2016-03-02 13:12:16 -05:00
|
|
|
url: entity_url,
|
|
|
|
title: noteable.title
|
2015-12-17 17:08:14 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
add_comment(data, issue_name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_settings
|
2016-01-07 16:07:13 -05:00
|
|
|
return unless api_url.present?
|
2015-12-17 17:08:14 -05:00
|
|
|
result = JiraService.get(
|
|
|
|
jira_api_test_url,
|
|
|
|
headers: {
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
'Authorization' => "Basic #{auth}"
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
case result.code
|
|
|
|
when 201, 200
|
|
|
|
Rails.logger.info("#{self.class.name} SUCCESS #{result.code}: Successfully connected to #{api_url}.")
|
|
|
|
true
|
|
|
|
else
|
|
|
|
Rails.logger.info("#{self.class.name} ERROR #{result.code}: #{result.parsed_response}")
|
|
|
|
false
|
|
|
|
end
|
|
|
|
rescue Errno::ECONNREFUSED => e
|
|
|
|
Rails.logger.info "#{self.class.name} ERROR: #{e.message}. API URL: #{api_url}."
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def build_api_url_from_project_url
|
|
|
|
server = URI(project_url)
|
2016-06-29 09:23:44 -04:00
|
|
|
default_ports = [["http", 80], ["https", 443]].include?([server.scheme, server.port])
|
2015-12-17 17:08:14 -05:00
|
|
|
server_url = "#{server.scheme}://#{server.host}"
|
|
|
|
server_url.concat(":#{server.port}") unless default_ports
|
|
|
|
"#{server_url}/rest/api/#{DEFAULT_API_VERSION}"
|
|
|
|
rescue
|
|
|
|
"" # looks like project URL was not valid
|
|
|
|
end
|
|
|
|
|
|
|
|
def set_api_url
|
|
|
|
self.api_url = build_api_url_from_project_url if self.api_url.blank?
|
|
|
|
end
|
|
|
|
|
|
|
|
def set_jira_issue_transition_id
|
|
|
|
self.jira_issue_transition_id ||= "2"
|
|
|
|
end
|
|
|
|
|
|
|
|
def close_issue(entity, issue)
|
|
|
|
commit_id = if entity.is_a?(Commit)
|
|
|
|
entity.id
|
|
|
|
elsif entity.is_a?(MergeRequest)
|
2016-06-20 12:48:04 -04:00
|
|
|
entity.diff_head_sha
|
2015-12-17 17:08:14 -05:00
|
|
|
end
|
|
|
|
commit_url = build_entity_url(:commit, commit_id)
|
|
|
|
|
|
|
|
# Depending on the JIRA project's workflow, a comment during transition
|
|
|
|
# may or may not be allowed. Split the operation in to two calls so the
|
|
|
|
# comment always works.
|
|
|
|
transition_issue(issue)
|
|
|
|
add_issue_solved_comment(issue, commit_id, commit_url)
|
|
|
|
end
|
|
|
|
|
|
|
|
def transition_issue(issue)
|
|
|
|
message = {
|
|
|
|
transition: {
|
|
|
|
id: jira_issue_transition_id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
send_message(close_issue_url(issue.iid), message.to_json)
|
|
|
|
end
|
|
|
|
|
|
|
|
def add_issue_solved_comment(issue, commit_id, commit_url)
|
|
|
|
comment = {
|
|
|
|
body: "Issue solved with [#{commit_id}|#{commit_url}]."
|
|
|
|
}
|
|
|
|
|
|
|
|
send_message(comment_url(issue.iid), comment.to_json)
|
|
|
|
end
|
|
|
|
|
|
|
|
def add_comment(data, issue_name)
|
|
|
|
url = comment_url(issue_name)
|
|
|
|
user_name = data[:user][:name]
|
|
|
|
user_url = data[:user][:url]
|
|
|
|
entity_name = data[:entity][:name]
|
|
|
|
entity_url = data[:entity][:url]
|
2016-03-02 13:12:16 -05:00
|
|
|
entity_title = data[:entity][:title]
|
2015-12-17 17:08:14 -05:00
|
|
|
project_name = data[:project][:name]
|
|
|
|
|
|
|
|
message = {
|
2016-03-02 13:12:16 -05:00
|
|
|
body: %Q{[#{user_name}|#{user_url}] mentioned this issue in [a #{entity_name} of #{project_name}|#{entity_url}]:\n'#{entity_title}'}
|
2015-12-17 17:08:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
unless existing_comment?(issue_name, message[:body])
|
|
|
|
send_message(url, message.to_json)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def auth
|
|
|
|
require 'base64'
|
|
|
|
Base64.urlsafe_encode64("#{self.username}:#{self.password}")
|
|
|
|
end
|
|
|
|
|
|
|
|
def send_message(url, message)
|
2016-01-07 16:07:13 -05:00
|
|
|
return unless api_url.present?
|
2015-12-17 17:08:14 -05:00
|
|
|
result = JiraService.post(
|
|
|
|
url,
|
|
|
|
body: message,
|
|
|
|
headers: {
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
'Authorization' => "Basic #{auth}"
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
message = case result.code
|
|
|
|
when 201, 200, 204
|
|
|
|
"#{self.class.name} SUCCESS #{result.code}: Successfully posted to #{url}."
|
|
|
|
when 401
|
|
|
|
"#{self.class.name} ERROR 401: Unauthorized. Check the #{self.username} credentials and JIRA access permissions and try again."
|
|
|
|
else
|
|
|
|
"#{self.class.name} ERROR #{result.code}: #{result.parsed_response}"
|
|
|
|
end
|
|
|
|
|
|
|
|
Rails.logger.info(message)
|
|
|
|
message
|
|
|
|
rescue URI::InvalidURIError, Errno::ECONNREFUSED => e
|
|
|
|
Rails.logger.info "#{self.class.name} ERROR: #{e.message}. Hostname: #{url}."
|
|
|
|
end
|
|
|
|
|
|
|
|
def existing_comment?(issue_name, new_comment)
|
2016-01-07 16:07:13 -05:00
|
|
|
return unless api_url.present?
|
2015-12-17 17:08:14 -05:00
|
|
|
result = JiraService.get(
|
|
|
|
comment_url(issue_name),
|
|
|
|
headers: {
|
|
|
|
'Content-Type' => 'application/json',
|
|
|
|
'Authorization' => "Basic #{auth}"
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
case result.code
|
|
|
|
when 201, 200
|
|
|
|
existing_comments = JSON.parse(result.body)['comments']
|
|
|
|
|
|
|
|
if existing_comments.present?
|
|
|
|
return existing_comments.map { |comment| comment['body'].include?(new_comment) }.any?
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
false
|
|
|
|
rescue JSON::ParserError
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
|
|
|
def resource_url(resource)
|
|
|
|
"#{Settings.gitlab['url'].chomp("/")}#{resource}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def build_entity_url(entity_name, entity_id)
|
|
|
|
resource_url(
|
|
|
|
polymorphic_url(
|
|
|
|
[
|
|
|
|
self.project.namespace.becomes(Namespace),
|
|
|
|
self.project,
|
|
|
|
entity_name
|
|
|
|
],
|
|
|
|
id: entity_id,
|
|
|
|
routing_type: :path
|
|
|
|
)
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def close_issue_url(issue_name)
|
|
|
|
"#{self.api_url}/issue/#{issue_name}/transitions"
|
|
|
|
end
|
|
|
|
|
|
|
|
def comment_url(issue_name)
|
|
|
|
"#{self.api_url}/issue/#{issue_name}/comment"
|
|
|
|
end
|
|
|
|
|
|
|
|
def jira_api_test_url
|
|
|
|
"#{self.api_url}/myself"
|
|
|
|
end
|
2015-01-20 19:46:27 -05:00
|
|
|
end
|