2019-10-12 21:52:04 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
require 'rspec/mocks'
2018-03-27 19:54:05 +05:30
require 'toml-rb'
2014-09-02 18:07:02 +05:30
module TestEnv
2019-09-30 21:07:59 +05:30
extend ActiveSupport :: Concern
2014-09-02 18:07:02 +05:30
extend self
2018-03-17 18:26:18 +05:30
ComponentFailedToInstallError = Class . new ( StandardError )
2015-04-26 12:48:37 +05:30
# When developing the seed repository, comment out the branch you will modify.
BRANCH_SHA = {
2018-11-18 11:00:15 +05:30
'signed-commits' = > '6101e87' ,
2016-11-03 12:29:30 +05:30
'not-merged-branch' = > 'b83d6e3' ,
'branch-merged' = > '498214d' ,
2016-09-13 17:45:13 +05:30
'empty-branch' = > '7efb185' ,
2016-09-29 09:46:39 +05:30
'ends-with.json' = > '98b0d8b' ,
2016-09-13 17:45:13 +05:30
'flatten-dir' = > 'e56497b' ,
'feature' = > '0b4bc9a' ,
'feature_conflict' = > 'bb5206f' ,
'fix' = > '48f0be4' ,
'improve/awesome' = > '5937ac0' ,
2018-03-17 18:26:18 +05:30
'merged-target' = > '21751bf' ,
2016-09-13 17:45:13 +05:30
'markdown' = > '0ed8c6c' ,
2018-03-17 18:26:18 +05:30
'lfs' = > '55bc176' ,
2016-11-03 12:29:30 +05:30
'master' = > 'b83d6e3' ,
'merge-test' = > '5937ac0' ,
2016-09-13 17:45:13 +05:30
" 'test' " = > 'e56497b' ,
'orphaned-branch' = > '45127a9' ,
'binary-encoding' = > '7b1cf43' ,
'gitattributes' = > '5a62481' ,
'expand-collapse-diffs' = > '4842455' ,
2017-08-17 22:00:37 +05:30
'symlink-expand-diff' = > '81e6355' ,
2016-09-13 17:45:13 +05:30
'expand-collapse-files' = > '025db92' ,
'expand-collapse-lines' = > '238e82d' ,
2018-12-13 13:39:08 +05:30
'pages-deploy' = > '7897d5b' ,
'pages-deploy-target' = > '7975be0' ,
2019-12-21 20:55:43 +05:30
'audio' = > 'c3c21fd' ,
2016-09-13 17:45:13 +05:30
'video' = > '8879059' ,
2017-08-17 22:00:37 +05:30
'add-balsamiq-file' = > 'b89b56d' ,
2016-09-13 17:45:13 +05:30
'crlf-diff' = > '5938907' ,
2016-11-03 12:29:30 +05:30
'conflict-start' = > '824be60' ,
2016-09-13 17:45:13 +05:30
'conflict-resolvable' = > '1450cd6' ,
'conflict-binary-file' = > '259a6fb' ,
2016-11-03 12:29:30 +05:30
'conflict-contains-conflict-markers' = > '78a3086' ,
2016-09-13 17:45:13 +05:30
'conflict-missing-side' = > 'eb227b3' ,
'conflict-non-utf8' = > 'd0a293c' ,
'conflict-too-large' = > '39fa04f' ,
2017-08-17 22:00:37 +05:30
'deleted-image-test' = > '6c17798' ,
'wip' = > 'b9238ee' ,
'csv' = > '3dd0896' ,
'v1.1.0' = > 'b83d6e3' ,
2017-09-10 17:25:29 +05:30
'add-ipython-files' = > '93ee732' ,
'add-pdf-file' = > 'e774ebd' ,
2018-11-08 19:23:39 +05:30
'squash-large-files' = > '54cec52' ,
2018-03-17 18:26:18 +05:30
'add-pdf-text-binary' = > '79faa7b' ,
2018-11-08 19:23:39 +05:30
'add_images_and_changes' = > '010d106' ,
'update-gitlab-shell-v-6-0-1' = > '2f61d70' ,
2018-11-18 11:00:15 +05:30
'update-gitlab-shell-v-6-0-3' = > 'de78448' ,
2019-02-15 15:39:39 +05:30
'merge-commit-analyze-before' = > '1adbdef' ,
'merge-commit-analyze-side-branch' = > '8a99451' ,
'merge-commit-analyze-after' = > '646ece5' ,
2020-04-08 14:13:33 +05:30
'snippet/single-file' = > '43e4080' ,
'snippet/multiple-files' = > 'b80faa8' ,
'snippet/rename-and-edit-file' = > '220a1e4' ,
'snippet/edit-file' = > 'c2f074f' ,
'snippet/no-files' = > '671aaa8' ,
2018-11-20 20:47:30 +05:30
'2-mb-file' = > 'bf12d25' ,
2018-12-13 13:39:08 +05:30
'before-create-delete-modify-move' = > '845009f' ,
'between-create-delete-modify-move' = > '3f5f443' ,
'after-create-delete-modify-move' = > 'ba3faa7' ,
'with-codeowners' = > '219560e' ,
2019-02-15 15:39:39 +05:30
'submodule_inside_folder' = > 'b491b92' ,
2019-07-07 11:18:12 +05:30
'png-lfs' = > 'fe42f41' ,
2019-09-30 21:07:59 +05:30
'sha-starting-with-large-number' = > '8426165' ,
'invalid-utf8-diff-paths' = > '99e4853'
2017-08-17 22:00:37 +05:30
} . freeze
2015-09-11 14:41:01 +05:30
# gitlab-test-fork is a fork of gitlab-fork, but we don't necessarily
# need to keep all the branches in sync.
# We currently only need a subset of the branches
FORKED_BRANCH_SHA = {
2016-09-29 09:46:39 +05:30
'add-submodule-version-bump' = > '3f547c0' ,
'master' = > '5937ac0' ,
'remove-submodule' = > '2a33e0c' ,
'conflict-resolvable-fork' = > '404fa3f'
2017-08-17 22:00:37 +05:30
} . freeze
2015-04-26 12:48:37 +05:30
2017-09-10 17:25:29 +05:30
TMP_TEST_PATH = Rails . root . join ( 'tmp' , 'tests' , '**' )
2018-05-09 12:01:36 +05:30
REPOS_STORAGE = 'default' . freeze
2020-04-08 14:13:33 +05:30
SECOND_STORAGE_PATH = Rails . root . join ( 'tmp' , 'tests' , 'second_storage' )
2017-09-10 17:25:29 +05:30
2014-09-02 18:07:02 +05:30
# Test environment
#
# See gitlab.yml.example test section for paths
#
def init ( opts = { } )
2018-03-17 18:26:18 +05:30
unless Rails . env . test?
puts " \n TestEnv.init can only be run if `RAILS_ENV` is set to 'test' not ' #{ Rails . env } '! \n "
exit 1
end
2014-09-02 18:07:02 +05:30
# Disable mailer for spinach tests
disable_mailer if opts [ :mailer ] == false
2015-04-26 12:48:37 +05:30
clean_test_path
2014-09-02 18:07:02 +05:30
setup_gitlab_shell
2017-09-10 17:25:29 +05:30
setup_gitaly
2017-08-17 22:00:37 +05:30
2020-04-08 14:13:33 +05:30
# Feature specs are run through Workhorse
setup_workhorse
2018-03-17 18:26:18 +05:30
# Create repository for FactoryBot.create(:project)
2014-09-02 18:07:02 +05:30
setup_factory_repo
2015-09-11 14:41:01 +05:30
2018-03-17 18:26:18 +05:30
# Create repository for FactoryBot.create(:forked_project_with_submodules)
2015-09-11 14:41:01 +05:30
setup_forked_repo
2014-09-02 18:07:02 +05:30
end
2019-09-30 21:07:59 +05:30
included do | config |
config . append_before do
set_current_example_group
end
end
2014-09-02 18:07:02 +05:30
def disable_mailer
2017-09-10 17:25:29 +05:30
allow_any_instance_of ( NotificationService ) . to receive ( :mailer )
. and_return ( double . as_null_object )
2014-09-02 18:07:02 +05:30
end
def enable_mailer
2017-09-10 17:25:29 +05:30
allow_any_instance_of ( NotificationService ) . to receive ( :mailer )
. and_call_original
2015-04-26 12:48:37 +05:30
end
# Clean /tmp/tests
#
# Keeps gitlab-shell and gitlab-test
def clean_test_path
2017-09-10 17:25:29 +05:30
Dir [ TMP_TEST_PATH ] . each do | entry |
2019-10-12 21:52:04 +05:30
unless test_dirs . include? ( File . basename ( entry ) )
2015-04-26 12:48:37 +05:30
FileUtils . rm_rf ( entry )
end
end
2017-08-17 22:00:37 +05:30
FileUtils . mkdir_p ( repos_path )
2020-04-08 14:13:33 +05:30
FileUtils . mkdir_p ( SECOND_STORAGE_PATH )
2017-08-17 22:00:37 +05:30
FileUtils . mkdir_p ( backup_path )
FileUtils . mkdir_p ( pages_path )
2018-03-17 18:26:18 +05:30
FileUtils . mkdir_p ( artifacts_path )
2014-09-02 18:07:02 +05:30
end
2017-09-10 17:25:29 +05:30
def setup_gitlab_shell
2019-12-21 20:55:43 +05:30
FileUtils . mkdir_p ( Gitlab . config . gitlab_shell . path )
2017-09-10 17:25:29 +05:30
end
2017-08-17 22:00:37 +05:30
def setup_gitaly
2019-02-02 18:00:53 +05:30
install_gitaly_args = [ gitaly_dir , repos_path , gitaly_url ] . compact . join ( ',' )
2017-08-17 22:00:37 +05:30
2018-03-17 18:26:18 +05:30
component_timed_setup ( 'Gitaly' ,
install_dir : gitaly_dir ,
version : Gitlab :: GitalyClient . expected_server_version ,
2019-02-02 18:00:53 +05:30
task : " gitlab:gitaly:install[ #{ install_gitaly_args } ] " ) do
2019-02-15 15:39:39 +05:30
Gitlab :: SetupHelper . create_gitaly_configuration ( gitaly_dir , { 'default' = > repos_path } , force : true )
start_gitaly ( gitaly_dir )
end
2017-08-17 22:00:37 +05:30
end
2019-12-26 22:10:19 +05:30
def gitaly_socket_path
Gitlab :: GitalyClient . address ( 'default' ) . sub ( / \ Aunix: / , '' )
end
def gitaly_dir
File . dirname ( gitaly_socket_path )
end
2017-08-17 22:00:37 +05:30
def start_gitaly ( gitaly_dir )
2019-12-26 22:10:19 +05:30
if ci?
2017-09-10 17:25:29 +05:30
# Gitaly has been spawned outside this process already
return
end
spawn_script = Rails . root . join ( 'scripts/gitaly-test-spawn' ) . to_s
2018-10-15 14:42:47 +05:30
Bundler . with_original_env do
2019-12-26 22:10:19 +05:30
unless system ( spawn_script )
message = 'gitaly spawn failed'
message += " (try `rm -rf #{ gitaly_dir } ` ?) " unless ci?
raise message
end
2018-10-15 14:42:47 +05:30
end
2019-12-26 22:10:19 +05:30
2018-10-15 14:42:47 +05:30
@gitaly_pid = Integer ( File . read ( 'tmp/tests/gitaly.pid' ) )
2018-03-17 18:26:18 +05:30
Kernel . at_exit { stop_gitaly }
wait_gitaly
end
def wait_gitaly
sleep_time = 10
sleep_interval = 0 . 1
socket = Gitlab :: GitalyClient . address ( 'default' ) . sub ( 'unix:' , '' )
Integer ( sleep_time / sleep_interval ) . times do
2019-07-07 11:18:12 +05:30
Socket . unix ( socket )
return
rescue
sleep sleep_interval
2018-03-17 18:26:18 +05:30
end
raise " could not connect to gitaly at #{ socket . inspect } after #{ sleep_time } seconds "
2017-08-17 22:00:37 +05:30
end
def stop_gitaly
return unless @gitaly_pid
Process . kill ( 'KILL' , @gitaly_pid )
2018-03-17 18:26:18 +05:30
rescue Errno :: ESRCH
# The process can already be gone if the test run was INTerrupted.
2017-08-17 22:00:37 +05:30
end
2019-02-02 18:00:53 +05:30
def gitaly_url
ENV . fetch ( 'GITALY_REPO_URL' , nil )
end
2020-04-08 14:13:33 +05:30
def setup_workhorse
install_workhorse_args = [ workhorse_dir , workhorse_url ] . compact . join ( ',' )
component_timed_setup (
'GitLab Workhorse' ,
install_dir : workhorse_dir ,
version : Gitlab :: Workhorse . version ,
task : " gitlab:workhorse:install[ #{ install_workhorse_args } ] "
)
end
def workhorse_dir
@workhorse_path || = File . join ( 'tmp' , 'tests' , 'gitlab-workhorse' )
end
def with_workhorse ( workhorse_dir , host , port , upstream , & blk )
host = " [ #{ host } ] " if host . include? ( ':' )
listen_addr = [ host , port ] . join ( ':' )
workhorse_pid = spawn (
File . join ( workhorse_dir , 'gitlab-workhorse' ) ,
'-authSocket' , upstream ,
'-documentRoot' , Rails . root . join ( 'public' ) . to_s ,
'-listenAddr' , listen_addr ,
'-secretPath' , Gitlab :: Workhorse . secret_path . to_s ,
# TODO: Needed for workhorse + redis features.
# https://gitlab.com/gitlab-org/gitlab/-/issues/209245
#
# '-config', '',
'-logFile' , 'log/workhorse-test.log' ,
'-logFormat' , 'structured' ,
'-developmentMode' # to serve assets and rich error messages
)
begin
yield
ensure
Process . kill ( 'TERM' , workhorse_pid )
Process . wait ( workhorse_pid )
end
end
def workhorse_url
ENV . fetch ( 'GITLAB_WORKHORSE_URL' , nil )
end
2014-09-02 18:07:02 +05:30
def setup_factory_repo
2015-09-11 14:41:01 +05:30
setup_repo ( factory_repo_path , factory_repo_path_bare , factory_repo_name ,
BRANCH_SHA )
end
# This repo has a submodule commit that is not present in the main test
# repository.
def setup_forked_repo
setup_repo ( forked_repo_path , forked_repo_path_bare , forked_repo_name ,
FORKED_BRANCH_SHA )
end
2015-04-26 12:48:37 +05:30
2017-09-10 17:25:29 +05:30
def setup_repo ( repo_path , repo_path_bare , repo_name , refs )
2015-09-11 14:41:01 +05:30
clone_url = " https://gitlab.com/gitlab-org/ #{ repo_name } .git "
unless File . directory? ( repo_path )
2015-11-26 14:37:03 +05:30
system ( * %W( #{ Gitlab . config . git . bin_path } clone -q #{ clone_url } #{ repo_path } ) )
2015-04-26 12:48:37 +05:30
end
2014-09-02 18:07:02 +05:30
2017-09-10 17:25:29 +05:30
set_repo_refs ( repo_path , refs )
2015-04-26 12:48:37 +05:30
2017-08-17 22:00:37 +05:30
unless File . directory? ( repo_path_bare )
# We must copy bare repositories because we will push to them.
system ( git_env , * %W( #{ Gitlab . config . git . bin_path } clone -q --bare #{ repo_path } #{ repo_path_bare } ) )
end
2014-09-02 18:07:02 +05:30
end
2020-03-09 13:42:32 +05:30
def copy_repo ( subject , bare_repo : , refs : )
target_repo_path = File . expand_path ( repos_path + " / #{ subject . disk_path } .git " )
2018-10-15 14:42:47 +05:30
2014-09-02 18:07:02 +05:30
FileUtils . mkdir_p ( target_repo_path )
2017-09-10 17:25:29 +05:30
FileUtils . cp_r ( " #{ File . expand_path ( bare_repo ) } /. " , target_repo_path )
2014-09-02 18:07:02 +05:30
FileUtils . chmod_R 0755 , target_repo_path
end
2019-12-26 22:10:19 +05:30
def rm_storage_dir ( storage , dir )
Gitlab :: GitalyClient :: StorageSettings . allow_disk_access do
repos_path = Gitlab . config . repositories . storages [ storage ] . legacy_disk_path
target_repo_refs_path = File . join ( repos_path , dir )
FileUtils . remove_dir ( target_repo_refs_path )
end
rescue Errno :: ENOENT
end
def storage_dir_exists? ( storage , dir )
Gitlab :: GitalyClient :: StorageSettings . allow_disk_access do
repos_path = Gitlab . config . repositories . storages [ storage ] . legacy_disk_path
File . exist? ( File . join ( repos_path , dir ) )
end
end
2018-11-18 11:00:15 +05:30
def create_bare_repository ( path )
FileUtils . mkdir_p ( path )
system ( git_env , * %W( #{ Gitlab . config . git . bin_path } -C #{ path } init --bare ) ,
out : '/dev/null' ,
err : '/dev/null' )
end
2014-09-02 18:07:02 +05:30
def repos_path
2018-10-15 14:42:47 +05:30
@repos_path || = Gitlab . config . repositories . storages [ REPOS_STORAGE ] . legacy_disk_path
2014-09-02 18:07:02 +05:30
end
2015-04-26 12:48:37 +05:30
2015-09-11 14:41:01 +05:30
def backup_path
Gitlab . config . backup . path
end
2017-08-17 22:00:37 +05:30
def pages_path
Gitlab . config . pages . path
end
2018-03-17 18:26:18 +05:30
def artifacts_path
Gitlab . config . artifacts . storage_path
end
2016-04-02 18:10:28 +05:30
# When no cached assets exist, manually hit the root path to create them
#
# Otherwise they'd be created by the first test, often timing out and
# causing a transient test failure
2017-08-17 22:00:37 +05:30
def eager_load_driver_server
2016-04-02 18:10:28 +05:30
return unless defined? ( Capybara )
2017-08-17 22:00:37 +05:30
puts " Starting the Capybara driver server... "
Capybara . current_session . visit '/'
2016-04-02 18:10:28 +05:30
end
2017-09-10 17:25:29 +05:30
def factory_repo_path_bare
" #{ factory_repo_path } _bare "
end
def forked_repo_path_bare
" #{ forked_repo_path } _bare "
end
def with_empty_bare_repository ( name = nil )
path = Rails . root . join ( 'tmp/tests' , name || 'empty-bare-repository' ) . to_s
yield ( Rugged :: Repository . init_at ( path , :bare ) )
ensure
FileUtils . rm_rf ( path )
end
2019-09-30 21:07:59 +05:30
def current_example_group
Thread . current [ :current_example_group ]
end
# looking for a top-level `describe`
def topmost_example_group
example_group = current_example_group
example_group = example_group [ :parent_example_group ] until example_group [ :parent_example_group ] . nil?
example_group
end
2015-04-26 12:48:37 +05:30
private
2019-09-30 21:07:59 +05:30
def set_current_example_group
Thread . current [ :current_example_group ] = :: RSpec . current_example . metadata [ :example_group ]
end
2019-09-04 21:01:54 +05:30
# These are directories that should be preserved at cleanup time
def test_dirs
@test_dirs || = %w[
2019-10-12 21:52:04 +05:30
frontend
2019-09-04 21:01:54 +05:30
gitaly
gitlab - shell
gitlab - test
gitlab - test_bare
gitlab - test - fork
gitlab - test - fork_bare
2020-04-08 14:13:33 +05:30
gitlab - workhorse
gitlab_workhorse_secret
2019-09-04 21:01:54 +05:30
]
end
2015-04-26 12:48:37 +05:30
def factory_repo_path
@factory_repo_path || = Rails . root . join ( 'tmp' , 'tests' , factory_repo_name )
end
def factory_repo_name
'gitlab-test'
end
2015-09-11 14:41:01 +05:30
def forked_repo_path
@forked_repo_path || = Rails . root . join ( 'tmp' , 'tests' , forked_repo_name )
end
def forked_repo_name
'gitlab-test-fork'
end
2015-04-26 12:48:37 +05:30
# Prevent developer git configurations from being persisted to test
# repositories
def git_env
2015-09-11 14:41:01 +05:30
{ 'GIT_TEMPLATE_DIR' = > '' }
2015-04-26 12:48:37 +05:30
end
2016-09-29 09:46:39 +05:30
def set_repo_refs ( repo_path , branch_sha )
2017-09-10 17:25:29 +05:30
instructions = branch_sha . map { | branch , sha | " update refs/heads/ #{ branch } \x00 #{ sha } \x00 " } . join ( " \x00 " ) << " \x00 "
2017-08-17 22:00:37 +05:30
update_refs = %W( #{ Gitlab . config . git . bin_path } update-ref --stdin -z )
reset = proc do
2017-09-10 17:25:29 +05:30
Dir . chdir ( repo_path ) do
IO . popen ( update_refs , " w " ) { | io | io . write ( instructions ) }
$? . success?
end
2017-08-17 22:00:37 +05:30
end
2017-09-10 17:25:29 +05:30
# Try to reset without fetching to avoid using the network.
unless reset . call
raise 'Could not fetch test seed repository.' unless system ( * %W( #{ Gitlab . config . git . bin_path } -C #{ repo_path } fetch origin ) )
2019-10-12 21:52:04 +05:30
raise " Could not update test seed repository, please delete #{ repo_path } and try again " unless reset . call
2018-03-17 18:26:18 +05:30
end
end
def component_timed_setup ( component , install_dir : , version : , task : )
puts " \n ==> Setting up #{ component } ... "
start = Time . now
ensure_component_dir_name_is_correct! ( component , install_dir )
# On CI, once installed, components never need update
2019-12-26 22:10:19 +05:30
return if File . exist? ( install_dir ) && ci?
2018-03-17 18:26:18 +05:30
if component_needs_update? ( install_dir , version )
# Cleanup the component entirely to ensure we start fresh
FileUtils . rm_rf ( install_dir )
unless system ( 'rake' , task )
raise ComponentFailedToInstallError
end
end
yield if block_given?
rescue ComponentFailedToInstallError
puts " \n #{ component } failed to install, cleaning up #{ install_dir } ! \n "
FileUtils . rm_rf ( install_dir )
exit 1
ensure
2018-12-05 23:21:45 +05:30
puts " #{ component } set up in #{ Time . now - start } seconds... \n "
2018-03-17 18:26:18 +05:30
end
2019-12-26 22:10:19 +05:30
def ci?
ENV [ 'CI' ] . present?
end
2018-03-17 18:26:18 +05:30
def ensure_component_dir_name_is_correct! ( component , path )
actual_component_dir_name = File . basename ( path )
expected_component_dir_name = component . parameterize
unless actual_component_dir_name == expected_component_dir_name
puts " #{ component } install dir should be named ' #{ expected_component_dir_name } ', not ' #{ actual_component_dir_name } ' (full install path given was ' #{ path } ')! \n "
exit 1
2016-09-29 09:46:39 +05:30
end
end
2017-09-10 17:25:29 +05:30
def component_needs_update? ( component_folder , expected_version )
2018-03-17 18:26:18 +05:30
# Allow local overrides of the component for tests during development
return false if Rails . env . test? && File . symlink? ( component_folder )
2017-09-10 17:25:29 +05:30
version = File . read ( File . join ( component_folder , 'VERSION' ) ) . strip
# Notice that this will always yield true when using branch versions
# (`=branch_name`), but that actually makes sure the server is always based
# on the latest branch revision.
version != expected_version
rescue Errno :: ENOENT
true
end
2014-09-02 18:07:02 +05:30
end
2019-12-04 20:38:33 +05:30
require_relative ( '../../../ee/spec/support/helpers/ee/test_env' ) if Gitlab . ee?
:: TestEnv . prepend_if_ee ( '::EE::TestEnv' )
:: TestEnv . extend_if_ee ( '::EE::TestEnv' )