Class: Gitlab::Git::Repository

Inherits:
Object
  • Object
show all
Includes:
Popen
Defined in:
lib/gitlab_git/repository.rb

Defined Under Namespace

Classes: NoRepository

Instance Attribute Summary collapse

Instance Method Summary collapse

Methods included from Popen

#popen

Constructor Details

#initialize(path) ⇒ Repository

Returns a new instance of Repository.



28
29
30
31
32
# File 'lib/gitlab_git/repository.rb', line 28

def initialize(path)
  @path = path
  @name = path.split("/").last
  @root_ref = discover_default_branch
end

Instance Attribute Details

#gritObject (readonly)

Grit repo object



23
24
25
# File 'lib/gitlab_git/repository.rb', line 23

def grit
  @grit
end

#nameObject (readonly)

Directory name of repo



20
21
22
# File 'lib/gitlab_git/repository.rb', line 20

def name
  @name
end

#pathObject (readonly)

Full path to repo



17
18
19
# File 'lib/gitlab_git/repository.rb', line 17

def path
  @path
end

#root_refObject

Default branch in the repository



14
15
16
# File 'lib/gitlab_git/repository.rb', line 14

def root_ref
  @root_ref
end

#ruggedObject (readonly)

Rugged repo object



26
27
28
# File 'lib/gitlab_git/repository.rb', line 26

def rugged
  @rugged
end

Instance Method Details

#archive_repo(ref, storage_path, format = "tar.gz") ⇒ Object

Archive Project to .tar.gz

Already packed repo archives stored at app_root/tmp/repositories/project_name/project_name-commit-id.tag.gz



131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
# File 'lib/gitlab_git/repository.rb', line 131

def archive_repo(ref, storage_path, format = "tar.gz")
  ref = ref || self.root_ref
  commit = Gitlab::Git::Commit.find(self, ref)
  return nil unless commit

  extension = nil
  git_archive_format = nil
  pipe_cmd = nil

  case format
  when "tar.bz2", "tbz", "tbz2", "tb2", "bz2"
    extension = ".tar.bz2"
    pipe_cmd = %W(bzip2)
  when "tar"
    extension = ".tar"
    pipe_cmd = %W(cat)
  when "zip"
    extension = ".zip"
    git_archive_format = "zip"
    pipe_cmd = %W(cat)
  else
    # everything else should fall back to tar.gz
    extension = ".tar.gz"
    git_archive_format = nil
    pipe_cmd = %W(gzip -n)
  end

  # Build file path
  file_name = self.name.gsub("\.git", "") + "-" + commit.id.to_s + extension
  file_path = File.join(storage_path, self.name, file_name)

  # Put files into a directory before archiving
  prefix = File.basename(self.name) + "/"

  # Create file if not exists
  unless File.exists?(file_path)
    FileUtils.mkdir_p File.dirname(file_path)
    file = self.grit.archive_to_file(ref, prefix, file_path, git_archive_format, pipe_cmd)
  end

  file_path
end

#branch_namesObject

Returns an Array of branch names sorted by name ASC



53
54
55
# File 'lib/gitlab_git/repository.rb', line 53

def branch_names
  branches.map(&:name)
end

#branch_names_contains(commit) ⇒ Object

Returns branch names collection that contains the special commit(SHA1 or name)

Ex.

repo.branch_names_contains('master')


303
304
305
306
307
308
309
310
311
312
313
314
# File 'lib/gitlab_git/repository.rb', line 303

def branch_names_contains(commit)
  output = grit.git.native(:branch, {contains: true}, commit)

  # Fix encoding issue
  output = EncodingHelper::encode!(output)

  # The output is expected as follow
  #   fix-aaa
  #   fix-bbb
  # * master
  output.scan(/[^* \n]+/)
end

#branchesObject

Returns an Array of Branches



58
59
60
61
62
# File 'lib/gitlab_git/repository.rb', line 58

def branches
  rugged.branches.map do |rugged_ref|
    Branch.new(rugged_ref.name, rugged_ref.target)
  end.sort_by(&:name)
end

#commit_count(ref) ⇒ Object

Return total commits count accessible from passed ref



354
355
356
357
358
359
# File 'lib/gitlab_git/repository.rb', line 354

def commit_count(ref)
  walker = Rugged::Walker.new(rugged)
  walker.sorting(Rugged::SORT_TOPO | Rugged::SORT_REVERSE)
  walker.push(ref)
  walker.count
end

#commits_between(from, to) ⇒ Object

Delegate commits_between to Grit method



224
225
226
# File 'lib/gitlab_git/repository.rb', line 224

def commits_between(from, to)
  grit.commits_between(from, to)
end

#diff(from, to, *paths) ⇒ Object



232
233
234
# File 'lib/gitlab_git/repository.rb', line 232

def diff(from, to, *paths)
  grit.diff(from, to, *paths)
end

#discover_default_branchObject

Discovers the default branch based on the repository’s available branches

  • If no branches are present, returns nil

  • If one branch is present, returns its name

  • If two or more branches are present, returns current HEAD or master or first branch



106
107
108
109
110
111
112
113
114
115
116
117
118
# File 'lib/gitlab_git/repository.rb', line 106

def discover_default_branch
  if branch_names.length == 0
    nil
  elsif branch_names.length == 1
    branch_names.first
  elsif rugged_head && branch_names.include?(Ref.extract_branch_name(rugged_head.name))
    Ref.extract_branch_name(rugged_head.name)
  elsif branch_names.include?("master")
    "master"
  else
    branch_names.first
  end
end

#empty?Boolean

Returns:

  • (Boolean)


93
94
95
# File 'lib/gitlab_git/repository.rb', line 93

def empty?
  rugged.empty?
end

#find_commits(options = {}) ⇒ Object

Returns commits collection

Ex.

repo.find_commits(
  ref: 'master',
  max_count: 10,
  skip: 5,
  order: :date
)

+options+ is a Hash of optional arguments to git
  :ref is the ref from which to begin (SHA1 or name)
  :contains is the commit contained by the refs from which to begin (SHA1 or name)
  :max_count is the maximum number of commits to fetch
  :skip is the number of commits to skip
  :order is the commits order and allowed value is :date(default) or :topo


253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
# File 'lib/gitlab_git/repository.rb', line 253

def find_commits(options = {})
  actual_options = options.dup

  allowed_options = [:ref, :max_count, :skip, :contains, :order]

  actual_options.keep_if do |key, value|
    allowed_options.include?(key)
  end

  default_options = {pretty: 'raw', order: :date}

  actual_options = default_options.merge(actual_options)

  order = actual_options.delete(:order)

  case order
  when :date
    actual_options[:date_order] = true
  when :topo
    actual_options[:topo_order] = true
  end

  ref = actual_options.delete(:ref)

  containing_commit = actual_options.delete(:contains)

  args = []

  if ref
    args.push(ref)
  elsif containing_commit
    args.push(*branch_names_contains(containing_commit))
  else
    actual_options[:all] = true
  end

  output = grit.git.native(:rev_list, actual_options, *args)

  Grit::Commit.list_from_string(grit, output).map do |commit|
    Gitlab::Git::Commit.decorate(commit)
  end
rescue Grit::GitRuby::Repository::NoSuchShaFound
  []
end

#has_commits?Boolean

Returns:

  • (Boolean)


89
90
91
# File 'lib/gitlab_git/repository.rb', line 89

def has_commits?
  !empty?
end

#headsObject

Deprecated. Will be removed in 5.2



85
86
87
# File 'lib/gitlab_git/repository.rb', line 85

def heads
  @heads ||= grit.heads.sort_by(&:name)
end

#log(options) ⇒ Object

Delegate log to Grit method

Usage.

repo.log(
  ref: 'master',
  path: 'app/models',
  limit: 10,
  offset: 5,
)


202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
# File 'lib/gitlab_git/repository.rb', line 202

def log(options)
  default_options = {
    limit: 10,
    offset: 0,
    path: nil,
    ref: root_ref,
    follow: false
  }

  options = default_options.merge(options)

  grit.log(
    options[:ref] || root_ref,
    options[:path],
    max_count: options[:limit].to_i,
    skip: options[:offset].to_i,
    follow: options[:follow]
  )
end

#lookup(oid) ⇒ Object

Lookup for rugged object by oid



331
332
333
# File 'lib/gitlab_git/repository.rb', line 331

def lookup(oid)
  rugged.lookup(oid)
end

#merge_base_commit(from, to) ⇒ Object



228
229
230
# File 'lib/gitlab_git/repository.rb', line 228

def merge_base_commit(from, to)
  grit.git.native(:merge_base, {}, [to, from]).strip
end

#rawObject

Alias to old method for compatibility



41
42
43
# File 'lib/gitlab_git/repository.rb', line 41

def raw
  grit
end

#ref_namesObject

Returns an Array of branch and tag names



80
81
82
# File 'lib/gitlab_git/repository.rb', line 80

def ref_names
  branch_names + tag_names
end

#refs_hashObject

Get refs hash which key is SHA1 and value is ref object(Grit::Head or Grit::Remote or Grit::Tag)



318
319
320
321
322
323
324
325
326
327
328
# File 'lib/gitlab_git/repository.rb', line 318

def refs_hash
  # Initialize only when first call
  if @refs_hash.nil?
    @refs_hash = Hash.new { |h, k| h[k] = [] }

    grit.refs.each do |r|
      @refs_hash[r.commit.id] << r
    end
  end
  @refs_hash
end

#repo_exists?Boolean

Returns:

  • (Boolean)


97
98
99
# File 'lib/gitlab_git/repository.rb', line 97

def repo_exists?
  !!rugged
end

#rugged_headObject



120
121
122
123
124
# File 'lib/gitlab_git/repository.rb', line 120

def rugged_head
  rugged.head
rescue Rugged::ReferenceError
  nil
end

#search_files(query, ref = nil) ⇒ Object



180
181
182
183
184
185
186
187
188
189
190
# File 'lib/gitlab_git/repository.rb', line 180

def search_files(query, ref = nil)
  if ref.nil? || ref == ""
    ref = root_ref
  end

  greps = grit.grep(query, 3, ref)

  greps.map do |grep|
    Gitlab::Git::BlobSnippet.new(ref, grep.content, grep.startline, grep.filename)
  end
end

#sizeObject

Return repo size in megabytes



175
176
177
178
# File 'lib/gitlab_git/repository.rb', line 175

def size
  size = popen(%W(du -s), path).first.strip.to_i
  (size.to_f / 1024).round(2)
end

#submodules(ref) ⇒ Object

Return hash with submodules info for this repository

Ex.

{
  "rack"  => {
    "id" => "c67be4624545b4263184c4a0e8f887efd0a66320",
    "path" => "rack",
    "url" => "git://github.com/chneukirchen/rack.git"
  },
  "encoding" => {
    "id" => ....
  }
}


349
350
351
# File 'lib/gitlab_git/repository.rb', line 349

def submodules(ref)
  Grit::Submodule.config(grit, ref)
end

#tag_namesObject

Returns an Array of tag names



65
66
67
68
# File 'lib/gitlab_git/repository.rb', line 65

def tag_names
  # rugged.tags returns array of names
  rugged.tags.to_a
end

#tagsObject

Returns an Array of Tags



71
72
73
74
75
76
77
# File 'lib/gitlab_git/repository.rb', line 71

def tags
  rugged.refs.select do |ref|
    ref.name =~ /\Arefs\/tags/
  end.map do |rugged_ref|
    Tag.new(rugged_ref.name, rugged_ref.target)
  end.sort_by(&:name)
end