BigW Consortium Gitlab

teamcity_service.rb 3.4 KB
Newer Older
1 2 3 4 5
class TeamcityService < CiService
  include HTTParty

  prop_accessor :teamcity_url, :build_type, :username, :password

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 46 47 48 49
  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

  def to_param
    'teamcity'
  end

50 51 52 53
  def supported_events
    %w(push)
  end

54 55 56 57 58 59 60 61 62 63 64 65 66
  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_info(sha)
67 68 69 70
    url = URI.join(
      teamcity_url,
      "/httpAuth/app/rest/builds/branch:unspecified:any,number:#{sha}"
    ).to_s
71 72
    auth = {
      username: username,
73
      password: password
74
    }
75
    @response = HTTParty.get(url, verify: false, basic_auth: auth)
76 77
  end

Valery Sizov committed
78
  def build_page(sha, ref)
79 80 81 82 83
    build_info(sha) if @response.nil? || !@response.code

    if @response.code != 200
      # If actual build link can't be determined,
      # send user to build summary page.
84
      URI.join(teamcity_url, "/viewLog.html?buildTypeId=#{build_type}").to_s
85 86 87
    else
      # If actual build link is available, go to build result page.
      built_id = @response['build']['id']
88 89
      URI.join(
        teamcity_url,
90
        "/viewLog.html?buildId=#{built_id}&buildTypeId=#{build_type}"
91
      ).to_s
92 93 94
    end
  end

Valery Sizov committed
95
  def commit_status(sha, ref)
96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
    build_info(sha) if @response.nil? || !@response.code
    return :error unless @response.code == 200 || @response.code == 404

    status = if @response.code == 404
               'Pending'
             else
               @response['build']['status']
             end

    if status.include?('SUCCESS')
      'success'
    elsif status.include?('FAILURE')
      'failed'
    elsif status.include?('Pending')
      'pending'
    else
      :error
    end
  end

116
  def execute(data)
117
    return unless supported_events.include?(data[:object_kind])
118

119 120 121 122 123
    auth = {
      username: username,
      password: password,
    }

124
    branch = Gitlab::Git.ref_name(data[:ref])
125

126
    self.class.post(
127
      URI.join(teamcity_url, '/httpAuth/app/rest/buildQueue').to_s,
128 129 130 131 132 133
      body: "<build branchName=\"#{branch}\">"\
            "<buildType id=\"#{build_type}\"/>"\
            '</build>',
      headers: { 'Content-type' => 'application/xml' },
      basic_auth: auth
    )
134 135
  end
end