2018-11-08 19:23:39 +05:30
#!/usr/bin/env ruby
2018-12-13 13:39:08 +05:30
# frozen_string_literal: true
2018-11-08 19:23:39 +05:30
2018-12-05 23:21:45 +05:30
require 'gitlab'
#
# Configure credentials to be used with gitlab gem
#
Gitlab.configure do |config|
2018-12-13 13:39:08 +05:30
config.endpoint = 'https://gitlab.com/api/v4'
2018-12-05 23:21:45 +05:30
end
2018-11-08 19:23:39 +05:30
module Trigger
def self.ee?
ENV['CI_PROJECT_NAME'] == 'gitlab-ee' || File.exist?('CHANGELOG-EE.md')
end
2018-12-05 23:21:45 +05:30
class Base
def invoke!(post_comment: false)
pipeline = Gitlab.run_trigger(
downstream_project_path,
2018-12-13 13:39:08 +05:30
trigger_token,
2018-12-05 23:21:45 +05:30
ref,
variables)
2018-11-08 19:23:39 +05:30
2018-12-13 13:39:08 +05:30
puts "Triggered downstream pipeline: #{pipeline.web_url}\n"
2018-12-05 23:21:45 +05:30
puts "Waiting for downstream pipeline status"
2018-11-08 19:23:39 +05:30
2018-12-13 13:39:08 +05:30
Trigger::CommitComment.post!(pipeline, access_token) if post_comment
Trigger::Pipeline.new(downstream_project_path, pipeline.id, access_token)
2018-11-08 19:23:39 +05:30
end
private
2018-12-13 13:39:08 +05:30
# Must be overridden
2018-12-05 23:21:45 +05:30
def downstream_project_path
raise NotImplementedError
end
2018-12-13 13:39:08 +05:30
# Must be overridden
2018-12-05 23:21:45 +05:30
def ref
raise NotImplementedError
end
2018-12-13 13:39:08 +05:30
# Must be overridden
def trigger_token
raise NotImplementedError
end
# Must be overridden
def access_token
raise NotImplementedError
end
# Can be overridden
2018-12-05 23:21:45 +05:30
def extra_variables
{}
end
2018-12-13 13:39:08 +05:30
# Can be overridden
2018-12-05 23:21:45 +05:30
def version_param_value(version_file)
File.read(version_file).strip
end
def variables
base_variables.merge(extra_variables).merge(version_file_variables)
end
def base_variables
2018-11-08 19:23:39 +05:30
{
2019-02-15 15:39:39 +05:30
'GITLAB_REF_SLUG' => ENV['CI_COMMIT_TAG'] ? ENV['CI_COMMIT_REF_NAME'] : ENV['CI_COMMIT_REF_SLUG'],
2018-12-13 13:39:08 +05:30
'TRIGGERED_USER' => ENV['TRIGGERED_USER'] || ENV['GITLAB_USER_NAME'],
'TRIGGER_SOURCE' => ENV['CI_JOB_URL'],
'TOP_UPSTREAM_SOURCE_PROJECT' => ENV['CI_PROJECT_PATH'],
'TOP_UPSTREAM_SOURCE_JOB' => ENV['CI_JOB_URL'],
'TOP_UPSTREAM_SOURCE_SHA' => ENV['CI_COMMIT_SHA']
2018-11-08 19:23:39 +05:30
}
end
2018-12-05 23:21:45 +05:30
# Read version files from all components
def version_file_variables
Dir.glob("*_VERSION").each_with_object({}) do |version_file, params|
params[version_file] = version_param_value(version_file)
2018-11-08 19:23:39 +05:30
end
end
end
2018-12-05 23:21:45 +05:30
class Omnibus < Base
private
2018-11-08 19:23:39 +05:30
2018-12-05 23:21:45 +05:30
def downstream_project_path
2018-12-13 13:39:08 +05:30
'gitlab-org/omnibus-gitlab'
2018-11-08 19:23:39 +05:30
end
2018-12-05 23:21:45 +05:30
def ref
ENV['OMNIBUS_BRANCH'] || 'master'
end
2018-12-13 13:39:08 +05:30
def trigger_token
ENV['BUILD_TRIGGER_TOKEN']
end
def access_token
ENV['GITLAB_BOT_MULTI_PROJECT_PIPELINE_POLLING_TOKEN']
end
2018-12-05 23:21:45 +05:30
def extra_variables
{
'GITLAB_VERSION' => ENV['CI_COMMIT_SHA'],
'ALTERNATIVE_SOURCES' => 'true',
2019-02-15 15:39:39 +05:30
'ee' => Trigger.ee? ? 'true' : 'false',
'QA_BRANCH' => ENV['QA_BRANCH'] || 'master'
2018-12-05 23:21:45 +05:30
}
end
end
class CNG < Base
2018-11-08 19:23:39 +05:30
private
2018-12-05 23:21:45 +05:30
def downstream_project_path
ENV['CNG_PROJECT_PATH'] || 'gitlab-org/build/CNG-mirror'
end
def ref
ENV['CNG_BRANCH'] || 'master'
end
2018-12-13 13:39:08 +05:30
def trigger_token
ENV['BUILD_TRIGGER_TOKEN']
end
def access_token
ENV['GITLAB_BOT_MULTI_PROJECT_PIPELINE_POLLING_TOKEN']
end
2018-12-05 23:21:45 +05:30
def extra_variables
edition = Trigger.ee? ? 'EE' : 'CE'
{
2019-02-15 15:39:39 +05:30
# Back-compatibility until https://gitlab.com/gitlab-org/build/CNG/merge_requests/189 is merged
2018-12-05 23:21:45 +05:30
"GITLAB_#{edition}_VERSION" => ENV['CI_COMMIT_REF_NAME'],
2019-02-15 15:39:39 +05:30
"GITLAB_VERSION" => ENV['CI_COMMIT_REF_NAME'],
"GITLAB_TAG" => ENV['CI_COMMIT_TAG'],
2019-05-18 00:54:41 +05:30
"GITLAB_ASSETS_TAG" => ENV['CI_COMMIT_TAG'] ? ENV['CI_COMMIT_REF_NAME'] : ENV['CI_COMMIT_REF_SLUG'],
2018-12-13 13:39:08 +05:30
"#{edition}_PIPELINE" => 'true'
2018-11-08 19:23:39 +05:30
}
2018-12-05 23:21:45 +05:30
end
2018-11-08 19:23:39 +05:30
2018-12-05 23:21:45 +05:30
def version_param_value(_version_file)
raw_version = super
# if the version matches semver format, treat it as a tag and prepend `v`
if raw_version =~ Regexp.compile(/^\d+\.\d+\.\d+(-rc\d+)?(-ee)?$/)
"v#{raw_version}"
2018-11-08 19:23:39 +05:30
else
2018-12-05 23:21:45 +05:30
raw_version
2018-11-08 19:23:39 +05:30
end
end
2018-12-05 23:21:45 +05:30
end
2018-11-08 19:23:39 +05:30
2018-12-05 23:21:45 +05:30
class CommitComment
2018-12-13 13:39:08 +05:30
def self.post!(downstream_pipeline, access_token)
Gitlab.private_token = access_token
2018-12-05 23:21:45 +05:30
Gitlab.create_commit_comment(
ENV['CI_PROJECT_PATH'],
ENV['CI_COMMIT_SHA'],
"The [`#{ENV['CI_JOB_NAME']}`](#{ENV['CI_JOB_URL']}) job from pipeline #{ENV['CI_PIPELINE_URL']} triggered #{downstream_pipeline.web_url} downstream.")
2018-12-13 13:39:08 +05:30
rescue Gitlab::Error::Error => error
puts "Ignoring the following error: #{error}"
2018-11-08 19:23:39 +05:30
end
end
class Pipeline
INTERVAL = 60 # seconds
MAX_DURATION = 3600 * 3 # 3 hours
2018-12-13 13:39:08 +05:30
attr_reader :project, :id, :api_token
2018-12-05 23:21:45 +05:30
2018-12-13 13:39:08 +05:30
def initialize(project, id, api_token)
2018-12-05 23:21:45 +05:30
@project = project
@id = id
2018-12-13 13:39:08 +05:30
@api_token = api_token
2018-11-08 19:23:39 +05:30
@start = Time.now.to_i
2018-12-05 23:21:45 +05:30
# gitlab-bot's token "GitLab multi-project pipeline polling"
2018-12-13 13:39:08 +05:30
Gitlab.private_token = api_token
2018-11-08 19:23:39 +05:30
end
def wait!
loop do
raise "Pipeline timed out after waiting for #{duration} minutes!" if timeout?
case status
when :created, :pending, :running
print "."
sleep INTERVAL
when :success
puts "Pipeline succeeded in #{duration} minutes!"
break
else
raise "Pipeline did not succeed!"
end
STDOUT.flush
end
end
def timeout?
Time.now.to_i > (@start + MAX_DURATION)
end
def duration
(Time.now.to_i - @start) / 60
end
def status
2018-12-05 23:21:45 +05:30
Gitlab.pipeline(project, id).status.to_sym
rescue Gitlab::Error::Error => error
puts "Ignoring the following error: #{error}"
2018-11-08 19:23:39 +05:30
# Ignore GitLab API hiccups. If GitLab is really down, we'll hit the job
# timeout anyway.
:running
end
end
end
case ARGV[0]
when 'omnibus'
2018-12-13 13:39:08 +05:30
Trigger::Omnibus.new.invoke!(post_comment: true).wait!
2018-11-08 19:23:39 +05:30
when 'cng'
2018-12-13 13:39:08 +05:30
Trigger::CNG.new.invoke!.wait!
2018-11-08 19:23:39 +05:30
else
puts "Please provide a valid option:
omnibus - Triggers a pipeline that builds the omnibus-gitlab package
cng - Triggers a pipeline that builds images used by the GitLab helm chart"
end