summaryrefslogtreecommitdiff
path: root/spec/support/bare_repo_operations.rb
blob: 38d11992dc285c657d547dbbb841f8d18be5948d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
require 'zlib'

class BareRepoOperations
  # The ID of empty tree.
  # See http://stackoverflow.com/a/40884093/1856239 and https://github.com/git/git/blob/3ad8b5bf26362ac67c9020bf8c30eee54a84f56d/cache.h#L1011-L1012
  EMPTY_TREE_ID = '4b825dc642cb6eb9a060e54bf8d69288fbee4904'.freeze

  include Gitlab::Popen

  def initialize(path_to_repo)
    @path_to_repo = path_to_repo
  end

  # Based on https://stackoverflow.com/a/25556917/1856239
  def commit_file(file, dst_path, branch = 'master')
    head_id = execute(['show', '--format=format:%H', '--no-patch', branch], allow_failure: true)[0] || EMPTY_TREE_ID

    execute(['read-tree', '--empty'])
    execute(['read-tree', head_id])

    blob_id = execute(['hash-object', '--stdin', '-w']) do |stdin|
      stdin.write(file.read)
    end

    execute(['update-index', '--add', '--cacheinfo', '100644', blob_id[0], dst_path])

    tree_id = execute(['write-tree'])

    commit_tree_args = ['commit-tree', tree_id[0], '-m', "Add #{dst_path}"]
    commit_tree_args += ['-p', head_id] unless head_id == EMPTY_TREE_ID
    commit_id = execute(commit_tree_args)

    execute(['update-ref', "refs/heads/#{branch}", commit_id[0]])
  end

  private

  def execute(args, allow_failure: false)
    output, status = popen(base_args + args, nil) do |stdin|
      yield stdin if block_given?
    end

    unless status.zero?
      if allow_failure
        return []
      else
        raise "Got a non-zero exit code while calling out `#{args.join(' ')}`: #{output}"
      end
    end

    output.split("\n")
  end

  def base_args
    [
      Gitlab.config.git.bin_path,
      "--git-dir=#{@path_to_repo}"
    ]
  end
end