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

144 lines
3.7 KiB
Ruby
Raw Normal View History

2015-04-26 12:48:37 +05:30
class TeamcityService < CiService
2017-08-17 22:00:37 +05:30
include ReactiveService
2015-04-26 12:48:37 +05:30
prop_accessor :teamcity_url, :build_type, :username, :password
2015-12-23 02:04:40 +05:30
validates :teamcity_url, presence: true, 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
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
'JetBrains TeamCity CI'
end
def description
'A continuous integration and build server'
end
def help
'The build configuration in Teamcity must use the build format '\
'number %build.vcs.number% '\
'you will also want to configure monitoring of all branches so merge '\
'requests build, that setting is in the vsc root advanced settings.'
end
2017-08-17 22:00:37 +05:30
def self.to_param
2015-04-26 12:48:37 +05:30
'teamcity'
end
def fields
[
{ type: 'text', name: 'teamcity_url',
placeholder: 'TeamCity root URL like https://teamcity.example.com' },
{ type: 'text', name: 'build_type',
placeholder: 'Build configuration ID' },
{ type: 'text', name: 'username',
placeholder: 'A user with permissions to trigger a manual build' },
{ type: 'password', name: 'password' },
]
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)
response = get_path("httpAuth/app/rest/builds/branch:unspecified:any,number:#{sha}")
2015-04-26 12:48:37 +05:30
2017-08-17 22:00:37 +05:30
{ build_page: read_build_page(response), commit_status: read_commit_status(response) }
2015-04-26 12:48:37 +05:30
end
def execute(data)
return unless supported_events.include?(data[:object_kind])
auth = {
username: username,
password: password,
}
branch = Gitlab::Git.ref_name(data[:ref])
HTTParty.post(
build_url('httpAuth/app/rest/buildQueue'),
2016-06-02 11:05:42 +05:30
body: "<build branchName=\"#{branch}\">"\
"<buildType id=\"#{build_type}\"/>"\
'</build>',
headers: { 'Content-type' => 'application/xml' },
basic_auth: auth
)
2015-04-26 12:48:37 +05:30
end
private
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)
URI.join("#{teamcity_url}/", path).to_s
end
def get_path(path)
HTTParty.get(build_url(path), verify: false,
basic_auth: {
username: username,
password: password
})
end
2015-04-26 12:48:37 +05:30
end