mirror of
https://gitlab.com/gitlab-org/gitlab-foss.git
synced 2025-07-23 00:47:51 +00:00
835 lines
27 KiB
Ruby
Executable File
835 lines
27 KiB
Ruby
Executable File
#!/usr/bin/env ruby
|
|
# frozen_string_literal: true
|
|
|
|
# We need to take some precautions when using the `gitlab` gem in this project.
|
|
#
|
|
# See https://docs.gitlab.com/ee/development/pipelines/internals.html#using-the-gitlab-ruby-gem-in-the-canonical-project.
|
|
require 'gitlab'
|
|
require 'yaml'
|
|
require 'json'
|
|
require 'open3'
|
|
require 'tempfile'
|
|
require 'httparty'
|
|
require 'logger'
|
|
|
|
# Monkeypatch gitlab gem in order to increase per_page size when fetching registry repositories
|
|
# rubocop:disable Style/ClassAndModuleChildren, Gitlab/NoCodeCoverageComment -- monkeypatch
|
|
# :nocov:
|
|
#
|
|
# TODO: Remove this monkeypatch once https://github.com/NARKOZ/gitlab/pull/710 is part of a new gem release (currently v5.1.0 doens't contain it) and the release is used in this project.
|
|
class Gitlab::Client
|
|
module ContainerRegistry
|
|
def registry_repositories(project, options = {})
|
|
get("/projects/#{url_encode project}/registry/repositories", query: options)
|
|
end
|
|
end
|
|
end
|
|
# :nocov:
|
|
# rubocop:enable Style/ClassAndModuleChildren, Gitlab/NoCodeCoverageComment
|
|
|
|
module Trigger
|
|
def self.ee?
|
|
# Support former project name for `dev`
|
|
%w[gitlab gitlab-ee].include?(ENV['CI_PROJECT_NAME'])
|
|
end
|
|
|
|
def self.non_empty_variable_value(variable)
|
|
variable_value = ENV[variable]
|
|
|
|
return if variable_value.nil? || variable_value.empty?
|
|
|
|
variable_value
|
|
end
|
|
|
|
def self.variables_for_env_file(variables)
|
|
variables.map do |key, value|
|
|
%(#{key}=#{value})
|
|
end.join("\n")
|
|
end
|
|
|
|
class Base
|
|
# Can be overridden
|
|
STABLE_BRANCH_REGEX = /^[\d-]+-stable(-ee|-jh)?$/
|
|
def access_token
|
|
ENV['PROJECT_TOKEN_FOR_CI_SCRIPTS_API_USAGE']
|
|
end
|
|
|
|
def invoke!
|
|
pipeline_variables = variables
|
|
|
|
puts "Triggering downstream pipeline on #{downstream_project_path}"
|
|
puts "with variables #{pipeline_variables}"
|
|
|
|
pipeline = downstream_client.run_trigger(
|
|
downstream_project_path,
|
|
trigger_token,
|
|
ref,
|
|
pipeline_variables)
|
|
|
|
puts "Triggered downstream pipeline: #{pipeline.web_url}\n"
|
|
puts "Waiting for downstream pipeline status"
|
|
|
|
Trigger::Pipeline.new(downstream_project_path, pipeline.id, downstream_client)
|
|
end
|
|
|
|
def variables
|
|
simple_forwarded_variables.merge(base_variables, extra_variables, version_file_variables)
|
|
end
|
|
|
|
def simple_forwarded_variables
|
|
{
|
|
'TRIGGER_SOURCE' => ENV['CI_JOB_URL'],
|
|
'TOP_UPSTREAM_SOURCE_PROJECT' => ENV['CI_PROJECT_PATH'],
|
|
'TOP_UPSTREAM_SOURCE_REF' => ENV['CI_COMMIT_REF_NAME'],
|
|
'TOP_UPSTREAM_SOURCE_JOB' => ENV['CI_JOB_URL'],
|
|
'TOP_UPSTREAM_MERGE_REQUEST_PROJECT_ID' => ENV['CI_MERGE_REQUEST_PROJECT_ID'],
|
|
'TOP_UPSTREAM_MERGE_REQUEST_IID' => ENV['CI_MERGE_REQUEST_IID']
|
|
}
|
|
end
|
|
|
|
private
|
|
|
|
def com_gitlab_client
|
|
@com_gitlab_client ||= Gitlab.client(
|
|
endpoint: endpoint,
|
|
private_token: access_token
|
|
)
|
|
end
|
|
|
|
# This client is used for downstream build and pipeline status
|
|
# Can be overridden
|
|
def downstream_client
|
|
com_gitlab_client
|
|
end
|
|
|
|
# Must be overridden
|
|
def downstream_project_path
|
|
raise NotImplementedError
|
|
end
|
|
|
|
# Must be overridden
|
|
def ref_param_name
|
|
raise NotImplementedError
|
|
end
|
|
|
|
# Can be overridden
|
|
def primary_ref
|
|
'main'
|
|
end
|
|
|
|
# Can be overridden
|
|
def trigger_token
|
|
ENV['CI_JOB_TOKEN']
|
|
end
|
|
|
|
# Can be overridden
|
|
def extra_variables
|
|
{}
|
|
end
|
|
|
|
# Can be overridden
|
|
def version_param_value(version_file)
|
|
ENV[version_file]&.strip || File.read(version_file).strip
|
|
end
|
|
|
|
# Can be overridden
|
|
def trigger_stable_branch_if_detected?
|
|
false
|
|
end
|
|
|
|
def stable_branch?
|
|
ENV['CI_COMMIT_REF_NAME'] =~ STABLE_BRANCH_REGEX
|
|
end
|
|
|
|
def mr_target_stable_branch?
|
|
ENV['CI_MERGE_REQUEST_TARGET_BRANCH_NAME'] =~ STABLE_BRANCH_REGEX
|
|
end
|
|
|
|
def fallback_ref
|
|
return primary_ref unless trigger_stable_branch_if_detected?
|
|
|
|
if stable_branch?
|
|
normalize_stable_branch_name(ENV['CI_COMMIT_REF_NAME'])
|
|
elsif mr_target_stable_branch?
|
|
normalize_stable_branch_name(ENV['CI_MERGE_REQUEST_TARGET_BRANCH_NAME'])
|
|
else
|
|
primary_ref
|
|
end
|
|
end
|
|
|
|
def normalize_stable_branch_name(branch_name)
|
|
if ENV['CI_PROJECT_NAMESPACE'] == 'gitlab-cn'
|
|
branch_name.delete_suffix('-jh')
|
|
elsif ["gitlab-org", "gitlab-org/security"].include?(ENV['CI_PROJECT_NAMESPACE'])
|
|
branch_name.delete_suffix('-ee')
|
|
end
|
|
end
|
|
|
|
def ref
|
|
ENV.fetch(ref_param_name, fallback_ref)
|
|
end
|
|
|
|
def base_variables
|
|
{
|
|
'GITLAB_REF_SLUG' => ENV['CI_COMMIT_TAG'] ? ENV['CI_COMMIT_REF_NAME'] : ENV['CI_COMMIT_REF_SLUG'],
|
|
'TRIGGERED_USER' => ENV['TRIGGERED_USER'] || ENV['GITLAB_USER_NAME'],
|
|
'TOP_UPSTREAM_SOURCE_SHA' => ENV['CI_COMMIT_SHA']
|
|
}
|
|
end
|
|
|
|
# Read version files from all components
|
|
def version_file_variables
|
|
Dir.glob("*_VERSION").each_with_object({}) do |version_file, params| # rubocop:disable Rails/IndexWith -- Non-rails CI script
|
|
params[version_file] = version_param_value(version_file)
|
|
end
|
|
end
|
|
|
|
def endpoint
|
|
return "https://jihulab.com/api/v4" if ENV['CI_PROJECT_NAMESPACE'] == 'gitlab-cn'
|
|
|
|
"https://gitlab.com/api/v4"
|
|
end
|
|
end
|
|
|
|
# Variable creation for downstream CNG build triggers
|
|
#
|
|
# This class additionally contains logic to check if component versions are already present in the container registry
|
|
# If they are, it adds those jobs to the SKIP_JOB_REGEX variable to skip them in the CNG build pipeline
|
|
#
|
|
# In order to correctly compute container versions and skip jobs, following actions are performed:
|
|
# * container version shell script is fetched from upstream
|
|
# * versions.yml file is fetched which contains most of variables required for container version calculation
|
|
# * image digest of stable Debian and Alpine images are fetched (alpine-stable and alpine-debian jobs functionality)
|
|
# * all container versions are computed using same logic as CNG build pipeline
|
|
# * registry is checked for image existence and appropriate jobs are added to skip regex pattern
|
|
#
|
|
class CNG < Base
|
|
TriggerRefBranchCreationFailed = Class.new(StandardError)
|
|
|
|
DEFAULT_DEBIAN_IMAGE = "debian:bookworm-slim"
|
|
DEFAULT_ALPINE_IMAGE = "alpine:3.20"
|
|
DEFAULT_SKIPPED_JOBS = %w[final-images-listing].freeze
|
|
DEFAULT_SKIPPED_JOB_REGEX = "/#{DEFAULT_SKIPPED_JOBS.join('|')}/".freeze
|
|
STABLE_BASE_JOBS = %w[alpine-stable debian-stable].freeze
|
|
|
|
def variables
|
|
hash = without_trigger_vars(super.dup)
|
|
|
|
unless skip_redundant_jobs?
|
|
logger.info("Skipping redundant jobs is disabled, skipping existing container image check")
|
|
return hash
|
|
end
|
|
|
|
hash.merge({
|
|
**deploy_component_tag_variables,
|
|
'SKIP_IMAGE_TAGGING' => "true",
|
|
'SKIP_JOB_REGEX' => skip_job_regex,
|
|
'DEBIAN_IMAGE' => debian_image,
|
|
'DEBIAN_DIGEST' => debian_image.split('@').last,
|
|
'DEBIAN_BUILD_ARGS' => "--build-arg DEBIAN_IMAGE=#{ENV['GITLAB_DEPENDENCY_PROXY']}#{debian_image}",
|
|
'ALPINE_IMAGE' => alpine_image,
|
|
'ALPINE_DIGEST' => alpine_image.split('@').last,
|
|
'ALPINE_BUILD_ARGS' => "--build-arg ALPINE_IMAGE=#{ENV['GITLAB_DEPENDENCY_PROXY']}#{alpine_image}"
|
|
})
|
|
rescue TriggerRefBranchCreationFailed => e
|
|
# raise if pipeline runs in MR that updates ref to make sure branch for trigger is created
|
|
raise(e) if ref_update_mr?
|
|
|
|
logger.error("Error while creating trigger ref branch, err: #{e.message}")
|
|
logger.error(e.backtrace.join("\n"))
|
|
logger.error("Falling back to default variables")
|
|
without_trigger_vars(super.dup)
|
|
rescue StandardError => e
|
|
# if skipping redundant jobs is enabled and fetching jobs to skip failed, attempt fallback to default variables
|
|
raise(e) unless skip_redundant_jobs?
|
|
|
|
logger.error("Error while calculating variables, err: #{e.message}")
|
|
logger.error(e.backtrace.join("\n"))
|
|
logger.error("Falling back to default variables")
|
|
hash
|
|
end
|
|
|
|
def simple_forwarded_variables
|
|
super.merge({
|
|
'TOP_UPSTREAM_SOURCE_REF_SLUG' => ENV['CI_COMMIT_REF_SLUG']
|
|
})
|
|
end
|
|
|
|
private
|
|
|
|
def assets_tag_variable
|
|
tag = ENV['GLCI_ASSETS_IMAGE_TAG']
|
|
return { 'GITLAB_ASSETS_TAG' => tag } unless tag.nil? || tag.empty?
|
|
|
|
logger.warn("No image tag found in GLCI_ASSETS_IMAGE_TAG environment variable, enabling asset compilation in CNG pipeline")
|
|
{ 'COMPILE_ASSETS' => 'true' }
|
|
end
|
|
|
|
# overridden base class methods
|
|
def downstream_project_path
|
|
ENV.fetch('CNG_PROJECT_PATH', 'gitlab-org/build/CNG-mirror')
|
|
end
|
|
|
|
def ref
|
|
return @ref if @ref
|
|
return @ref = super if cng_commit_sha.to_s.empty?
|
|
|
|
# TODO: remove this hack once https://gitlab.com/gitlab-org/gitlab/-/issues/369583 is resolved
|
|
trigger_branch_name = "trigger-refs/#{cng_commit_sha}"
|
|
return @ref = trigger_branch_name if branch_exists?(trigger_branch_name)
|
|
|
|
downstream_client.create_branch(downstream_project_path, trigger_branch_name, cng_commit_sha)
|
|
logger.info("Created temp trigger branch '#{trigger_branch_name}' for commit '#{cng_commit_sha}'")
|
|
@ref = trigger_branch_name
|
|
rescue StandardError => e
|
|
# redundancy in case explicit branch existence api request failed
|
|
return trigger_branch_name if e.message.include?("already exists")
|
|
|
|
@ref = super
|
|
raise TriggerRefBranchCreationFailed, e.message
|
|
end
|
|
|
|
def ref_param_name
|
|
'CNG_BRANCH'
|
|
end
|
|
|
|
def primary_ref
|
|
return "main-jh" if ENV['CI_PROJECT_NAMESPACE'] == 'gitlab-cn'
|
|
|
|
"master"
|
|
end
|
|
|
|
def trigger_stable_branch_if_detected?
|
|
true
|
|
end
|
|
|
|
def gitlab_ref_slug
|
|
if ENV['CI_COMMIT_TAG']
|
|
ENV['CI_COMMIT_REF_NAME']
|
|
else
|
|
ENV['CI_COMMIT_SHA']
|
|
end
|
|
end
|
|
|
|
def gitlab_version
|
|
ENV['CI_COMMIT_SHA']
|
|
end
|
|
|
|
def base_variables
|
|
super.merge('GITLAB_REF_SLUG' => gitlab_ref_slug)
|
|
end
|
|
|
|
def extra_variables
|
|
{
|
|
"TRIGGER_BRANCH" => ref,
|
|
"GITLAB_VERSION" => gitlab_version,
|
|
"GITLAB_TAG" => ENV['CI_COMMIT_TAG'], # Always set a value, even an empty string, so that the downstream pipeline can correctly check it.
|
|
"FORCE_RAILS_IMAGE_BUILDS" => 'true',
|
|
"CE_PIPELINE" => Trigger.ee? ? nil : "true", # Always set a value, even an empty string, so that the downstream pipeline can correctly check it.
|
|
"EE_PIPELINE" => Trigger.ee? ? "true" : nil, # Always set a value, even an empty string, so that the downstream pipeline can correctly check it.
|
|
"FULL_RUBY_VERSION" => RUBY_VERSION,
|
|
"SKIP_JOB_REGEX" => DEFAULT_SKIPPED_JOB_REGEX,
|
|
"DEBIAN_IMAGE" => DEFAULT_DEBIAN_IMAGE, # Make sure default values are always set to not end up as empty string
|
|
"ALPINE_IMAGE" => DEFAULT_ALPINE_IMAGE, # Make sure default values are always set to not end up as empty string
|
|
**default_build_vars,
|
|
**assets_tag_variable
|
|
}
|
|
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 Regexp.compile(/^\d+\.\d+\.\d+(-rc\d+)?(-ee)?$/).match?(raw_version)
|
|
"v#{raw_version}"
|
|
else
|
|
raw_version
|
|
end
|
|
end
|
|
|
|
def access_token
|
|
ENV["CNG_ACCESS_TOKEN"].then { |token| token.to_s.empty? ? super : token }
|
|
end
|
|
# overridden base class methods
|
|
|
|
# Logger with file output
|
|
#
|
|
# @return [Logger]
|
|
def logger
|
|
@logger ||= Logger.new(ENV.fetch("CNG_VAR_SETUP_LOG_FILE", "tmp/cng-var-setup.log"))
|
|
end
|
|
|
|
# Specific commit sha to be used instead of branch if defined
|
|
#
|
|
# @return [String]
|
|
def cng_commit_sha
|
|
@cng_commit_sha ||= ENV['CNG_COMMIT_SHA']
|
|
end
|
|
|
|
# Default variables used in CNG builds that affect container version values
|
|
#
|
|
# @return [Hash]
|
|
def default_build_vars
|
|
@default_build_vars ||= {
|
|
"CONTAINER_VERSION_SUFFIX" => ENV.fetch("CI_PROJECT_PATH_SLUG", "upstream-trigger"),
|
|
"CACHE_BUSTER" => "false",
|
|
"ARCH_LIST" => ENV.fetch("ARCH_LIST", "amd64")
|
|
}
|
|
end
|
|
|
|
# Skip redundant build jobs by calculating if container images are already present in the registry
|
|
#
|
|
# @return [Boolean]
|
|
def skip_redundant_jobs?
|
|
ENV["CNG_SKIP_REDUNDANT_JOBS"] == "true"
|
|
end
|
|
|
|
# Pipeline is part of MR that updates cng-mirror ref
|
|
#
|
|
# @return [Boolean]
|
|
def ref_update_mr?
|
|
ENV["CI_MERGE_REQUEST_TARGET_BRANCH_NAME"]&.match?(%r{renovate-e2e/cng\S+digest})
|
|
end
|
|
|
|
# Skipped job regex based on existing container tags in the registry
|
|
#
|
|
# @return [String]
|
|
def skip_job_regex
|
|
"/#{[*DEFAULT_SKIPPED_JOBS, *STABLE_BASE_JOBS, *skippable_jobs].join('|')}/"
|
|
end
|
|
|
|
# Branch existence check
|
|
#
|
|
# @param branch_name [String]
|
|
# @return [Boolean]
|
|
def branch_exists?(branch_name)
|
|
!!downstream_client.branch(downstream_project_path, branch_name)
|
|
rescue Gitlab::Error::ResponseError
|
|
false
|
|
end
|
|
|
|
def without_trigger_vars(hash)
|
|
hash.except('TRIGGER_SOURCE', 'TRIGGERED_USER')
|
|
end
|
|
|
|
# Repository file tree in form of the output of `git ls-tree` command
|
|
#
|
|
# @return [String]
|
|
def repo_tree
|
|
logger.info("Fetching repo tree for ref '#{ref}'")
|
|
downstream_client
|
|
.repo_tree(downstream_project_path, ref: ref, per_page: 100).auto_paginate
|
|
.select { |node| node["type"] == "tree" }
|
|
.map { |node| "#{node['mode']} #{node['type']} #{node['id']} #{node['path']}" }
|
|
.join("\n")
|
|
end
|
|
|
|
# Script used for container version calculations in CNG build jobs
|
|
#
|
|
# @return [String]
|
|
def container_versions_script
|
|
logger.info("Fetching container versions script for ref '#{ref}'")
|
|
downstream_client.file_contents(
|
|
downstream_project_path,
|
|
"build-scripts/container_versions.sh",
|
|
ref
|
|
)
|
|
end
|
|
|
|
# Debian image with digest
|
|
#
|
|
# @return [String]
|
|
def debian_image
|
|
@debian_image ||= docker_image_with_digest(cng_versions["DEBIAN_IMAGE"])
|
|
end
|
|
|
|
# Alpine image with digest
|
|
#
|
|
# @return [String]
|
|
def alpine_image
|
|
@alpine_image ||= docker_image_with_digest(cng_versions["ALPINE_IMAGE"])
|
|
end
|
|
|
|
# Edition postfix
|
|
#
|
|
# @return [String]
|
|
def edition
|
|
@edition ||= Trigger.ee? ? "ee" : "ce"
|
|
end
|
|
|
|
# Component versions used in CNG builds
|
|
#
|
|
# @return [Hash]
|
|
def cng_versions
|
|
@cng_versions ||= YAML
|
|
.safe_load(downstream_client.file_contents(downstream_project_path, "ci_files/variables.yml", ref))
|
|
.fetch("variables")
|
|
end
|
|
|
|
# Environment variables required for container version fetching
|
|
# All these variables influence final container version values
|
|
#
|
|
# @return [Hash]
|
|
def version_fetch_env_variables
|
|
{
|
|
**cng_versions,
|
|
**version_file_variables,
|
|
**default_build_vars,
|
|
"GITLAB_VERSION" => gitlab_version,
|
|
"RUBY_VERSION" => RUBY_VERSION,
|
|
"DEBIAN_DIGEST" => debian_image.split("@").last,
|
|
"ALPINE_DIGEST" => alpine_image.split("@").last,
|
|
"REPOSITORY_TREE" => repo_tree
|
|
}
|
|
end
|
|
|
|
# Image tags used by CNG deployments
|
|
#
|
|
# @return [Hash]
|
|
def deploy_component_tag_variables
|
|
{
|
|
"GITALY_TAG" => container_versions["gitaly"],
|
|
"GITLAB_SHELL_TAG" => container_versions["gitlab-shell"],
|
|
"GITLAB_TOOLBOX_TAG" => container_versions["gitlab-toolbox-#{edition}"],
|
|
"GITLAB_SIDEKIQ_TAG" => container_versions["gitlab-sidekiq-#{edition}"],
|
|
"GITLAB_WEBSERVICE_TAG" => container_versions["gitlab-webservice-#{edition}"],
|
|
"GITLAB_WORKHORSE_TAG" => container_versions["gitlab-workhorse-#{edition}"],
|
|
"GITLAB_KAS_TAG" => container_versions["gitlab-kas"],
|
|
"GITLAB_CONTAINER_REGISTRY_TAG" => container_versions["gitlab-container-registry"]
|
|
}
|
|
end
|
|
|
|
# Container versions for all components in CNG build pipeline
|
|
#
|
|
# @return [Hash]
|
|
def container_versions
|
|
@container_versions ||= Tempfile.create('container-versions') do |file|
|
|
file.write(container_versions_script)
|
|
file.close
|
|
|
|
build_vars = version_fetch_env_variables.transform_values(&:to_s)
|
|
logger.info("Computing container versions using following env variables:\n#{JSON.pretty_generate(build_vars)}")
|
|
out, status = Open3.capture2e(build_vars, "bash -c 'source #{file.path} && get_all_versions'")
|
|
raise "Failed to fetch container versions! #{out}" unless status.success?
|
|
|
|
component_versions = out.split("\n")
|
|
unless component_versions.all? { |line| line.match?(/^[A-Za-z0-9_\-]+=[^=]+$/) }
|
|
raise "Invalid container versions output format! Expected key=value pairs got:\n#{out}"
|
|
end
|
|
|
|
component_versions
|
|
.to_h { |entry| entry.split("=") }
|
|
.reject { |name, _version| Trigger.ee? ? name.end_with?("-ce") : name.end_with?("-ee") }
|
|
.tap { |versions| logger.info("Computed container versions:\n#{JSON.pretty_generate(versions)}") }
|
|
end
|
|
end
|
|
|
|
# List of jobs that can be skipped because tag is already present in the registry
|
|
#
|
|
# @return [Array]
|
|
def skippable_jobs
|
|
jobs = container_versions.keys
|
|
logger.info("Fetching container registry repositories for project '#{downstream_project_path}'")
|
|
repositories = downstream_client.registry_repositories(downstream_project_path, per_page: 100).auto_paginate
|
|
build_repositories = repositories.each_with_object({}) do |repo, hash|
|
|
job = jobs.find { |job| repo.name.end_with?(job) }
|
|
next unless job
|
|
|
|
hash[job] = repo.id
|
|
end
|
|
logger.info("Checking repositories (#{build_repositories.keys.join(', ')}) for existing tags")
|
|
existing_tags = container_versions.select do |job, tag|
|
|
downstream_client.registry_repository_tag(downstream_project_path, build_repositories[job], tag)
|
|
logger.info("Tag '#{tag}' exists in the registry, job '#{job}' will be skipped")
|
|
rescue Gitlab::Error::ResponseError => e
|
|
if e.is_a?(Gitlab::Error::NotFound)
|
|
logger.info("Tag '#{tag}' does not exist in the registry, job '#{job}' will not skipped")
|
|
else
|
|
logger.error("Failed to do a tag '#{tag}' lookup, err: #{e.message}, job '#{job}' will not be skipped")
|
|
end
|
|
|
|
false
|
|
end
|
|
|
|
jobs_to_skip = existing_tags.keys
|
|
return jobs_to_skip unless jobs == jobs_to_skip
|
|
|
|
# Trigger job will fail with no error message if downstream pipeline has no jobs or only skipped/manual jobs.
|
|
# Trigger gitlab-rails job if that is the case for downstream pipeline to work correctly.
|
|
logger.warn("Retaining gitlab-rails job to avoid empty downstream pipeline to not fail trigger job!")
|
|
jobs_to_skip.reject { |name| name.start_with?("gitlab-rails") }
|
|
end
|
|
|
|
# rubocop:disable Gitlab/HTTParty -- CI script
|
|
|
|
# Fetch Docker image with digest from DockerHub
|
|
#
|
|
# @param docker_image [String]
|
|
# @return [String]
|
|
def docker_image_with_digest(docker_image)
|
|
image, tag = docker_image.split(":")
|
|
|
|
logger.info("Fetching digest for image '#{docker_image}'")
|
|
auth_url = "https://auth.docker.io/token?service=registry.docker.io&scope=repository:library/#{image}:pull"
|
|
auth_response = HTTParty.get(auth_url)
|
|
raise "Failed to get auth token" unless auth_response.success?
|
|
|
|
token = JSON.parse(auth_response.body)['token']
|
|
manifest_url = "https://registry.hub.docker.com/v2/library/#{image}/manifests/#{tag}"
|
|
response = HTTParty.head(manifest_url, headers: {
|
|
'Authorization' => "Bearer #{token}",
|
|
'Accept' => 'application/vnd.docker.distribution.manifest.v2+json'
|
|
})
|
|
raise "Failed to fetch image '#{docker_image}' digest" unless response.success?
|
|
|
|
digest = response.headers['docker-content-digest'] || raise("Failed to get image digest")
|
|
"#{image}:#{tag}@#{digest}"
|
|
end
|
|
# rubocop:enable Gitlab/HTTParty -- CI script
|
|
end
|
|
|
|
# For GitLab documentation review apps
|
|
class Docs < Base
|
|
def access_token
|
|
ENV['DOCS_PROJECT_API_TOKEN'] || super
|
|
end
|
|
|
|
SUCCESS_MESSAGE = <<~MSG
|
|
=> You should now be able to preview your changes under the following URL:
|
|
|
|
%<app_url>s
|
|
|
|
=> For more information, see the documentation
|
|
=> https://docs.gitlab.com/ee/development/documentation/review_apps.html
|
|
|
|
=> If something doesn't work, drop a line in the #docs chat channel.
|
|
MSG
|
|
|
|
def deploy!
|
|
invoke!.wait!
|
|
display_success_message
|
|
end
|
|
|
|
#
|
|
# Remove a remote environment in the docs-gitlab-com project.
|
|
#
|
|
def cleanup!
|
|
environment = com_gitlab_client.environments(downstream_project_path, name: downstream_environment).first
|
|
return unless environment
|
|
|
|
environment = com_gitlab_client.stop_environment(downstream_project_path, environment.id)
|
|
if environment.state == 'stopped'
|
|
puts "=> Downstream environment '#{downstream_environment}' stopped."
|
|
else
|
|
puts "=> Downstream environment '#{downstream_environment}' failed to stop."
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def downstream_environment
|
|
"upstream-review/mr-${CI_MERGE_REQUEST_IID}"
|
|
end
|
|
|
|
def review_slug
|
|
identifier = ENV['CI_MERGE_REQUEST_IID'] || ENV['CI_COMMIT_REF_SLUG']
|
|
|
|
"#{project_slug}-#{identifier}"
|
|
end
|
|
|
|
def downstream_project_path
|
|
ENV.fetch('DOCS_PROJECT_PATH', 'gitlab-org/technical-writing/docs-gitlab-com')
|
|
end
|
|
|
|
def ref_param_name
|
|
'DOCS_BRANCH'
|
|
end
|
|
|
|
def trigger_token
|
|
ENV['DOCS_TRIGGER_TOKEN']
|
|
end
|
|
|
|
def extra_variables
|
|
{
|
|
"BRANCH_#{project_slug.upcase}" => ENV['CI_COMMIT_REF_NAME'],
|
|
"MERGE_REQUEST_IID_#{project_slug.upcase}" => ENV['CI_MERGE_REQUEST_IID'],
|
|
"REVIEW_SLUG" => review_slug
|
|
}
|
|
end
|
|
|
|
def project_slug
|
|
case ENV['CI_PROJECT_PATH']
|
|
when 'gitlab-org/gitlab-foss'
|
|
'ce'
|
|
when 'gitlab-org/gitlab'
|
|
'ee'
|
|
when 'gitlab-org/gitlab-runner'
|
|
'runner'
|
|
when 'gitlab-org/omnibus-gitlab'
|
|
'omnibus'
|
|
when 'gitlab-org/charts/gitlab'
|
|
'charts'
|
|
when 'gitlab-org/cloud-native/gitlab-operator'
|
|
'operator'
|
|
end
|
|
end
|
|
|
|
def app_url
|
|
"https://docs.gitlab.com/upstream-review-mr-#{review_slug}/"
|
|
end
|
|
|
|
def display_success_message
|
|
puts format(SUCCESS_MESSAGE, app_url: app_url)
|
|
end
|
|
end
|
|
|
|
class DatabaseTesting < Base
|
|
IDENTIFIABLE_NOTE_TAG = 'gitlab-org/database-team/gitlab-com-database-testing:identifiable-note'
|
|
|
|
def invoke!
|
|
pipeline = super
|
|
project_path = variables['TOP_UPSTREAM_SOURCE_PROJECT']
|
|
merge_request_id = variables['TOP_UPSTREAM_MERGE_REQUEST_IID']
|
|
comment = <<~COMMENT.strip
|
|
<!-- #{IDENTIFIABLE_NOTE_TAG} -->
|
|
Started database testing [pipeline](https://ops.gitlab.net/#{downstream_project_path}/-/pipelines/#{pipeline.id}) (limited access). This comment will be updated once the pipeline has finished running.
|
|
COMMENT
|
|
|
|
# Look for an existing note
|
|
db_testing_notes = com_gitlab_client
|
|
.merge_request_notes(project_path, merge_request_id)
|
|
.auto_paginate.select do |note|
|
|
note.body.include?(IDENTIFIABLE_NOTE_TAG)
|
|
end
|
|
|
|
return unless db_testing_notes.empty?
|
|
|
|
# This is the first note
|
|
note = com_gitlab_client.create_merge_request_note(project_path, merge_request_id, comment)
|
|
|
|
puts "Posted comment to:\n"
|
|
puts "https://gitlab.com/#{project_path}/-/merge_requests/#{merge_request_id}#note_#{note.id}"
|
|
end
|
|
|
|
private
|
|
|
|
def ops_gitlab_client
|
|
# No access token is needed here - we only use this client to trigger pipelines,
|
|
# and the trigger token authenticates the request to the pipeline
|
|
@ops_gitlab_client ||= Gitlab.client(
|
|
endpoint: 'https://ops.gitlab.net/api/v4'
|
|
)
|
|
end
|
|
|
|
def downstream_client
|
|
ops_gitlab_client
|
|
end
|
|
|
|
def trigger_token
|
|
ENV['GITLABCOM_DATABASE_TESTING_TRIGGER_TOKEN']
|
|
end
|
|
|
|
def downstream_project_path
|
|
ENV.fetch('GITLABCOM_DATABASE_TESTING_PROJECT_PATH', 'gitlab-com/database-team/gitlab-com-database-testing')
|
|
end
|
|
|
|
def extra_variables
|
|
{
|
|
'GITLAB_COMMIT_SHA' => Trigger.non_empty_variable_value('CI_MERGE_REQUEST_SOURCE_BRANCH_SHA') || ENV['CI_COMMIT_SHA'],
|
|
'TRIGGERED_USER_LOGIN' => ENV['GITLAB_USER_LOGIN'],
|
|
'TOP_UPSTREAM_SOURCE_SHA' => Trigger.non_empty_variable_value('CI_MERGE_REQUEST_SOURCE_BRANCH_SHA') || ENV['CI_COMMIT_SHA']
|
|
}
|
|
end
|
|
|
|
def ref_param_name
|
|
'GITLABCOM_DATABASE_TESTING_TRIGGER_REF'
|
|
end
|
|
|
|
def primary_ref
|
|
'master'
|
|
end
|
|
end
|
|
|
|
class Pipeline
|
|
INTERVAL = 60 # seconds
|
|
MAX_DURATION = 3600 * 3 # 3 hours
|
|
|
|
attr_reader :id
|
|
|
|
def self.unscoped_class_name
|
|
name.split('::').last
|
|
end
|
|
|
|
def self.gitlab_api_method_name
|
|
unscoped_class_name.downcase
|
|
end
|
|
|
|
def initialize(project, id, gitlab_client)
|
|
@project = project
|
|
@id = id
|
|
@gitlab_client = gitlab_client
|
|
@start_time = Time.now.to_i
|
|
end
|
|
|
|
def wait!
|
|
(MAX_DURATION / INTERVAL).times do
|
|
case status
|
|
when :created, :pending, :running
|
|
print "."
|
|
sleep INTERVAL
|
|
when :success
|
|
puts "#{self.class.unscoped_class_name} succeeded in #{duration} minutes!"
|
|
return
|
|
else
|
|
raise "#{self.class.unscoped_class_name} did not succeed!"
|
|
end
|
|
|
|
$stdout.flush
|
|
end
|
|
|
|
raise "#{self.class.unscoped_class_name} timed out after waiting for #{duration} minutes!"
|
|
end
|
|
|
|
def duration
|
|
(Time.now.to_i - start_time) / 60
|
|
end
|
|
|
|
def status
|
|
gitlab_client.public_send(self.class.gitlab_api_method_name, project, id).status.to_sym # rubocop:disable GitlabSecurity/PublicSend
|
|
rescue Gitlab::Error::Error => error
|
|
puts "Ignoring the following error: #{error}"
|
|
# Ignore GitLab API hiccups. If GitLab is really down, we'll hit the job
|
|
# timeout anyway.
|
|
:running
|
|
end
|
|
|
|
private
|
|
|
|
attr_reader :project, :gitlab_client, :start_time
|
|
end
|
|
end
|
|
|
|
if $PROGRAM_NAME == __FILE__
|
|
case ARGV[0]
|
|
when 'gitlab-com-database-testing'
|
|
Trigger::DatabaseTesting.new.invoke!
|
|
when 'docs'
|
|
docs_trigger = Trigger::Docs.new
|
|
|
|
case ARGV[1]
|
|
when 'deploy'
|
|
docs_trigger.deploy!
|
|
when 'cleanup'
|
|
docs_trigger.cleanup!
|
|
else
|
|
puts 'usage: trigger-build docs <deploy|cleanup>'
|
|
exit 1
|
|
end
|
|
else
|
|
puts "Please provide a valid option:
|
|
docs - Triggers a pipeline that builds a documentation review app by using the docs-gitlab-com project
|
|
omnibus - Triggers a pipeline that builds the omnibus-gitlab package
|
|
gitlab-com-database-testing - Triggers a pipeline that tests database changes on GitLab.com data"
|
|
end
|
|
end
|