Merge branch '47818-post-comment-on-commit-when-downstream-pipeline-is-created-qa' into 'master'
Post a commit note with the downstream pipeline URL for omnibus triggers See merge request gitlab-org/gitlab-ce!21512
This commit is contained in:
commit
a7633d4a08
3 changed files with 131 additions and 109 deletions
|
@ -130,7 +130,6 @@ stages:
|
||||||
|
|
||||||
.single-script-job: &single-script-job
|
.single-script-job: &single-script-job
|
||||||
image: ruby:2.4-alpine
|
image: ruby:2.4-alpine
|
||||||
before_script: []
|
|
||||||
stage: test
|
stage: test
|
||||||
cache: {}
|
cache: {}
|
||||||
dependencies: []
|
dependencies: []
|
||||||
|
@ -259,6 +258,7 @@ package-and-qa:
|
||||||
SCRIPT_NAME: trigger-build
|
SCRIPT_NAME: trigger-build
|
||||||
retry: 0
|
retry: 0
|
||||||
script:
|
script:
|
||||||
|
- gem install gitlab --no-document
|
||||||
- ./$SCRIPT_NAME omnibus
|
- ./$SCRIPT_NAME omnibus
|
||||||
when: manual
|
when: manual
|
||||||
only:
|
only:
|
||||||
|
@ -285,7 +285,7 @@ review-docs-deploy-manual:
|
||||||
<<: *review-docs
|
<<: *review-docs
|
||||||
stage: build
|
stage: build
|
||||||
script:
|
script:
|
||||||
- gem install gitlab --no-ri --no-rdoc
|
- gem install gitlab --no-document
|
||||||
- ./$SCRIPT_NAME deploy
|
- ./$SCRIPT_NAME deploy
|
||||||
when: manual
|
when: manual
|
||||||
only:
|
only:
|
||||||
|
@ -299,7 +299,7 @@ review-docs-deploy:
|
||||||
<<: *review-docs
|
<<: *review-docs
|
||||||
stage: post-test
|
stage: post-test
|
||||||
script:
|
script:
|
||||||
- gem install gitlab --no-ri --no-rdoc
|
- gem install gitlab --no-document
|
||||||
- ./$SCRIPT_NAME deploy
|
- ./$SCRIPT_NAME deploy
|
||||||
only:
|
only:
|
||||||
- /(^docs[\/-].*|.*-docs$)/@gitlab-org/gitlab-ce
|
- /(^docs[\/-].*|.*-docs$)/@gitlab-org/gitlab-ce
|
||||||
|
@ -314,7 +314,7 @@ review-docs-cleanup:
|
||||||
name: review-docs/$CI_COMMIT_REF_SLUG
|
name: review-docs/$CI_COMMIT_REF_SLUG
|
||||||
action: stop
|
action: stop
|
||||||
script:
|
script:
|
||||||
- gem install gitlab --no-ri --no-rdoc
|
- gem install gitlab --no-document
|
||||||
- ./$SCRIPT_NAME cleanup
|
- ./$SCRIPT_NAME cleanup
|
||||||
when: manual
|
when: manual
|
||||||
only:
|
only:
|
||||||
|
@ -333,8 +333,8 @@ cloud-native-image:
|
||||||
GIT_DEPTH: "1"
|
GIT_DEPTH: "1"
|
||||||
cache: {}
|
cache: {}
|
||||||
script:
|
script:
|
||||||
- gem install gitlab --no-ri --no-rdoc
|
- gem install gitlab --no-document
|
||||||
- BUILD_TRIGGER_TOKEN=$CI_JOB_TOKEN scripts/trigger-build cng
|
- CNG_PROJECT_PATH="gitlab-org/build/CNG" BUILD_TRIGGER_TOKEN=$CI_JOB_TOKEN ./scripts/trigger-build cng
|
||||||
only:
|
only:
|
||||||
- tags@gitlab-org/gitlab-ce
|
- tags@gitlab-org/gitlab-ce
|
||||||
- tags@gitlab-org/gitlab-ee
|
- tags@gitlab-org/gitlab-ee
|
||||||
|
@ -366,7 +366,7 @@ update-tests-metadata:
|
||||||
- rspec_flaky/
|
- rspec_flaky/
|
||||||
policy: push
|
policy: push
|
||||||
script:
|
script:
|
||||||
- retry gem install fog-aws mime-types activesupport --no-ri --no-rdoc
|
- retry gem install fog-aws mime-types activesupport --no-document
|
||||||
- scripts/merge-reports ${KNAPSACK_RSPEC_SUITE_REPORT_PATH} knapsack/${CI_PROJECT_NAME}/rspec-pg_node_*.json
|
- scripts/merge-reports ${KNAPSACK_RSPEC_SUITE_REPORT_PATH} knapsack/${CI_PROJECT_NAME}/rspec-pg_node_*.json
|
||||||
- scripts/merge-reports ${FLAKY_RSPEC_SUITE_REPORT_PATH} rspec_flaky/all_*_*.json
|
- scripts/merge-reports ${FLAKY_RSPEC_SUITE_REPORT_PATH} rspec_flaky/all_*_*.json
|
||||||
- FLAKY_RSPEC_GENERATE_REPORT=1 scripts/prune-old-flaky-specs ${FLAKY_RSPEC_SUITE_REPORT_PATH}
|
- FLAKY_RSPEC_GENERATE_REPORT=1 scripts/prune-old-flaky-specs ${FLAKY_RSPEC_SUITE_REPORT_PATH}
|
||||||
|
|
|
@ -1,132 +1,160 @@
|
||||||
#!/usr/bin/env ruby
|
#!/usr/bin/env ruby
|
||||||
|
|
||||||
require 'net/http'
|
require 'gitlab'
|
||||||
require 'json'
|
|
||||||
require 'cgi'
|
#
|
||||||
|
# Configure credentials to be used with gitlab gem
|
||||||
|
#
|
||||||
|
Gitlab.configure do |config|
|
||||||
|
config.endpoint = 'https://gitlab.com/api/v4'
|
||||||
|
config.private_token = ENV['GITLAB_QA_ACCESS_TOKEN'] # gitlab-qa bot access token
|
||||||
|
end
|
||||||
|
|
||||||
module Trigger
|
module Trigger
|
||||||
OMNIBUS_PROJECT_PATH = 'gitlab-org/omnibus-gitlab'.freeze
|
|
||||||
CNG_PROJECT_PATH = 'gitlab-org/build/CNG'.freeze
|
|
||||||
TOKEN = ENV['BUILD_TRIGGER_TOKEN']
|
TOKEN = ENV['BUILD_TRIGGER_TOKEN']
|
||||||
|
|
||||||
def self.ee?
|
def self.ee?
|
||||||
ENV['CI_PROJECT_NAME'] == 'gitlab-ee' || File.exist?('CHANGELOG-EE.md')
|
ENV['CI_PROJECT_NAME'] == 'gitlab-ee' || File.exist?('CHANGELOG-EE.md')
|
||||||
end
|
end
|
||||||
|
|
||||||
class Omnibus
|
class Base
|
||||||
def initialize
|
def initialize(api_token)
|
||||||
@uri = URI("https://gitlab.com/api/v4/projects/#{CGI.escape(Trigger::OMNIBUS_PROJECT_PATH)}/trigger/pipeline")
|
Gitlab.private_token = api_token
|
||||||
@params = env_params.merge(file_params).merge(token: Trigger::TOKEN)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def invoke!
|
def invoke!(post_comment: false)
|
||||||
res = Net::HTTP.post_form(@uri, @params)
|
pipeline = Gitlab.run_trigger(
|
||||||
id = JSON.parse(res.body)['id']
|
downstream_project_path,
|
||||||
project = Trigger::OMNIBUS_PROJECT_PATH
|
Trigger::TOKEN,
|
||||||
|
ref,
|
||||||
|
variables)
|
||||||
|
|
||||||
if id
|
puts "Triggered #{pipeline.web_url}"
|
||||||
puts "Triggered https://gitlab.com/#{project}/pipelines/#{id}"
|
puts "Waiting for downstream pipeline status"
|
||||||
puts "Waiting for downstream pipeline status"
|
|
||||||
else
|
begin
|
||||||
raise "Trigger failed! The response from the trigger is: #{res.body}"
|
Trigger::CommitComment.post!(downstream_project_path, pipeline) if post_comment
|
||||||
|
rescue Gitlab::Error::Error => error
|
||||||
|
puts "Ignoring the following error: #{error}"
|
||||||
end
|
end
|
||||||
|
Trigger::Pipeline.new(downstream_project_path, pipeline.id)
|
||||||
Trigger::Pipeline.new(project, id)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def env_params
|
# Must be overriden
|
||||||
|
def downstream_project_path
|
||||||
|
raise NotImplementedError
|
||||||
|
end
|
||||||
|
|
||||||
|
# Must be overriden
|
||||||
|
def ref
|
||||||
|
raise NotImplementedError
|
||||||
|
end
|
||||||
|
|
||||||
|
# Can be overriden
|
||||||
|
def extra_variables
|
||||||
|
{}
|
||||||
|
end
|
||||||
|
|
||||||
|
# Can be overriden
|
||||||
|
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
|
||||||
{
|
{
|
||||||
"ref" => ENV["OMNIBUS_BRANCH"] || "master",
|
'TRIGGERED_USER' => ENV['GITLAB_USER_NAME'],
|
||||||
"variables[GITLAB_VERSION]" => ENV["CI_COMMIT_SHA"],
|
'TRIGGER_SOURCE' => ENV['CI_JOB_URL']
|
||||||
"variables[ALTERNATIVE_SOURCES]" => true,
|
|
||||||
"variables[ee]" => Trigger.ee? ? 'true' : 'false',
|
|
||||||
"variables[TRIGGERED_USER]" => ENV["GITLAB_USER_NAME"],
|
|
||||||
"variables[TRIGGER_SOURCE]" => "https://gitlab.com/gitlab-org/#{ENV['CI_PROJECT_NAME']}/-/jobs/#{ENV['CI_JOB_ID']}"
|
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
def file_params
|
# Read version files from all components
|
||||||
Hash.new.tap do |params|
|
def version_file_variables
|
||||||
Dir.glob("*_VERSION").each do |version_file|
|
Dir.glob("*_VERSION").each_with_object({}) do |version_file, params|
|
||||||
params["variables[#{version_file}]"] = File.read(version_file).strip
|
params[version_file] = version_param_value(version_file)
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
class CNG
|
class Omnibus < Base
|
||||||
def initialize
|
|
||||||
@uri = URI("https://gitlab.com/api/v4/projects/#{CGI.escape(Trigger::CNG_PROJECT_PATH)}/trigger/pipeline")
|
|
||||||
@ref_name = ENV['CI_COMMIT_REF_NAME']
|
|
||||||
@username = ENV['GITLAB_USER_NAME']
|
|
||||||
@project_name = ENV['CI_PROJECT_NAME']
|
|
||||||
@job_id = ENV['CI_JOB_ID']
|
|
||||||
@params = env_params.merge(file_params).merge(token: Trigger::TOKEN)
|
|
||||||
end
|
|
||||||
|
|
||||||
#
|
|
||||||
# Trigger a pipeline
|
|
||||||
#
|
|
||||||
def invoke!
|
|
||||||
res = Net::HTTP.post_form(@uri, @params)
|
|
||||||
id = JSON.parse(res.body)['id']
|
|
||||||
project = Trigger::CNG_PROJECT_PATH
|
|
||||||
|
|
||||||
if id
|
|
||||||
puts "Triggered https://gitlab.com/#{project}/pipelines/#{id}"
|
|
||||||
puts "Waiting for downstream pipeline status"
|
|
||||||
else
|
|
||||||
raise "Trigger failed! The response from the trigger is: #{res.body}"
|
|
||||||
end
|
|
||||||
|
|
||||||
Trigger::Pipeline.new(project, id)
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def env_params
|
def downstream_project_path
|
||||||
params = {
|
'gitlab-org/omnibus-gitlab'.freeze
|
||||||
"ref" => ENV["CNG_BRANCH"] || "master",
|
|
||||||
"variables[TRIGGERED_USER]" => @username,
|
|
||||||
"variables[TRIGGER_SOURCE]" => "https://gitlab.com/gitlab-org/#{@project_name}/-/jobs/#{@job_id}"
|
|
||||||
}
|
|
||||||
|
|
||||||
if Trigger.ee?
|
|
||||||
params["variables[GITLAB_EE_VERSION]"] = @ref_name
|
|
||||||
params["variables[EE_PIPELINE]"] = 'true'
|
|
||||||
else
|
|
||||||
params["variables[GITLAB_CE_VERSION]"] = @ref_name
|
|
||||||
params["variables[CE_PIPELINE]"] = 'true'
|
|
||||||
end
|
|
||||||
|
|
||||||
params
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# Read version files from all components
|
def ref
|
||||||
def file_params
|
ENV['OMNIBUS_BRANCH'] || 'master'
|
||||||
Dir.glob("*_VERSION").each_with_object({}) do |version_file, params|
|
end
|
||||||
raw_version = File.read(version_file).strip
|
|
||||||
# if the version matches semver format, treat it as a tag and prepend `v`
|
|
||||||
version = if raw_version =~ Regexp.compile(/^\d+\.\d+\.\d+(-rc\d+)?(-ee)?$/)
|
|
||||||
"v#{raw_version}"
|
|
||||||
else
|
|
||||||
raw_version
|
|
||||||
end
|
|
||||||
|
|
||||||
params["variables[#{version_file}]"] = version
|
def extra_variables
|
||||||
|
{
|
||||||
|
'GITLAB_VERSION' => ENV['CI_COMMIT_SHA'],
|
||||||
|
'ALTERNATIVE_SOURCES' => 'true',
|
||||||
|
'ee' => Trigger.ee? ? 'true' : 'false'
|
||||||
|
}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class CNG < Base
|
||||||
|
private
|
||||||
|
|
||||||
|
def downstream_project_path
|
||||||
|
ENV['CNG_PROJECT_PATH'] || 'gitlab-org/build/CNG-mirror'
|
||||||
|
end
|
||||||
|
|
||||||
|
def ref
|
||||||
|
ENV['CNG_BRANCH'] || 'master'
|
||||||
|
end
|
||||||
|
|
||||||
|
def extra_variables
|
||||||
|
edition = Trigger.ee? ? 'EE' : 'CE'
|
||||||
|
|
||||||
|
{
|
||||||
|
"GITLAB_#{edition}_VERSION" => ENV['CI_COMMIT_REF_NAME'],
|
||||||
|
"#{edition}_PIPELINE" => 'true'
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
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}"
|
||||||
|
else
|
||||||
|
raw_version
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class CommitComment
|
||||||
|
def self.post!(downstream_project_path, downstream_pipeline)
|
||||||
|
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.")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
class Pipeline
|
class Pipeline
|
||||||
INTERVAL = 60 # seconds
|
INTERVAL = 60 # seconds
|
||||||
MAX_DURATION = 3600 * 3 # 3 hours
|
MAX_DURATION = 3600 * 3 # 3 hours
|
||||||
|
|
||||||
|
attr_reader :project, :id
|
||||||
|
|
||||||
def initialize(project, id)
|
def initialize(project, id)
|
||||||
|
@project = project
|
||||||
|
@id = id
|
||||||
@start = Time.now.to_i
|
@start = Time.now.to_i
|
||||||
@uri = URI("https://gitlab.com/api/v4/projects/#{CGI.escape(project)}/pipelines/#{id}")
|
|
||||||
|
# gitlab-bot's token "GitLab multi-project pipeline polling"
|
||||||
|
Gitlab.private_token = ENV['GITLAB_BOT_MULTI_PROJECT_PIPELINE_POLLING_TOKEN']
|
||||||
end
|
end
|
||||||
|
|
||||||
def wait!
|
def wait!
|
||||||
|
@ -157,15 +185,9 @@ module Trigger
|
||||||
end
|
end
|
||||||
|
|
||||||
def status
|
def status
|
||||||
req = Net::HTTP::Get.new(@uri)
|
Gitlab.pipeline(project, id).status.to_sym
|
||||||
req['PRIVATE-TOKEN'] = ENV['GITLAB_QA_ACCESS_TOKEN']
|
rescue Gitlab::Error::Error => error
|
||||||
|
puts "Ignoring the following error: #{error}"
|
||||||
res = Net::HTTP.start(@uri.hostname, @uri.port, use_ssl: true) do |http|
|
|
||||||
http.request(req)
|
|
||||||
end
|
|
||||||
|
|
||||||
JSON.parse(res.body)['status'].to_s.to_sym
|
|
||||||
rescue JSON::ParserError
|
|
||||||
# Ignore GitLab API hiccups. If GitLab is really down, we'll hit the job
|
# Ignore GitLab API hiccups. If GitLab is really down, we'll hit the job
|
||||||
# timeout anyway.
|
# timeout anyway.
|
||||||
:running
|
:running
|
||||||
|
@ -175,9 +197,9 @@ end
|
||||||
|
|
||||||
case ARGV[0]
|
case ARGV[0]
|
||||||
when 'omnibus'
|
when 'omnibus'
|
||||||
Trigger::Omnibus.new.invoke!.wait!
|
Trigger::Omnibus.new(ENV['GITLAB_QA_ACCESS_TOKEN']).invoke!(post_comment: true).wait!
|
||||||
when 'cng'
|
when 'cng'
|
||||||
Trigger::CNG.new.invoke!.wait!
|
Trigger::CNG.new(ENV['GITLAB_QA_ACCESS_TOKEN']).invoke!.wait!
|
||||||
else
|
else
|
||||||
puts "Please provide a valid option:
|
puts "Please provide a valid option:
|
||||||
omnibus - Triggers a pipeline that builds the omnibus-gitlab package
|
omnibus - Triggers a pipeline that builds the omnibus-gitlab package
|
||||||
|
|
|
@ -6,8 +6,8 @@ require 'gitlab'
|
||||||
# Configure credentials to be used with gitlab gem
|
# Configure credentials to be used with gitlab gem
|
||||||
#
|
#
|
||||||
Gitlab.configure do |config|
|
Gitlab.configure do |config|
|
||||||
config.endpoint = 'https://gitlab.com/api/v4'
|
config.endpoint = 'https://gitlab.com/api/v4'
|
||||||
config.private_token = ENV["DOCS_API_TOKEN"] # GitLab Docs bot access token with Developer access to gitlab-docs
|
config.private_token = ENV["DOCS_API_TOKEN"] # GitLab Docs bot access token with Developer access to gitlab-docs
|
||||||
end
|
end
|
||||||
|
|
||||||
#
|
#
|
||||||
|
@ -99,7 +99,7 @@ def trigger_pipeline
|
||||||
|
|
||||||
puts "=> Follow the status of the triggered pipeline:"
|
puts "=> Follow the status of the triggered pipeline:"
|
||||||
puts ""
|
puts ""
|
||||||
puts "https://gitlab.com/gitlab-com/gitlab-docs/pipelines/#{pipeline.id}"
|
puts pipeline.web_url
|
||||||
puts ""
|
puts ""
|
||||||
puts "=> In a few minutes, you will be able to preview your changes under the following URL:"
|
puts "=> In a few minutes, you will be able to preview your changes under the following URL:"
|
||||||
puts ""
|
puts ""
|
||||||
|
|
Loading…
Reference in a new issue