debian-mirror-gitlab/app/models/project_services/teamcity_service.rb

191 lines
5 KiB
Ruby
Raw Normal View History

2018-11-20 20:47:30 +05:30
# frozen_string_literal: true
2015-04-26 12:48:37 +05:30
class TeamcityService < CiService
2017-08-17 22:00:37 +05:30
include ReactiveService
2019-09-30 21:07:59 +05:30
include ServicePushDataValidations
2017-08-17 22:00:37 +05:30
2015-04-26 12:48:37 +05:30
prop_accessor :teamcity_url, :build_type, :username, :password
2018-11-08 19:23:39 +05:30
validates :teamcity_url, presence: true, public_url: true, if: :activated?
2015-04-26 12:48:37 +05:30
validates :build_type, presence: true, if: :activated?
validates :username,
presence: true,
2015-12-23 02:04:40 +05:30
if: ->(service) { service.activated? && service.password }
2015-04-26 12:48:37 +05:30
validates :password,
presence: true,
2015-12-23 02:04:40 +05:30
if: ->(service) { service.activated? && service.username }
2015-04-26 12:48:37 +05:30
attr_accessor :response
after_save :compose_service_hook, if: :activated?
2015-10-24 18:46:33 +05:30
before_update :reset_password
2015-04-26 12:48:37 +05:30
2019-09-30 21:07:59 +05:30
class << self
def to_param
'teamcity'
end
def supported_events
%w(push merge_request)
end
def event_description(event)
case event
when 'push', 'push_events'
'TeamCity CI will be triggered after every push to the repository except branch delete'
when 'merge_request', 'merge_request_events'
'TeamCity CI will be triggered after a merge request has been created or updated'
end
end
end
2015-04-26 12:48:37 +05:30
def compose_service_hook
hook = service_hook || build_service_hook
hook.save
end
2015-10-24 18:46:33 +05:30
def reset_password
if teamcity_url_changed? && !password_touched?
self.password = nil
end
end
2015-04-26 12:48:37 +05:30
def title
2021-04-29 21:17:54 +05:30
'JetBrains TeamCity'
2015-04-26 12:48:37 +05:30
end
def description
2021-04-29 21:17:54 +05:30
s_('ProjectService|Run CI/CD pipelines with JetBrains TeamCity.')
2015-04-26 12:48:37 +05:30
end
def help
2021-04-29 21:17:54 +05:30
s_('To run CI/CD pipelines with JetBrains TeamCity, input the GitLab project details in the TeamCity project Version Control Settings.')
2015-04-26 12:48:37 +05:30
end
def fields
[
2021-04-29 21:17:54 +05:30
{
type: 'text',
name: 'teamcity_url',
title: s_('ProjectService|TeamCity server URL'),
placeholder: 'https://teamcity.example.com',
required: true
},
{
type: 'text',
name: 'build_type',
help: s_('ProjectService|The build configuration ID of the TeamCity project.'),
required: true
},
{
type: 'text',
name: 'username',
help: s_('ProjectService|Must have permission to trigger a manual build in TeamCity.')
},
{
type: 'password',
name: 'password',
non_empty_password_title: s_('ProjectService|Enter new password'),
non_empty_password_help: s_('ProjectService|Leave blank to use your current password')
}
2015-04-26 12:48:37 +05:30
]
end
def build_page(sha, ref)
2017-08-17 22:00:37 +05:30
with_reactive_cache(sha, ref) {|cached| cached[:build_page] }
2015-04-26 12:48:37 +05:30
end
def commit_status(sha, ref)
2017-08-17 22:00:37 +05:30
with_reactive_cache(sha, ref) {|cached| cached[:commit_status] }
end
2015-04-26 12:48:37 +05:30
2017-08-17 22:00:37 +05:30
def calculate_reactive_cache(sha, ref)
2019-03-02 22:35:43 +05:30
response = get_path("httpAuth/app/rest/builds/branch:unspecified:any,revision:#{sha}")
2015-04-26 12:48:37 +05:30
2020-04-22 19:07:51 +05:30
if response
{ build_page: read_build_page(response), commit_status: read_commit_status(response) }
else
{ build_page: teamcity_url, commit_status: :error }
end
2015-04-26 12:48:37 +05:30
end
def execute(data)
2019-09-30 21:07:59 +05:30
case data[:object_kind]
when 'push'
execute_push(data)
when 'merge_request'
execute_merge_request(data)
end
end
2015-04-26 12:48:37 +05:30
2019-09-30 21:07:59 +05:30
private
2015-04-26 12:48:37 +05:30
2019-09-30 21:07:59 +05:30
def execute_push(data)
2015-04-26 12:48:37 +05:30
branch = Gitlab::Git.ref_name(data[:ref])
2019-09-30 21:07:59 +05:30
post_to_build_queue(data, branch) if push_valid?(data)
2015-04-26 12:48:37 +05:30
end
2019-09-30 21:07:59 +05:30
def execute_merge_request(data)
branch = data[:object_attributes][:source_branch]
post_to_build_queue(data, branch) if merge_request_valid?(data)
end
2017-08-17 22:00:37 +05:30
def read_build_page(response)
if response.code != 200
# If actual build link can't be determined,
# send user to build summary page.
build_url("viewLog.html?buildTypeId=#{build_type}")
else
# If actual build link is available, go to build result page.
built_id = response['build']['id']
build_url("viewLog.html?buildId=#{built_id}&buildTypeId=#{build_type}")
end
end
def read_commit_status(response)
return :error unless response.code == 200 || response.code == 404
status = if response.code == 404
'Pending'
else
response['build']['status']
end
return :error unless status.present?
if status.include?('SUCCESS')
'success'
elsif status.include?('FAILURE')
'failed'
elsif status.include?('Pending')
'pending'
else
:error
end
end
def build_url(path)
2019-02-15 15:39:39 +05:30
Gitlab::Utils.append_path(teamcity_url, path)
end
def get_path(path)
2021-07-02 01:05:55 +05:30
Gitlab::HTTP.try_get(build_url(path), verify: false, basic_auth: basic_auth, extra_log_info: { project_id: project_id }, use_read_total_timeout: true)
2019-09-30 21:07:59 +05:30
end
def post_to_build_queue(data, branch)
Gitlab::HTTP.post(
build_url('httpAuth/app/rest/buildQueue'),
body: "<build branchName=#{branch.encode(xml: :attr)}>"\
"<buildType id=#{build_type.encode(xml: :attr)}/>"\
'</build>',
headers: { 'Content-type' => 'application/xml' },
2021-07-02 01:05:55 +05:30
basic_auth: basic_auth,
use_read_total_timeout: true
2019-09-30 21:07:59 +05:30
)
end
def basic_auth
{ username: username, password: password }
end
2015-04-26 12:48:37 +05:30
end