2013-11-09 01:20:57 -05:00
|
|
|
# vcs
|
|
|
|
|
|
|
|
ENV.delete('PWD')
|
|
|
|
|
|
|
|
unless File.respond_to? :realpath
|
|
|
|
require 'pathname'
|
|
|
|
def File.realpath(arg)
|
|
|
|
Pathname(arg).realpath.to_s
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-12-14 20:02:41 -05:00
|
|
|
def IO.pread(*args)
|
|
|
|
STDERR.puts(*args.inspect) if $DEBUG
|
|
|
|
popen(*args) {|f|f.read}
|
|
|
|
end
|
|
|
|
|
|
|
|
if RUBY_VERSION < "1.9"
|
|
|
|
class IO
|
|
|
|
@orig_popen = method(:popen)
|
|
|
|
|
|
|
|
if defined?(fork)
|
|
|
|
def self.popen(command, *rest, &block)
|
|
|
|
if !(Array === command)
|
|
|
|
@orig_popen.call(command, *rest, &block)
|
|
|
|
elsif block
|
|
|
|
@orig_popen.call("-", *rest) {|f| f ? yield(f) : exec(*command)}
|
|
|
|
else
|
|
|
|
@orig_popen.call("-", *rest) or exec(*command)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
else
|
|
|
|
require 'shellwords'
|
|
|
|
def self.popen(command, *rest, &block)
|
|
|
|
command = command.shelljoin if Array === command
|
|
|
|
@orig_popen.call(command, *rest, &block)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-11-09 01:20:57 -05:00
|
|
|
class VCS
|
|
|
|
class NotFoundError < RuntimeError; end
|
|
|
|
|
|
|
|
@@dirs = []
|
|
|
|
def self.register(dir)
|
|
|
|
@@dirs << [dir, self]
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.detect(path)
|
|
|
|
@@dirs.each do |dir, klass|
|
|
|
|
return klass.new(path) if File.directory?(File.join(path, dir))
|
|
|
|
prev = path
|
|
|
|
loop {
|
|
|
|
curr = File.realpath(File.join(prev, '..'))
|
|
|
|
break if curr == prev # stop at the root directory
|
|
|
|
return klass.new(path) if File.directory?(File.join(curr, dir))
|
|
|
|
prev = curr
|
|
|
|
}
|
|
|
|
end
|
|
|
|
raise VCS::NotFoundError, "does not seem to be under a vcs: #{path}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def initialize(path)
|
|
|
|
@srcdir = path
|
|
|
|
super()
|
|
|
|
end
|
|
|
|
|
2014-02-27 20:23:31 -05:00
|
|
|
NullDevice = defined?(IO::NULL) ? IO::NULL :
|
|
|
|
%w[/dev/null NUL NIL: NL:].find {|dev| File.exist?(dev)}
|
|
|
|
|
2013-11-09 01:20:57 -05:00
|
|
|
# return a pair of strings, the last revision and the last revision in which
|
|
|
|
# +path+ was modified.
|
|
|
|
def get_revisions(path)
|
2014-12-14 20:02:46 -05:00
|
|
|
if String === path or path.respond_to?(:to_path)
|
|
|
|
path = relative_to(path)
|
|
|
|
end
|
2014-12-13 18:55:33 -05:00
|
|
|
last, changed, modified, *rest = (
|
2014-02-27 20:23:31 -05:00
|
|
|
begin
|
|
|
|
if NullDevice
|
|
|
|
save_stderr = STDERR.dup
|
|
|
|
STDERR.reopen NullDevice, 'w'
|
|
|
|
end
|
2014-12-13 18:55:33 -05:00
|
|
|
self.class.get_revisions(path, @srcdir)
|
2014-02-27 20:23:31 -05:00
|
|
|
ensure
|
|
|
|
if save_stderr
|
|
|
|
STDERR.reopen save_stderr
|
|
|
|
save_stderr.close
|
|
|
|
end
|
|
|
|
end
|
2014-12-13 18:55:33 -05:00
|
|
|
)
|
2013-11-09 11:37:46 -05:00
|
|
|
last or raise VCS::NotFoundError, "last revision not found"
|
|
|
|
changed or raise VCS::NotFoundError, "changed revision not found"
|
2014-08-22 02:36:46 -04:00
|
|
|
if modified
|
|
|
|
/\A(\d+)-(\d+)-(\d+)\D(\d+):(\d+):(\d+(?:\.\d+)?)\s*(?:Z|([-+]\d\d)(\d\d))\z/ =~ modified or
|
|
|
|
raise "unknown time format - #{modified}"
|
2014-11-25 01:26:26 -05:00
|
|
|
begin
|
|
|
|
modified = Time.new(*$~[1..6].map(&:to_i), ($7 ? "#{$7}:#{$8}" : "+00:00"))
|
|
|
|
rescue ArgumentError
|
|
|
|
modified = Time.utc(*$~[1..6]) + $7.to_i * 3600 + $8.to_i * 60
|
|
|
|
end
|
2014-08-22 02:36:46 -04:00
|
|
|
end
|
2013-11-09 08:35:39 -05:00
|
|
|
return last, changed, modified, *rest
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def relative_to(path)
|
|
|
|
if path
|
|
|
|
srcdir = File.realpath(@srcdir)
|
2014-12-14 20:02:46 -05:00
|
|
|
path = File.realdirpath(path)
|
2013-11-09 01:20:57 -05:00
|
|
|
list1 = srcdir.split(%r{/})
|
|
|
|
list2 = path.split(%r{/})
|
|
|
|
while !list1.empty? && !list2.empty? && list1.first == list2.first
|
|
|
|
list1.shift
|
|
|
|
list2.shift
|
|
|
|
end
|
|
|
|
if list1.empty? && list2.empty?
|
|
|
|
"."
|
|
|
|
else
|
|
|
|
([".."] * list1.length + list2).join("/")
|
|
|
|
end
|
|
|
|
else
|
|
|
|
'.'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class SVN < self
|
|
|
|
register(".svn")
|
|
|
|
|
2014-12-13 18:55:33 -05:00
|
|
|
def self.get_revisions(path, srcdir = nil)
|
2014-12-14 20:02:46 -05:00
|
|
|
if srcdir and (String === path or path.respond_to?(:to_path))
|
2014-12-13 18:55:33 -05:00
|
|
|
path = File.join(srcdir, path)
|
|
|
|
end
|
2014-12-14 20:02:41 -05:00
|
|
|
info_xml = IO.pread(%W"svn info --xml #{path}")
|
2013-11-09 01:20:57 -05:00
|
|
|
_, last, _, changed, _ = info_xml.split(/revision="(\d+)"/)
|
2013-11-09 08:35:39 -05:00
|
|
|
modified = info_xml[/<date>([^<>]*)/, 1]
|
|
|
|
[last, changed, modified]
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class GIT < self
|
|
|
|
register(".git")
|
|
|
|
|
2014-12-13 18:55:33 -05:00
|
|
|
def self.get_revisions(path, srcdir = nil)
|
2014-12-14 20:02:41 -05:00
|
|
|
logcmd = %W[git log -n1 --date=iso]
|
|
|
|
logcmd[1, 0] = ["-C", srcdir] if srcdir
|
|
|
|
logcmd << "--grep=^ *git-svn-id: .*@[0-9][0-9]*"
|
2013-11-09 01:20:57 -05:00
|
|
|
idpat = /git-svn-id: .*?@(\d+) \S+\Z/
|
2014-12-14 20:02:41 -05:00
|
|
|
last = IO.pread(logcmd)[idpat, 1]
|
2013-11-09 08:35:39 -05:00
|
|
|
if path
|
2014-12-14 20:02:41 -05:00
|
|
|
log = IO.pread(logcmd + [path])
|
2013-11-09 08:35:39 -05:00
|
|
|
changed = log[idpat, 1]
|
|
|
|
else
|
|
|
|
changed = last
|
|
|
|
end
|
2014-08-22 02:36:46 -04:00
|
|
|
modified = log[/^Date:\s+(.*)/, 1]
|
2013-11-09 08:35:39 -05:00
|
|
|
[last, changed, modified]
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|