BigW Consortium Gitlab

test_env.rb 7.16 KB
Newer Older
1 2
require 'rspec/mocks'

3 4 5
module TestEnv
  extend self

6 7
  # When developing the seed repository, comment out the branch you will modify.
  BRANCH_SHA = {
8 9
    'not-merged-branch'                  => 'b83d6e3',
    'branch-merged'                      => '498214d',
10
    'empty-branch'                       => '7efb185',
11
    'ends-with.json'                     => '98b0d8b',
12 13 14 15 16 17 18
    'flatten-dir'                        => 'e56497b',
    'feature'                            => '0b4bc9a',
    'feature_conflict'                   => 'bb5206f',
    'fix'                                => '48f0be4',
    'improve/awesome'                    => '5937ac0',
    'markdown'                           => '0ed8c6c',
    'lfs'                                => 'be93687',
19
    'master'                             => 'b83d6e3',
20
    'merge-test'                         => '5937ac0',
21 22 23 24 25
    "'test'"                             => 'e56497b',
    'orphaned-branch'                    => '45127a9',
    'binary-encoding'                    => '7b1cf43',
    'gitattributes'                      => '5a62481',
    'expand-collapse-diffs'              => '4842455',
26
    'symlink-expand-diff'                => '81e6355',
27 28 29 30
    'expand-collapse-files'              => '025db92',
    'expand-collapse-lines'              => '238e82d',
    'video'                              => '8879059',
    'crlf-diff'                          => '5938907',
Sean McGivern committed
31
    'conflict-start'                     => '824be60',
32 33
    'conflict-resolvable'                => '1450cd6',
    'conflict-binary-file'               => '259a6fb',
Sean McGivern committed
34
    'conflict-contains-conflict-markers' => '78a3086',
35
    'conflict-missing-side'              => 'eb227b3',
36
    'conflict-non-utf8'                  => 'd0a293c',
37
    'conflict-too-large'                 => '39fa04f',
38 39
    'deleted-image-test'                 => '6c17798',
    'wip'                                => 'b9238ee'
40 41
  }

42 43 44 45
  # 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 = {
46 47 48 49
    'add-submodule-version-bump' => '3f547c0',
    'master'                     => '5937ac0',
    'remove-submodule'           => '2a33e0c',
    'conflict-resolvable-fork'   => '404fa3f'
50
  }
51

52 53
  # Test environment
  #
54
  # See gitlab.yml.example test section for paths
55
  #
56
  def init(opts = {})
57 58 59
    # Disable mailer for spinach tests
    disable_mailer if opts[:mailer] == false

Robert Speicher committed
60
    clean_test_path
61

62
    FileUtils.mkdir_p(repos_path)
63
    FileUtils.mkdir_p(backup_path)
64

65 66 67
    # Setup GitLab shell for test instance
    setup_gitlab_shell

68 69
    # Create repository for FactoryGirl.create(:project)
    setup_factory_repo
70

71 72
    # Create repository for FactoryGirl.create(:forked_project_with_submodules)
    setup_forked_repo
73 74
  end

75
  def disable_mailer
76 77
    allow_any_instance_of(NotificationService).to receive(:mailer).
      and_return(double.as_null_object)
78
  end
79

80
  def enable_mailer
81 82
    allow_any_instance_of(NotificationService).to receive(:mailer).
      and_call_original
83 84
  end

85
  def disable_pre_receive
86
    allow_any_instance_of(Gitlab::Git::Hook).to receive(:trigger).and_return([true, nil])
87 88
  end

Robert Speicher committed
89 90 91 92 93 94 95
  # Clean /tmp/tests
  #
  # Keeps gitlab-shell and gitlab-test
  def clean_test_path
    tmp_test_path = Rails.root.join('tmp', 'tests', '**')

    Dir[tmp_test_path].each do |entry|
96
      unless File.basename(entry) =~ /\Agitlab-(shell|test|test-fork)\z/
Robert Speicher committed
97 98 99 100 101
        FileUtils.rm_rf(entry)
      end
    end
  end

102
  def setup_gitlab_shell
103
    unless File.directory?(Gitlab.config.gitlab_shell.path)
104 105 106
      unless system('rake', 'gitlab:shell:install')
        raise 'Can`t clone gitlab-shell'
      end
Robert Speicher committed
107
    end
108
  end
109

110
  def setup_factory_repo
111 112 113 114 115 116 117 118 119 120 121 122 123
    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

  def setup_repo(repo_path, repo_path_bare, repo_name, branch_sha)
    clone_url = "https://gitlab.com/gitlab-org/#{repo_name}.git"
124

125
    unless File.directory?(repo_path)
126
      system(*%W(#{Gitlab.config.git.bin_path} clone -q #{clone_url} #{repo_path}))
127 128
    end

129
    set_repo_refs(repo_path, branch_sha)
130 131

    # We must copy bare repositories because we will push to them.
132
    system(git_env, *%W(#{Gitlab.config.git.bin_path} clone -q --bare #{repo_path} #{repo_path_bare}))
133 134
  end

135
  def copy_repo(project)
136
    base_repo_path = File.expand_path(factory_repo_path_bare)
137
    target_repo_path = File.expand_path(project.repository_storage_path + "/#{project.namespace.path}/#{project.path}.git")
138 139 140
    FileUtils.mkdir_p(target_repo_path)
    FileUtils.cp_r("#{base_repo_path}/.", target_repo_path)
    FileUtils.chmod_R 0755, target_repo_path
141
    set_repo_refs(target_repo_path, BRANCH_SHA)
142 143
  end

144
  def repos_path
145
    Gitlab.config.repositories.storages.default
146
  end
147

148 149 150 151
  def backup_path
    Gitlab.config.backup.path
  end

152 153
  def copy_forked_repo_with_submodules(project)
    base_repo_path = File.expand_path(forked_repo_path_bare)
154
    target_repo_path = File.expand_path(project.repository_storage_path + "/#{project.namespace.path}/#{project.path}.git")
155 156 157
    FileUtils.mkdir_p(target_repo_path)
    FileUtils.cp_r("#{base_repo_path}/.", target_repo_path)
    FileUtils.chmod_R 0755, target_repo_path
158
    set_repo_refs(target_repo_path, FORKED_BRANCH_SHA)
159 160
  end

161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176
  # 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
  def warm_asset_cache
    return if warm_asset_cache?
    return unless defined?(Capybara)

    Capybara.current_session.driver.visit '/'
  end

  def warm_asset_cache?
    cache = Rails.root.join(*%w(tmp cache assets test))
    Dir.exist?(cache) && Dir.entries(cache).length > 2
  end

177 178 179
  private

  def factory_repo_path
180 181 182 183
    @factory_repo_path ||= Rails.root.join('tmp', 'tests', factory_repo_name)
  end

  def factory_repo_path_bare
Robert Speicher committed
184
    "#{factory_repo_path}_bare"
185 186 187 188 189
  end

  def factory_repo_name
    'gitlab-test'
  end
190

191 192 193 194 195 196 197 198 199 200 201 202
  def forked_repo_path
    @forked_repo_path ||= Rails.root.join('tmp', 'tests', forked_repo_name)
  end

  def forked_repo_path_bare
    "#{forked_repo_path}_bare"
  end

  def forked_repo_name
    'gitlab-test-fork'
  end

203 204 205
  # Prevent developer git configurations from being persisted to test
  # repositories
  def git_env
206
    { 'GIT_TEMPLATE_DIR' => '' }
207
  end
208 209

  def set_repo_refs(repo_path, branch_sha)
210 211 212 213 214 215 216
    instructions = branch_sha.map {|branch, sha| "update refs/heads/#{branch}\x00#{sha}\x00" }.join("\x00") << "\x00"
    update_refs = %W(#{Gitlab.config.git.bin_path} update-ref --stdin -z)
    reset = proc do
      IO.popen(update_refs, "w") {|io| io.write(instructions) }
      $?.success?
    end

217
    Dir.chdir(repo_path) do
218 219 220 221
      # 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} fetch origin))
        raise 'The fetched test seed does not contain the required revision.' unless reset.call
222 223 224
      end
    end
  end
225
end