2021-05-10 14:10:41 -04:00
# frozen_string_literal: true
module Integrations
2021-05-25 14:10:42 -04:00
class Bamboo < BaseCi
2021-05-10 14:10:41 -04:00
include ActionView :: Helpers :: UrlHelper
include ReactiveService
prop_accessor :bamboo_url , :build_key , :username , :password
validates :bamboo_url , presence : true , public_url : true , if : :activated?
validates :build_key , presence : true , if : :activated?
validates :username ,
presence : true ,
if : - > ( service ) { service . activated? && service . password }
validates :password ,
presence : true ,
if : - > ( service ) { service . activated? && service . username }
attr_accessor :response
2021-07-21 08:09:35 -04:00
before_validation :reset_password
2021-05-10 14:10:41 -04:00
def reset_password
if bamboo_url_changed? && ! password_touched?
self . password = nil
end
end
def title
s_ ( 'BambooService|Atlassian Bamboo' )
end
def description
s_ ( 'BambooService|Run CI/CD pipelines with Atlassian Bamboo.' )
end
def help
docs_link = link_to _ ( 'Learn more.' ) , Rails . application . routes . url_helpers . help_page_url ( 'user/project/integrations/bamboo' ) , target : '_blank' , rel : 'noopener noreferrer'
s_ ( 'BambooService|Run CI/CD pipelines with Atlassian Bamboo. You must set up automatic revision labeling and a repository trigger in Bamboo. %{docs_link}' ) . html_safe % { docs_link : docs_link . html_safe }
end
def self . to_param
'bamboo'
end
def fields
[
{
type : 'text' ,
name : 'bamboo_url' ,
title : s_ ( 'BambooService|Bamboo URL' ) ,
placeholder : s_ ( 'https://bamboo.example.com' ) ,
help : s_ ( 'BambooService|Bamboo service root URL.' ) ,
required : true
} ,
{
type : 'text' ,
name : 'build_key' ,
placeholder : s_ ( 'KEY' ) ,
help : s_ ( 'BambooService|Bamboo build plan key.' ) ,
required : true
} ,
{
type : 'text' ,
name : 'username' ,
help : s_ ( 'BambooService|The user with API access to the Bamboo server.' )
} ,
{
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' )
}
]
end
def build_page ( sha , ref )
with_reactive_cache ( sha , ref ) { | cached | cached [ :build_page ] }
end
def commit_status ( sha , ref )
with_reactive_cache ( sha , ref ) { | cached | cached [ :commit_status ] }
end
def execute ( data )
return unless supported_events . include? ( data [ :object_kind ] )
get_path ( " updateAndBuild.action " , { buildKey : build_key } )
end
def calculate_reactive_cache ( sha , ref )
response = try_get_path ( " rest/api/latest/result/byChangeset/ #{ sha } " )
{ build_page : read_build_page ( response ) , commit_status : read_commit_status ( response ) }
end
private
def get_build_result ( response )
return if response & . code != 200
# May be nil if no result, a single result hash, or an array if multiple results for a given changeset.
result = response . dig ( 'results' , 'results' , 'result' )
# In case of multiple results, arbitrarily assume the last one is the most relevant.
return result . last if result . is_a? ( Array )
result
end
def read_build_page ( response )
result = get_build_result ( response )
key =
if result . blank?
# If actual build link can't be determined, send user to build summary page.
build_key
else
# If actual build link is available, go to build result page.
result . dig ( 'planResultKey' , 'key' )
end
build_url ( " browse/ #{ key } " )
end
def read_commit_status ( response )
return :error unless response && ( response . code == 200 || response . code == 404 )
result = get_build_result ( response )
status =
if result . blank?
'Pending'
else
result . dig ( 'buildState' )
end
return :error unless status . present?
if status . include? ( 'Success' )
'success'
elsif status . include? ( 'Failed' )
'failed'
elsif status . include? ( 'Pending' )
'pending'
else
:error
end
end
def try_get_path ( path , query_params = { } )
params = build_get_params ( query_params )
params [ :extra_log_info ] = { project_id : project_id }
Gitlab :: HTTP . try_get ( build_url ( path ) , params )
end
def get_path ( path , query_params = { } )
Gitlab :: HTTP . get ( build_url ( path ) , build_get_params ( query_params ) )
end
def build_url ( path )
Gitlab :: Utils . append_path ( bamboo_url , path )
end
def build_get_params ( query_params )
params = { verify : false , query : query_params }
return params if username . blank? && password . blank?
query_params [ :os_authType ] = 'basic'
params [ :basic_auth ] = basic_auth
2021-07-01 17:08:38 -04:00
params [ :use_read_total_timeout ] = true
2021-05-10 14:10:41 -04:00
params
end
def basic_auth
{ username : username , password : password }
end
end
end