Class: Redmine::Scm::Adapters::MercurialAdapter
Defined Under Namespace
Classes: HgCommandAborted, HgCommandArgumentError, Revision
Constant Summary
collapse
- HG_BIN =
Mercurial executable name
Redmine::Configuration['scm_mercurial_command'] || "hg"
- HELPERS_DIR =
File.dirname(__FILE__) + "/mercurial"
- HG_HELPER_EXT =
"#{HELPERS_DIR}/redminehelper.py"
- TEMPLATE_NAME =
"hg-template"
- TEMPLATE_EXTENSION =
"tmpl"
Class Method Summary
collapse
Instance Method Summary
collapse
-
#annotate(path, identifier = nil) ⇒ Object
-
#branches ⇒ Object
-
#branchmap ⇒ Object
Returns map of => ‘nodeid’, ….
-
#cat(path, identifier = nil) ⇒ Object
-
#diff(path, identifier_from, identifier_to = nil) ⇒ Object
-
#each_revision(path = nil, identifier_from = nil, identifier_to = nil, options = {}) ⇒ Object
Iterates the revisions by using a template file that makes Mercurial produce a xml output.
-
#entries(path = nil, identifier = nil, options = {}) ⇒ Object
-
#info ⇒ Object
-
#nodes_in_branch(branch, options = {}) ⇒ Object
Returns list of nodes in the specified branch.
-
#path_encoding ⇒ Object
-
#revisions(path = nil, identifier_from = nil, identifier_to = nil, options = {}) ⇒ Object
-
#tagmap ⇒ Object
Returns map of => ‘nodeid’, ….
-
#tags ⇒ Object
-
#valid_name?(name) ⇒ Boolean
#adapter_name, client_version_above?, client_version_string, #default_branch, #entry, #initialize, logger, #properties, #root_url, shell_quote, shell_quote_command, shellout, #supports_annotate?, #supports_cat?, #url, #with_leading_slash, #with_trailing_slash, #without_leading_slash, #without_trailing_slash
shell_quote, shell_quote_command
Class Method Details
.client_available ⇒ Object
52
53
54
55
56
57
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 52
def client_available
client_version_above?([5, 1]) &&
(python_version <=> [3, 5]) >= 0
end
|
.client_command ⇒ Object
40
41
42
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 40
def client_command
@@bin ||= HG_BIN
end
|
.client_version ⇒ Object
48
49
50
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 48
def client_version
@@client_version ||= (hgversion || [])
end
|
.hgdebuginstall_from_command_line ⇒ Object
86
87
88
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 86
def hgdebuginstall_from_command_line
shellout("#{sq_bin} debuginstall") {|io| io.read}.to_s
end
|
.hgversion ⇒ Object
59
60
61
62
63
64
65
66
67
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 59
def hgversion
theversion = hgversion_from_command_line.b
if m = theversion.match(%r{\A(.*?)((\d+\.)+\d+)})
m[2].scan(%r{\d+}).collect(&:to_i)
end
end
|
.hgversion_from_command_line ⇒ Object
69
70
71
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 69
def hgversion_from_command_line
shellout("#{sq_bin} --version") {|io| io.read}.to_s
end
|
.python_version ⇒ Object
73
74
75
76
77
78
79
80
81
82
83
84
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 73
def python_version
@@python_version ||= begin
debuginstall = hgdebuginstall_from_command_line
if (m = debuginstall.match(/checking Python version \(([\d.]+)\)/))
m[1].scan(%r{\d+})
.collect(&:to_i)
.presence
else
nil
end
end
end
|
.sq_bin ⇒ Object
44
45
46
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 44
def sq_bin
@@sq_bin ||= shell_quote_command
end
|
.template_path ⇒ Object
90
91
92
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 90
def template_path
@@template_path ||= template_path_for(client_version)
end
|
.template_path_for(version) ⇒ Object
Instance Method Details
#annotate(path, identifier = nil) ⇒ Object
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 296
def annotate(path, identifier=nil)
p = CGI.escape(scm_iconv(@path_encoding, 'UTF-8', path))
blame = Annotate.new
hg 'rhannotate', '-ncu', "-r#{CGI.escape(hgrev(identifier))}", '--', hgtarget(p) do |io|
io.each_line do |line|
next unless line.b =~ %r{^([^:]+)\s(\d+)\s([0-9a-f]+):\s(.*)$}
r = Revision.new(:author => $1.strip, :revision => $2, :scmid => $3,
:identifier => $3)
blame.add_line($4.rstrip, r)
end
end
blame
rescue HgCommandAborted
Annotate.new
end
|
#branches ⇒ Object
127
128
129
130
131
132
133
134
135
136
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 127
def branches
brs = []
as_ary(summary['repository']['branch']).each do |e|
br = Branch.new(CGI.unescape(e['name']))
br.revision = e['revision']
br.scmid = e['node']
brs << br
end
brs
end
|
#branchmap ⇒ Object
Returns map of => ‘nodeid’, …
139
140
141
142
143
144
145
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 139
def branchmap
map = {}
branches.each do |b|
map[b.to_s] = b.scmid
end
map
end
|
#cat(path, identifier = nil) ⇒ Object
286
287
288
289
290
291
292
293
294
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 286
def cat(path, identifier=nil)
p = CGI.escape(scm_iconv(@path_encoding, 'UTF-8', path))
hg 'rhcat', "-r#{CGI.escape(hgrev(identifier))}", '--', hgtarget(p) do |io|
io.binmode
io.read
end
rescue HgCommandAborted
nil end
|
#diff(path, identifier_from, identifier_to = nil) ⇒ Object
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 264
def diff(path, identifier_from, identifier_to=nil)
hg_args = %w|rhdiff|
if identifier_to
hg_args << "-r#{hgrev(identifier_to)}" << "-r#{hgrev(identifier_from)}"
else
hg_args << "-c#{hgrev(identifier_from)}"
end
unless path.blank?
p = scm_iconv(@path_encoding, 'UTF-8', path)
hg_args << '--' << CGI.escape(hgtarget(p))
end
diff = []
hg(*hg_args) do |io|
io.each_line do |line|
diff << line
end
end
diff
rescue HgCommandAborted
nil end
|
#each_revision(path = nil, identifier_from = nil, identifier_to = nil, options = {}) ⇒ Object
Iterates the revisions by using a template file that makes Mercurial produce a xml output.
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 204
def each_revision(path=nil, identifier_from=nil, identifier_to=nil, options={})
hg_args = ['log', '--debug', '-C', "--style=#{self.class.template_path}"]
hg_args << "-r#{hgrev(identifier_from)}:#{hgrev(identifier_to)}"
hg_args << "--limit=#{options[:limit]}" if options[:limit]
hg_args << '--' << hgtarget(path) unless path.blank?
log = hg(*hg_args) do |io|
output = io.read.force_encoding('UTF-8')
begin
parse_xml("#{output}</log>")['log']
rescue
end
end
as_ary(log['logentry']).each do |le|
cpalist = as_ary(le['paths']['path-copied']).map do |e|
[e['__content__'], e['copyfrom-path']].map do |s|
scm_iconv('UTF-8', @path_encoding, CGI.unescape(s))
end
end
cpmap = Hash[*cpalist.flatten]
paths = as_ary(le['paths']['path']).map do |e|
p = scm_iconv('UTF-8', @path_encoding, CGI.unescape(e['__content__']))
{:action => e['action'],
:path => with_leading_slash(p),
:from_path => (cpmap.member?(p) ? with_leading_slash(cpmap[p]) : nil),
:from_revision => (cpmap.member?(p) ? le['node'] : nil)}
end
paths.sort_by!{|e| e[:path]}
parents_ary = []
as_ary(le['parents']['parent']).map do |par|
parents_ary << par['__content__'] if par['__content__'] != "0000000000000000000000000000000000000000"
end
yield Revision.new(:revision => le['revision'],
:scmid => le['node'],
:author =>
CGI.unescape(
begin
le['author']['__content__']
rescue
''
end
),
:time => Time.parse(le['date']['__content__']),
:message => CGI.unescape(le['msg']['__content__'] || ''),
:paths => paths,
:parents => parents_ary)
end
self
end
|
#entries(path = nil, identifier = nil, options = {}) ⇒ Object
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 161
def entries(path=nil, identifier=nil, options={})
p1 = scm_iconv(@path_encoding, 'UTF-8', path)
manifest = hg('rhmanifest', "-r#{CGI.escape(hgrev(identifier))}",
'--', CGI.escape(without_leading_slash(p1.to_s))) do |io|
output = io.read.force_encoding('UTF-8')
begin
parse_xml(output)['rhmanifest']['repository']['manifest']
rescue
end
end
path_prefix = path.blank? ? '' : with_trailing_slash(path)
entries = Entries.new
as_ary(manifest['dir']).each do |e|
n = CGI.unescape(e['name'])
p = "#{path_prefix}#{n}"
entries << Entry.new(:name => n, :path => p, :kind => 'dir')
end
as_ary(manifest['file']).each do |e|
n = CGI.unescape(e['name'])
p = "#{path_prefix}#{n}"
lr = Revision.new(:revision => e['revision'], :scmid => e['node'],
:identifier => e['node'],
:time => Time.at(e['time'].to_i))
entries << Entry.new(:name => n, :path => p, :kind => 'file',
:size => e['size'].to_i, :lastrev => lr)
end
entries
rescue HgCommandAborted
nil end
|
#info ⇒ Object
103
104
105
106
107
108
109
110
111
112
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 103
def info
tip = summary['repository']['tip']
Info.new(:root_url => CGI.unescape(summary['repository']['root']),
:lastrev => Revision.new(:revision => tip['revision'],
:scmid => tip['node']))
rescue => e
logger.error "hg: error during getting info: #{e.message}"
nil
end
|
#nodes_in_branch(branch, options = {}) ⇒ Object
Returns list of nodes in the specified branch
256
257
258
259
260
261
262
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 256
def nodes_in_branch(branch, options={})
hg_args = ['rhlog', '--template={node}\n', "--rhbranch=#{CGI.escape(branch)}"]
hg_args << "--from=#{CGI.escape(branch)}"
hg_args << '--to=0'
hg_args << "--limit=#{options[:limit]}" if options[:limit]
hg(*hg_args) {|io| io.readlines.map {|e| e.chomp}}
end
|
#path_encoding ⇒ Object
99
100
101
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 99
def path_encoding
@path_encoding
end
|
#revisions(path = nil, identifier_from = nil, identifier_to = nil, options = {}) ⇒ Object
196
197
198
199
200
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 196
def revisions(path=nil, identifier_from=nil, identifier_to=nil, options={})
revs = Revisions.new
each_revision(path, identifier_from, identifier_to, options) {|e| revs << e}
revs
end
|
#tagmap ⇒ Object
Returns map of => ‘nodeid’, …
119
120
121
122
123
124
125
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 119
def tagmap
map = {}
as_ary(summary['repository']['tag']).each do |e|
map[CGI.unescape(e['name'])] = e['node']
end
map
end
|
114
115
116
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 114
def tags
as_ary(summary['repository']['tag']).map {|e| CGI.unescape(e['name'])}
end
|
#valid_name?(name) ⇒ Boolean
314
315
316
317
318
319
320
321
|
# File 'lib/redmine/scm/adapters/mercurial_adapter.rb', line 314
def valid_name?(name)
return false unless name.nil? || name.is_a?(String)
true
end
|