teamcity_service.rb 3.77 KB
Newer Older
1
class TeamcityService < CiService
2 3
  include ReactiveService

4 5
  prop_accessor :teamcity_url, :build_type, :username, :password

Robert Speicher's avatar
Robert Speicher committed
6
  validates :teamcity_url, presence: true, url: true, if: :activated?
7
  validates :build_type, presence: true, if: :activated?
8 9
  validates :username,
    presence: true,
10
    if: ->(service) { service.activated? && service.password }
11 12
  validates :password,
    presence: true,
13
    if: ->(service) { service.activated? && service.username }
14 15 16 17

  attr_accessor :response

  after_save :compose_service_hook, if: :activated?
18
  before_update :reset_password
19 20 21 22 23 24

  def compose_service_hook
    hook = service_hook || build_service_hook
    hook.save
  end

25
  def reset_password
26
    if teamcity_url_changed? && !password_touched?
27 28 29 30
      self.password = nil
    end
  end

31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
  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

46
  def self.to_param
47 48 49 50 51 52
    'teamcity'
  end

  def fields
    [
      { type: 'text', name: 'teamcity_url',
53
        placeholder: 'TeamCity root URL like https://teamcity.example.com', required: true },
54
      { type: 'text', name: 'build_type',
55
        placeholder: 'Build configuration ID', required: true },
56 57
      { type: 'text', name: 'username',
        placeholder: 'A user with permissions to trigger a manual build' },
58
      { type: 'password', name: 'password' }
59 60 61
    ]
  end

Valery Sizov's avatar
Valery Sizov committed
62
  def build_page(sha, ref)
63
    with_reactive_cache(sha, ref) {|cached| cached[:build_page] }
64 65
  end

Valery Sizov's avatar
Valery Sizov committed
66
  def commit_status(sha, ref)
67 68
    with_reactive_cache(sha, ref) {|cached| cached[:commit_status] }
  end
69

70 71
  def calculate_reactive_cache(sha, ref)
    response = get_path("httpAuth/app/rest/builds/branch:unspecified:any,number:#{sha}")
72

73
    { build_page: read_build_page(response), commit_status: read_commit_status(response) }
74 75
  end

76
  def execute(data)
77
    return unless supported_events.include?(data[:object_kind])
78

79 80
    auth = {
      username: username,
81
      password: password
82 83
    }

84
    branch = Gitlab::Git.ref_name(data[:ref])
85

86
    Gitlab::HTTP.post(
87
      build_url('httpAuth/app/rest/buildQueue'),
88 89 90 91 92 93
      body: "<build branchName=\"#{branch}\">"\
            "<buildType id=\"#{build_type}\"/>"\
            '</build>',
      headers: { 'Content-type' => 'application/xml' },
      basic_auth: auth
    )
94
  end
95 96 97

  private

98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
  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

132 133 134 135 136
  def build_url(path)
    URI.join("#{teamcity_url}/", path).to_s
  end

  def get_path(path)
137 138 139 140 141
    Gitlab::HTTP.get(build_url(path), verify: false,
                                      basic_auth: {
                                        username: username,
                                        password: password
                                      })
142
  end
143
end