2013-11-09 01:20:57 -05:00
|
|
|
# vcs
|
2015-01-17 21:22:50 -05:00
|
|
|
require 'fileutils'
|
2019-06-23 05:28:32 -04:00
|
|
|
require 'optparse'
|
2013-11-09 01:20:57 -05:00
|
|
|
|
2016-07-02 17:01:04 -04:00
|
|
|
# This library is used by several other tools/ scripts to detect the current
|
|
|
|
# VCS in use (e.g. SVN, Git) or to interact with that VCS.
|
|
|
|
|
2013-11-09 01:20:57 -05:00
|
|
|
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)
|
2018-02-04 22:19:39 -05:00
|
|
|
STDERR.puts(args.inspect) if $DEBUG
|
2014-12-14 20:02:41 -05:00
|
|
|
popen(*args) {|f|f.read}
|
|
|
|
end
|
|
|
|
|
2015-12-24 01:35:04 -05:00
|
|
|
if RUBY_VERSION < "2.0"
|
2014-12-14 20:02:41 -05:00
|
|
|
class IO
|
|
|
|
@orig_popen = method(:popen)
|
|
|
|
|
|
|
|
if defined?(fork)
|
|
|
|
def self.popen(command, *rest, &block)
|
2016-11-06 19:04:20 -05:00
|
|
|
if command.kind_of?(Hash)
|
|
|
|
env = command
|
|
|
|
command = rest.shift
|
|
|
|
end
|
2015-12-24 01:35:49 -05:00
|
|
|
opts = rest.last
|
|
|
|
if opts.kind_of?(Hash)
|
2015-02-23 09:19:45 -05:00
|
|
|
dir = opts.delete(:chdir)
|
2015-12-24 01:35:04 -05:00
|
|
|
rest.pop if opts.empty?
|
2019-06-02 23:26:23 -04:00
|
|
|
opts.delete(:external_encoding)
|
2015-02-23 09:19:45 -05:00
|
|
|
end
|
2015-12-24 01:35:49 -05:00
|
|
|
|
2015-02-23 09:19:45 -05:00
|
|
|
if block
|
|
|
|
@orig_popen.call("-", *rest) do |f|
|
|
|
|
if f
|
|
|
|
yield(f)
|
|
|
|
else
|
|
|
|
Dir.chdir(dir) if dir
|
2016-11-06 19:04:20 -05:00
|
|
|
ENV.replace(env) if env
|
2015-02-23 09:19:45 -05:00
|
|
|
exec(*command)
|
|
|
|
end
|
|
|
|
end
|
2014-12-14 20:02:41 -05:00
|
|
|
else
|
2015-02-23 09:19:45 -05:00
|
|
|
f = @orig_popen.call("-", *rest)
|
|
|
|
unless f
|
|
|
|
Dir.chdir(dir) if dir
|
2016-11-06 19:04:20 -05:00
|
|
|
ENV.replace(env) if env
|
2015-02-23 09:19:45 -05:00
|
|
|
exec(*command)
|
|
|
|
end
|
|
|
|
f
|
2014-12-14 20:02:41 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
else
|
|
|
|
require 'shellwords'
|
|
|
|
def self.popen(command, *rest, &block)
|
2016-11-06 19:04:20 -05:00
|
|
|
if command.kind_of?(Hash)
|
|
|
|
env = command
|
|
|
|
oldenv = ENV.to_hash
|
|
|
|
command = rest.shift
|
|
|
|
end
|
2015-12-24 01:35:49 -05:00
|
|
|
opts = rest.last
|
|
|
|
if opts.kind_of?(Hash)
|
2015-02-23 09:19:45 -05:00
|
|
|
dir = opts.delete(:chdir)
|
2015-12-24 01:35:04 -05:00
|
|
|
rest.pop if opts.empty?
|
2019-06-02 23:26:23 -04:00
|
|
|
opts.delete(:external_encoding)
|
2015-02-23 09:19:45 -05:00
|
|
|
end
|
2015-12-24 01:35:49 -05:00
|
|
|
|
2014-12-14 20:02:41 -05:00
|
|
|
command = command.shelljoin if Array === command
|
2015-02-23 09:19:45 -05:00
|
|
|
Dir.chdir(dir || ".") do
|
2016-11-06 19:04:20 -05:00
|
|
|
ENV.replace(env) if env
|
2015-02-23 09:19:45 -05:00
|
|
|
@orig_popen.call(command, *rest, &block)
|
2016-11-06 19:04:20 -05:00
|
|
|
ENV.replace(oldenv) if oldenv
|
2015-02-23 09:19:45 -05:00
|
|
|
end
|
2014-12-14 20:02:41 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-11-06 21:54:27 -05:00
|
|
|
else
|
|
|
|
module DebugPOpen
|
2016-11-06 22:47:46 -05:00
|
|
|
verbose, $VERBOSE = $VERBOSE, nil if RUBY_VERSION < "2.1"
|
2016-11-06 21:54:27 -05:00
|
|
|
refine IO.singleton_class do
|
|
|
|
def popen(*args)
|
|
|
|
STDERR.puts args.inspect if $DEBUG
|
|
|
|
super
|
|
|
|
end
|
|
|
|
end
|
2016-11-06 22:47:46 -05:00
|
|
|
ensure
|
|
|
|
$VERBOSE = verbose unless verbose.nil?
|
2016-11-06 21:54:27 -05:00
|
|
|
end
|
|
|
|
using DebugPOpen
|
2018-01-26 08:38:00 -05:00
|
|
|
module DebugSystem
|
2018-01-30 02:34:06 -05:00
|
|
|
def system(*args)
|
|
|
|
STDERR.puts args.inspect if $DEBUG
|
|
|
|
exception = false
|
|
|
|
opts = Hash.try_convert(args[-1])
|
2018-01-27 01:11:23 -05:00
|
|
|
if RUBY_VERSION >= "2.6"
|
2018-01-30 02:34:06 -05:00
|
|
|
unless opts
|
|
|
|
opts = {}
|
|
|
|
args << opts
|
|
|
|
end
|
2018-01-29 21:20:02 -05:00
|
|
|
exception = opts.fetch(:exception) {opts[:exception] = true}
|
2018-01-30 02:34:06 -05:00
|
|
|
elsif opts
|
2018-01-29 21:20:02 -05:00
|
|
|
exception = opts.delete(:exception) {true}
|
2018-02-04 22:19:39 -05:00
|
|
|
args.pop if opts.empty?
|
2018-01-27 01:11:23 -05:00
|
|
|
end
|
2018-01-30 02:34:06 -05:00
|
|
|
ret = super(*args)
|
2018-01-26 08:38:00 -05:00
|
|
|
raise "Command failed with status (#$?): #{args[0]}" if exception and !ret
|
|
|
|
ret
|
|
|
|
end
|
|
|
|
end
|
|
|
|
module Kernel
|
|
|
|
prepend(DebugSystem)
|
|
|
|
end
|
2014-12-14 20:02:41 -05:00
|
|
|
end
|
|
|
|
|
2013-11-09 01:20:57 -05:00
|
|
|
class VCS
|
2018-01-27 01:11:23 -05:00
|
|
|
prepend(DebugSystem) if defined?(DebugSystem)
|
2013-11-09 01:20:57 -05:00
|
|
|
class NotFoundError < RuntimeError; end
|
|
|
|
|
|
|
|
@@dirs = []
|
2016-01-20 03:17:57 -05:00
|
|
|
def self.register(dir, &pred)
|
|
|
|
@@dirs << [dir, self, pred]
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
|
|
|
|
2019-08-12 23:24:55 -04:00
|
|
|
def self.detect(path = '.', options = {}, argv = ::ARGV)
|
2019-05-28 19:22:02 -04:00
|
|
|
uplevel_limit = options.fetch(:uplevel_limit, 0)
|
2018-12-09 20:58:27 -05:00
|
|
|
curr = path
|
|
|
|
begin
|
|
|
|
@@dirs.each do |dir, klass, pred|
|
2019-06-23 05:28:32 -04:00
|
|
|
if pred ? pred[curr, dir] : File.directory?(File.join(curr, dir))
|
|
|
|
vcs = klass.new(curr)
|
|
|
|
vcs.parse_options(argv)
|
|
|
|
return vcs
|
|
|
|
end
|
2018-12-09 20:58:27 -05:00
|
|
|
end
|
2018-12-09 20:58:28 -05:00
|
|
|
if uplevel_limit
|
|
|
|
break if uplevel_limit.zero?
|
|
|
|
uplevel_limit -= 1
|
|
|
|
end
|
2018-12-09 20:58:27 -05:00
|
|
|
prev, curr = curr, File.realpath(File.join(curr, '..'))
|
|
|
|
end until curr == prev # stop at the root directory
|
2013-11-09 01:20:57 -05:00
|
|
|
raise VCS::NotFoundError, "does not seem to be under a vcs: #{path}"
|
|
|
|
end
|
|
|
|
|
2015-01-20 09:43:30 -05:00
|
|
|
def self.local_path?(path)
|
|
|
|
String === path or path.respond_to?(:to_path)
|
|
|
|
end
|
|
|
|
|
2016-01-08 21:15:45 -05:00
|
|
|
attr_reader :srcdir
|
|
|
|
|
2013-11-09 01:20:57 -05:00
|
|
|
def initialize(path)
|
|
|
|
@srcdir = path
|
|
|
|
super()
|
|
|
|
end
|
|
|
|
|
2019-06-23 05:28:32 -04:00
|
|
|
def parse_options(opts, parser = OptionParser.new)
|
|
|
|
case opts
|
|
|
|
when Array
|
|
|
|
parser.on("--[no-]dryrun") {|v| @dryrun = v}
|
|
|
|
parser.on("--[no-]debug") {|v| @debug = v}
|
|
|
|
parser.parse(opts)
|
2019-06-27 23:41:03 -04:00
|
|
|
@debug = $DEBUG unless defined?(@debug)
|
2019-06-23 05:28:32 -04:00
|
|
|
@dryrun = @debug unless defined?(@dryrun)
|
|
|
|
when Hash
|
|
|
|
unless (keys = opts.keys - [:debug, :dryrun]).empty?
|
|
|
|
raise "Unknown options: #{keys.join(', ')}"
|
|
|
|
end
|
|
|
|
@debug = opts.fetch(:debug) {$DEBUG}
|
|
|
|
@dryrun = opts.fetch(:dryrun) {@debug}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
attr_reader :dryrun, :debug
|
|
|
|
alias dryrun? dryrun
|
|
|
|
alias debug? debug
|
|
|
|
|
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)
|
2015-01-20 09:43:30 -05:00
|
|
|
if self.class.local_path?(path)
|
2014-12-14 20:02:46 -05:00
|
|
|
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
|
2019-06-23 05:28:32 -04:00
|
|
|
_get_revisions(path, @srcdir)
|
2017-12-13 07:06:11 -05:00
|
|
|
rescue Errno::ENOENT => e
|
|
|
|
raise VCS::NotFoundError, e.message
|
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-12-15 15:44:54 -05:00
|
|
|
match = $~[1..6].map { |x| x.to_i }
|
2014-12-15 19:08:14 -05:00
|
|
|
off = $7 ? "#{$7}:#{$8}" : "+00:00"
|
|
|
|
match << off
|
2014-11-25 01:26:26 -05:00
|
|
|
begin
|
2014-12-15 19:08:14 -05:00
|
|
|
modified = Time.new(*match)
|
2014-11-25 01:26:26 -05: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
|
|
|
|
|
2015-05-23 05:36:33 -04:00
|
|
|
def modified(path)
|
2016-08-07 06:34:10 -04:00
|
|
|
_, _, modified, * = get_revisions(path)
|
2015-05-23 05:36:33 -04:00
|
|
|
modified
|
|
|
|
end
|
|
|
|
|
2013-11-09 01:20:57 -05:00
|
|
|
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
|
2015-01-22 21:11:02 -05:00
|
|
|
end
|
2015-01-20 18:57:38 -05:00
|
|
|
|
2015-01-22 21:11:02 -05:00
|
|
|
def after_export(dir)
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
|
|
|
|
2019-04-27 13:00:39 -04:00
|
|
|
def revision_name(rev)
|
|
|
|
self.class.revision_name(rev)
|
|
|
|
end
|
|
|
|
|
2019-05-22 10:18:37 -04:00
|
|
|
def short_revision(rev)
|
|
|
|
self.class.short_revision(rev)
|
|
|
|
end
|
|
|
|
|
2013-11-09 01:20:57 -05:00
|
|
|
class SVN < self
|
|
|
|
register(".svn")
|
2017-04-17 22:58:45 -04:00
|
|
|
COMMAND = ENV['SVN'] || 'svn'
|
2013-11-09 01:20:57 -05:00
|
|
|
|
2019-04-27 13:00:39 -04:00
|
|
|
def self.revision_name(rev)
|
|
|
|
"r#{rev}"
|
|
|
|
end
|
|
|
|
|
2019-05-22 10:18:37 -04:00
|
|
|
def self.short_revision(rev)
|
|
|
|
rev
|
|
|
|
end
|
|
|
|
|
2019-06-23 05:28:32 -04:00
|
|
|
def _get_revisions(path, srcdir = nil)
|
2019-08-25 05:27:57 -04:00
|
|
|
if srcdir and self.class.local_path?(path)
|
2014-12-13 18:55:33 -05:00
|
|
|
path = File.join(srcdir, path)
|
|
|
|
end
|
2015-01-17 21:22:50 -05:00
|
|
|
if srcdir
|
2017-12-13 03:38:43 -05:00
|
|
|
info_xml = IO.pread(%W"#{COMMAND} info --xml #{srcdir}")
|
2015-01-17 21:22:50 -05:00
|
|
|
info_xml = nil unless info_xml[/<url>(.*)<\/url>/, 1] == path.to_s
|
|
|
|
end
|
2017-12-13 03:38:43 -05:00
|
|
|
info_xml ||= IO.pread(%W"#{COMMAND} 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]
|
2015-01-17 08:56:28 -05:00
|
|
|
branch = info_xml[%r'<relative-url>\^/(?:branches/|tags/)?([^<>]+)', 1]
|
2015-01-17 07:56:10 -05:00
|
|
|
[last, changed, modified, branch]
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
2014-12-14 20:02:52 -05:00
|
|
|
|
2015-01-20 09:43:30 -05:00
|
|
|
def self.search_root(path)
|
|
|
|
return unless local_path?(path)
|
|
|
|
parent = File.realpath(path)
|
|
|
|
begin
|
|
|
|
parent = File.dirname(wkdir = parent)
|
|
|
|
return wkdir if File.directory?(wkdir + "/.svn")
|
|
|
|
end until parent == wkdir
|
|
|
|
end
|
|
|
|
|
2015-01-17 21:22:50 -05:00
|
|
|
def get_info
|
2017-12-13 03:38:43 -05:00
|
|
|
@info ||= IO.pread(%W"#{COMMAND} info --xml #{@srcdir}")
|
2015-01-17 21:22:50 -05:00
|
|
|
end
|
|
|
|
|
2014-12-14 20:02:52 -05:00
|
|
|
def url
|
2015-01-17 21:22:50 -05:00
|
|
|
unless @url
|
|
|
|
url = get_info[/<root>(.*)<\/root>/, 1]
|
2014-12-14 20:02:52 -05:00
|
|
|
@url = URI.parse(url+"/") if url
|
|
|
|
end
|
|
|
|
@url
|
|
|
|
end
|
|
|
|
|
2015-01-17 21:22:50 -05:00
|
|
|
def wcroot
|
2015-01-18 00:25:04 -05:00
|
|
|
unless @wcroot
|
|
|
|
info = get_info
|
|
|
|
@wcroot = info[/<wcroot-abspath>(.*)<\/wcroot-abspath>/, 1]
|
2015-01-20 09:43:30 -05:00
|
|
|
@wcroot ||= self.class.search_root(@srcdir)
|
2015-01-18 00:25:04 -05:00
|
|
|
end
|
|
|
|
@wcroot
|
2015-01-17 21:22:50 -05:00
|
|
|
end
|
|
|
|
|
2014-12-14 20:02:52 -05:00
|
|
|
def branch(name)
|
2019-05-22 10:18:09 -04:00
|
|
|
return trunk if name == "trunk"
|
2014-12-14 20:02:52 -05:00
|
|
|
url + "branches/#{name}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def tag(name)
|
|
|
|
url + "tags/#{name}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def trunk
|
|
|
|
url + "trunk"
|
|
|
|
end
|
2019-07-01 10:16:47 -04:00
|
|
|
alias master trunk
|
2014-12-14 20:02:52 -05:00
|
|
|
|
|
|
|
def branch_list(pat)
|
2017-12-13 03:38:43 -05:00
|
|
|
IO.popen(%W"#{COMMAND} ls #{branch('')}") do |f|
|
2014-12-14 20:02:52 -05:00
|
|
|
f.each do |line|
|
2014-12-23 12:12:25 -05:00
|
|
|
line.chomp!
|
2014-12-14 20:02:52 -05:00
|
|
|
line.chomp!('/')
|
|
|
|
yield(line) if File.fnmatch?(pat, line)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def grep(pat, tag, *files, &block)
|
2017-04-17 22:58:45 -04:00
|
|
|
cmd = %W"#{COMMAND} cat"
|
2014-12-14 20:02:52 -05:00
|
|
|
files.map! {|n| File.join(tag, n)} if tag
|
|
|
|
set = block.binding.eval("proc {|match| $~ = match}")
|
2017-12-13 03:38:43 -05:00
|
|
|
IO.popen([cmd, *files]) do |f|
|
2014-12-14 20:02:52 -05:00
|
|
|
f.grep(pat) do |s|
|
|
|
|
set[$~]
|
|
|
|
yield s
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-01-20 18:57:38 -05:00
|
|
|
def export(revision, url, dir, keep_temp = false)
|
2015-01-18 00:24:14 -05:00
|
|
|
if @srcdir and (rootdir = wcroot)
|
2015-01-17 21:22:50 -05:00
|
|
|
srcdir = File.realpath(@srcdir)
|
2015-01-18 00:24:14 -05:00
|
|
|
rootdir << "/"
|
2015-01-17 21:22:50 -05:00
|
|
|
if srcdir.start_with?(rootdir)
|
|
|
|
subdir = srcdir[rootdir.size..-1]
|
|
|
|
subdir = nil if subdir.empty?
|
|
|
|
FileUtils.mkdir_p(svndir = dir+"/.svn")
|
|
|
|
FileUtils.ln_s(Dir.glob(rootdir+"/.svn/*"), svndir)
|
2017-04-17 22:58:45 -04:00
|
|
|
system(COMMAND, "-q", "revert", "-R", subdir || ".", :chdir => dir) or return false
|
2015-01-20 18:57:38 -05:00
|
|
|
FileUtils.rm_rf(svndir) unless keep_temp
|
2015-01-17 21:22:50 -05:00
|
|
|
if subdir
|
|
|
|
tmpdir = Dir.mktmpdir("tmp-co.", "#{dir}/#{subdir}")
|
|
|
|
File.rename(tmpdir, tmpdir = "#{dir}/#{File.basename(tmpdir)}")
|
|
|
|
FileUtils.mv(Dir.glob("#{dir}/#{subdir}/{.[^.]*,..?*,*}"), tmpdir)
|
|
|
|
begin
|
|
|
|
Dir.rmdir("#{dir}/#{subdir}")
|
|
|
|
end until (subdir = File.dirname(subdir)) == '.'
|
|
|
|
FileUtils.mv(Dir.glob("#{tmpdir}/#{subdir}/{.[^.]*,..?*,*}"), dir)
|
|
|
|
Dir.rmdir(tmpdir)
|
|
|
|
end
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
2017-12-13 03:38:43 -05:00
|
|
|
IO.popen(%W"#{COMMAND} export -r #{revision} #{url} #{dir}") do |pipe|
|
2014-12-14 20:02:52 -05:00
|
|
|
pipe.each {|line| /^A/ =~ line or yield line}
|
|
|
|
end
|
|
|
|
$?.success?
|
|
|
|
end
|
2015-01-20 18:57:38 -05:00
|
|
|
|
|
|
|
def after_export(dir)
|
|
|
|
FileUtils.rm_rf(dir+"/.svn")
|
|
|
|
end
|
2016-11-06 07:57:45 -05:00
|
|
|
|
2019-04-27 23:11:59 -04:00
|
|
|
def branch_beginning(url)
|
|
|
|
# `--limit` of svn-log is useless in this case, because it is
|
|
|
|
# applied before `--search`.
|
|
|
|
rev = IO.pread(%W[ #{COMMAND} log --xml
|
|
|
|
--search=matz --search-and=has\ started
|
|
|
|
-- #{url}/version.h])[/<logentry\s+revision="(\d+)"/m, 1]
|
|
|
|
rev.to_i if rev
|
|
|
|
end
|
|
|
|
|
2017-03-24 10:05:50 -04:00
|
|
|
def export_changelog(url, from, to, path)
|
2019-04-27 23:16:40 -04:00
|
|
|
range = [to || 'HEAD', (from ? from+1 : branch_beginning(url))].compact.join(':')
|
2017-12-13 03:38:43 -05:00
|
|
|
IO.popen({'TZ' => 'JST-9', 'LANG' => 'C', 'LC_ALL' => 'C'},
|
2017-04-17 22:58:45 -04:00
|
|
|
%W"#{COMMAND} log -r#{range} #{url}") do |r|
|
2016-11-06 07:57:45 -05:00
|
|
|
open(path, 'w') do |w|
|
|
|
|
IO.copy_stream(r, w)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2017-07-26 08:44:12 -04:00
|
|
|
|
|
|
|
def commit
|
2019-06-23 05:28:32 -04:00
|
|
|
args = %W"#{COMMAND} commit"
|
|
|
|
if dryrun?
|
|
|
|
STDERR.puts(args.inspect)
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
system(*args)
|
2017-07-26 08:44:12 -04:00
|
|
|
end
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
class GIT < self
|
2016-01-20 03:17:57 -05:00
|
|
|
register(".git") {|path, dir| File.exist?(File.join(path, dir))}
|
2017-04-17 22:58:45 -04:00
|
|
|
COMMAND = ENV["GIT"] || 'git'
|
2013-11-09 01:20:57 -05:00
|
|
|
|
2019-06-23 05:28:32 -04:00
|
|
|
def cmd_args(cmds, srcdir = nil)
|
2019-06-02 23:26:23 -04:00
|
|
|
(opts = cmds.last).kind_of?(Hash) or cmds << (opts = {})
|
|
|
|
opts[:external_encoding] ||= "UTF-8"
|
2019-06-23 05:28:32 -04:00
|
|
|
if srcdir and self.class.local_path?(srcdir)
|
2017-12-13 03:38:43 -05:00
|
|
|
opts[:chdir] ||= srcdir
|
|
|
|
end
|
2019-06-23 05:28:32 -04:00
|
|
|
STDERR.puts cmds.inspect if debug?
|
2017-12-13 03:38:43 -05:00
|
|
|
cmds
|
|
|
|
end
|
|
|
|
|
2019-06-23 05:28:32 -04:00
|
|
|
def cmd_pipe_at(srcdir, cmds, &block)
|
|
|
|
without_gitconfig { IO.popen(*cmd_args(cmds, srcdir), &block) }
|
2017-12-13 03:38:43 -05:00
|
|
|
end
|
|
|
|
|
2019-06-23 05:28:32 -04:00
|
|
|
def cmd_read_at(srcdir, cmds)
|
2019-06-05 07:07:19 -04:00
|
|
|
without_gitconfig { IO.pread(*cmd_args(cmds, srcdir)) }
|
2017-12-13 03:38:43 -05:00
|
|
|
end
|
|
|
|
|
2019-06-23 05:28:32 -04:00
|
|
|
def cmd_pipe(*cmds, &block)
|
|
|
|
cmd_pipe_at(@srcdir, cmds, &block)
|
|
|
|
end
|
|
|
|
|
|
|
|
def cmd_read(*cmds)
|
|
|
|
cmd_read_at(@srcdir, cmds)
|
|
|
|
end
|
|
|
|
|
|
|
|
def _get_revisions(path, srcdir = nil)
|
2019-06-04 22:53:24 -04:00
|
|
|
gitcmd = [COMMAND]
|
2019-05-22 10:18:37 -04:00
|
|
|
last = cmd_read_at(srcdir, [[*gitcmd, 'rev-parse', 'HEAD']]).rstrip
|
2019-06-04 08:27:46 -04:00
|
|
|
log = cmd_read_at(srcdir, [[*gitcmd, 'log', '-n1', '--date=iso', '--pretty=fuller', *path]])
|
2019-04-22 08:23:37 -04:00
|
|
|
changed = log[/\Acommit (\h+)/, 1]
|
2019-06-04 08:27:46 -04:00
|
|
|
modified = log[/^CommitDate:\s+(.*)/, 1]
|
2019-04-27 13:00:39 -04:00
|
|
|
branch = cmd_read_at(srcdir, [gitcmd + %W[symbolic-ref --short HEAD]])
|
2019-05-09 01:02:01 -04:00
|
|
|
if branch.empty?
|
2019-05-29 02:04:31 -04:00
|
|
|
branch_list = cmd_read_at(srcdir, [gitcmd + %W[branch --list --contains HEAD]]).lines.to_a
|
2019-05-30 13:30:29 -04:00
|
|
|
branch, = branch_list.grep(/\A\*/)
|
|
|
|
case branch
|
|
|
|
when /\A\* *\(\S+ detached at (.*)\)\Z/
|
|
|
|
branch = $1
|
|
|
|
branch = nil if last.start_with?(branch)
|
|
|
|
when /\A\* (\S+)\Z/
|
|
|
|
branch = $1
|
|
|
|
else
|
|
|
|
branch = nil
|
|
|
|
end
|
|
|
|
unless branch
|
|
|
|
branch_list.each {|b| b.strip!}
|
|
|
|
branch_list.delete_if {|b| / / =~ b}
|
|
|
|
branch = branch_list.min_by(&:length) || ""
|
|
|
|
end
|
2019-05-09 01:02:01 -04:00
|
|
|
end
|
2019-04-27 21:41:11 -04:00
|
|
|
branch.chomp!
|
2019-05-09 01:11:43 -04:00
|
|
|
branch = ":detached:" if branch.empty?
|
2019-05-02 01:44:47 -04:00
|
|
|
upstream = cmd_read_at(srcdir, [gitcmd + %W[branch --list --format=%(upstream:short) #{branch}]])
|
2019-05-01 07:47:00 -04:00
|
|
|
upstream.chomp!
|
|
|
|
title = cmd_read_at(srcdir, [gitcmd + %W[log --format=%s -n1 #{upstream}..HEAD]])
|
2015-02-17 22:43:14 -05:00
|
|
|
title = nil if title.empty?
|
|
|
|
[last, changed, modified, branch, title]
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
2014-12-14 20:02:52 -05:00
|
|
|
|
2019-04-27 13:00:39 -04:00
|
|
|
def self.revision_name(rev)
|
2019-05-22 10:18:37 -04:00
|
|
|
short_revision(rev)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.short_revision(rev)
|
|
|
|
rev[0, 10]
|
2019-04-27 13:00:39 -04:00
|
|
|
end
|
|
|
|
|
2019-06-23 05:28:32 -04:00
|
|
|
def without_gitconfig
|
2019-06-05 07:07:19 -04:00
|
|
|
home = ENV.delete('HOME')
|
|
|
|
yield
|
|
|
|
ensure
|
|
|
|
ENV['HOME'] = home if home
|
|
|
|
end
|
|
|
|
|
2016-11-08 02:45:19 -05:00
|
|
|
def initialize(*)
|
|
|
|
super
|
|
|
|
if srcdir = @srcdir and self.class.local_path?(srcdir)
|
|
|
|
@srcdir = File.realpath(srcdir)
|
|
|
|
end
|
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2015-01-16 19:17:36 -05:00
|
|
|
Branch = Struct.new(:to_str)
|
|
|
|
|
2014-12-14 20:02:52 -05:00
|
|
|
def branch(name)
|
2015-01-16 19:17:36 -05:00
|
|
|
Branch.new(name)
|
2014-12-14 20:02:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
alias tag branch
|
|
|
|
|
2019-07-01 10:16:47 -04:00
|
|
|
def master
|
|
|
|
branch("master")
|
2014-12-14 20:02:52 -05:00
|
|
|
end
|
2019-07-01 10:16:47 -04:00
|
|
|
alias trunk master
|
2014-12-14 20:02:52 -05:00
|
|
|
|
|
|
|
def stable
|
2019-06-04 22:53:24 -04:00
|
|
|
cmd = %W"#{COMMAND} for-each-ref --format=\%(refname:short) refs/heads/ruby_[0-9]*"
|
2016-11-08 01:39:46 -05:00
|
|
|
branch(cmd_read(cmd)[/.*^(ruby_\d+_\d+)$/m, 1])
|
2014-12-14 20:02:52 -05:00
|
|
|
end
|
|
|
|
|
2014-12-23 12:12:25 -05:00
|
|
|
def branch_list(pat)
|
2019-06-04 22:53:24 -04:00
|
|
|
cmd = %W"#{COMMAND} for-each-ref --format=\%(refname:short) refs/heads/#{pat}"
|
2016-11-08 01:39:46 -05:00
|
|
|
cmd_pipe(cmd) {|f|
|
2014-12-23 12:12:25 -05:00
|
|
|
f.each {|line|
|
|
|
|
line.chomp!
|
|
|
|
yield line
|
|
|
|
}
|
|
|
|
}
|
2014-12-14 20:02:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def grep(pat, tag, *files, &block)
|
2019-06-04 22:53:24 -04:00
|
|
|
cmd = %W[#{COMMAND} grep -h --perl-regexp #{tag} --]
|
2014-12-14 20:02:52 -05:00
|
|
|
set = block.binding.eval("proc {|match| $~ = match}")
|
2016-11-08 01:39:46 -05:00
|
|
|
cmd_pipe(cmd+files) do |f|
|
2014-12-14 20:02:52 -05:00
|
|
|
f.grep(pat) do |s|
|
|
|
|
set[$~]
|
|
|
|
yield s
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-01-20 18:57:38 -05:00
|
|
|
def export(revision, url, dir, keep_temp = false)
|
2019-08-25 00:58:30 -04:00
|
|
|
system(COMMAND, "clone", "-s", (@srcdir || '.').to_s, "-b", url, dir)
|
2014-12-14 20:02:52 -05:00
|
|
|
end
|
2015-01-20 18:57:38 -05:00
|
|
|
|
|
|
|
def after_export(dir)
|
2017-04-09 21:22:50 -04:00
|
|
|
FileUtils.rm_rf(Dir.glob("#{dir}/.git*"))
|
2015-01-20 18:57:38 -05:00
|
|
|
end
|
2016-11-06 07:57:45 -05:00
|
|
|
|
2019-04-27 23:11:59 -04:00
|
|
|
def branch_beginning(url)
|
2019-06-04 22:53:24 -04:00
|
|
|
cmd_read(%W[ #{COMMAND} log -n1 --format=format:%H
|
2019-04-27 23:04:15 -04:00
|
|
|
--author=matz --committer=matz --grep=has\ started
|
2019-04-27 11:30:16 -04:00
|
|
|
-- version.h include/ruby/version.h])
|
|
|
|
end
|
|
|
|
|
2017-03-24 10:05:50 -04:00
|
|
|
def export_changelog(url, from, to, path)
|
2019-04-27 23:16:40 -04:00
|
|
|
from, to = [from, to].map do |rev|
|
2016-11-06 22:25:28 -05:00
|
|
|
rev or next
|
2019-04-27 10:14:59 -04:00
|
|
|
if Integer === rev
|
|
|
|
rev = cmd_read({'LANG' => 'C', 'LC_ALL' => 'C'},
|
2019-06-04 22:53:24 -04:00
|
|
|
%W"#{COMMAND} log -n1 --format=format:%H" <<
|
2019-04-27 10:14:59 -04:00
|
|
|
"--grep=^ *git-svn-id: .*@#{rev} ")
|
|
|
|
end
|
2016-11-06 22:25:28 -05:00
|
|
|
rev unless rev.empty?
|
2019-04-27 23:16:40 -04:00
|
|
|
end
|
2019-08-24 22:38:15 -04:00
|
|
|
unless /./.match(from) or /./.match(from = branch_beginning(url))
|
2019-08-24 22:24:53 -04:00
|
|
|
warn "no starting commit found", uplevel: 1
|
2019-08-24 22:38:15 -04:00
|
|
|
from = nil
|
2019-04-27 23:16:40 -04:00
|
|
|
end
|
2019-08-25 03:59:45 -04:00
|
|
|
_rev = cmd_read({'LANG' => 'C', 'LC_ALL' => 'C'},
|
|
|
|
%W"#{COMMAND} show-ref notes/commits")
|
|
|
|
unless $?.success?
|
|
|
|
raise "need notes/commits tree; run `git fetch origin refs/notes/commits:refs/notes/commits` in the repository"
|
|
|
|
end
|
2019-08-24 22:38:15 -04:00
|
|
|
range = [from, (to || 'HEAD')].compact.join('^..')
|
2016-11-08 01:39:46 -05:00
|
|
|
cmd_pipe({'TZ' => 'JST-9', 'LANG' => 'C', 'LC_ALL' => 'C'},
|
2019-08-12 06:01:57 -04:00
|
|
|
%W"#{COMMAND} log --format=medium --notes=commits --date=iso-local --topo-order #{range}", "rb") do |r|
|
2019-04-27 10:14:59 -04:00
|
|
|
format_changelog(r, path)
|
2016-11-06 07:57:45 -05:00
|
|
|
end
|
|
|
|
end
|
2019-04-23 02:43:51 -04:00
|
|
|
|
2019-04-27 10:14:59 -04:00
|
|
|
def format_changelog(r, path)
|
|
|
|
IO.copy_stream(r, path)
|
|
|
|
end
|
|
|
|
|
2019-07-30 03:44:29 -04:00
|
|
|
def upstream
|
2019-06-04 22:53:24 -04:00
|
|
|
(branch = cmd_read(%W"#{COMMAND} symbolic-ref --short HEAD")).chomp!
|
|
|
|
(upstream = cmd_read(%W"#{COMMAND} branch --list --format=%(upstream) #{branch}")).chomp!
|
2019-05-08 21:01:31 -04:00
|
|
|
while ref = upstream[%r"\Arefs/heads/(.*)", 1]
|
2019-06-04 22:53:24 -04:00
|
|
|
upstream = cmd_read(%W"#{COMMAND} branch --list --format=%(upstream) #{ref}")
|
2019-05-08 21:01:31 -04:00
|
|
|
end
|
|
|
|
unless %r"\Arefs/remotes/([^/]+)/(.*)" =~ upstream
|
|
|
|
raise "Upstream not found"
|
|
|
|
end
|
2019-07-30 03:44:29 -04:00
|
|
|
[$1, $2]
|
|
|
|
end
|
|
|
|
|
|
|
|
def commit(opts = {})
|
|
|
|
args = [COMMAND, "push"]
|
|
|
|
args << "-n" if dryrun
|
|
|
|
remote, branch = upstream
|
2019-08-12 23:24:15 -04:00
|
|
|
args << remote
|
|
|
|
branches = %W[refs/notes/commits:refs/notes/commits HEAD:#{branch}]
|
2019-06-23 05:28:32 -04:00
|
|
|
if dryrun?
|
2019-08-12 23:24:15 -04:00
|
|
|
branches.each do |b|
|
|
|
|
STDERR.puts((args + [b]).inspect)
|
|
|
|
end
|
2019-06-23 05:28:32 -04:00
|
|
|
return true
|
|
|
|
end
|
2019-08-12 23:24:15 -04:00
|
|
|
branches.each do |b|
|
|
|
|
system(*(args + [b])) or return false
|
|
|
|
end
|
2019-04-23 02:43:51 -04:00
|
|
|
true
|
|
|
|
end
|
2019-04-23 02:39:36 -04:00
|
|
|
end
|
2017-07-26 08:44:12 -04:00
|
|
|
|
2019-04-23 02:39:36 -04:00
|
|
|
class GITSVN < GIT
|
2019-04-27 13:00:39 -04:00
|
|
|
def self.revision_name(rev)
|
2019-05-22 08:14:12 -04:00
|
|
|
SVN.revision_name(rev)
|
2019-04-27 13:00:39 -04:00
|
|
|
end
|
|
|
|
|
2019-05-22 10:18:37 -04:00
|
|
|
def self.short_revision(rev)
|
|
|
|
SVN.short_revision(rev)
|
|
|
|
end
|
|
|
|
|
2019-04-27 10:14:59 -04:00
|
|
|
def format_changelog(r, path)
|
|
|
|
open(path, 'w') do |w|
|
|
|
|
sep = "-"*72
|
|
|
|
w.puts sep
|
|
|
|
while s = r.gets('')
|
|
|
|
author = s[/^Author:\s*(\S+)/, 1]
|
|
|
|
time = s[/^Date:\s*(.+)/, 1]
|
|
|
|
s = r.gets('')
|
|
|
|
s.gsub!(/^ {4}/, '')
|
|
|
|
s.sub!(/^git-svn-id: .*@(\d+) .*\n+\z/, '')
|
|
|
|
rev = $1
|
|
|
|
s.gsub!(/^ {8}/, '') if /^(?! {8}|$)/ !~ s
|
|
|
|
s.sub!(/\n\n\z/, "\n")
|
|
|
|
if /\A(\d+)-(\d+)-(\d+)/ =~ time
|
|
|
|
date = Time.new($1.to_i, $2.to_i, $3.to_i).strftime("%a, %d %b %Y")
|
|
|
|
end
|
|
|
|
lines = s.count("\n")
|
|
|
|
lines = "#{lines} line#{lines == 1 ? '' : 's'}"
|
|
|
|
w.puts "r#{rev} | #{author} | #{time} (#{date}) | #{lines}\n\n"
|
|
|
|
w.puts s, sep
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-02-10 01:42:35 -05:00
|
|
|
def last_changed_revision
|
2017-07-26 08:44:12 -04:00
|
|
|
rev = cmd_read(%W"#{COMMAND} svn info"+[STDERR=>[:child, :out]])[/^Last Changed Rev: (\d+)/, 1]
|
2017-12-18 22:17:37 -05:00
|
|
|
com = cmd_read(%W"#{COMMAND} svn find-rev r#{rev}").chomp
|
2018-02-10 01:42:35 -05:00
|
|
|
return rev, com
|
|
|
|
end
|
|
|
|
|
|
|
|
def commit(opts = {})
|
|
|
|
rev, com = last_changed_revision
|
2018-01-29 23:17:32 -05:00
|
|
|
head = cmd_read(%W"#{COMMAND} symbolic-ref --short HEAD").chomp
|
2017-12-18 22:17:37 -05:00
|
|
|
|
2018-01-26 08:34:09 -05:00
|
|
|
commits = cmd_read([COMMAND, "log", "--reverse", "--format=%H %ae %ce", "#{com}..@"], "rb").split("\n")
|
|
|
|
commits.each_with_index do |l, i|
|
|
|
|
r, a, c = l.split
|
|
|
|
dcommit = [COMMAND, "svn", "dcommit"]
|
2018-02-10 01:42:35 -05:00
|
|
|
dcommit.insert(-2, "-n") if dryrun
|
2018-01-26 08:34:09 -05:00
|
|
|
dcommit << "--add-author-from" unless a == c
|
|
|
|
dcommit << r
|
|
|
|
system(*dcommit) or return false
|
2018-01-29 23:17:32 -05:00
|
|
|
system(COMMAND, "checkout", head) or return false
|
2018-01-26 08:34:09 -05:00
|
|
|
system(COMMAND, "rebase") or return false
|
2017-12-18 22:17:37 -05:00
|
|
|
end
|
|
|
|
|
2018-01-26 08:34:09 -05:00
|
|
|
if rev
|
2017-08-26 17:02:23 -04:00
|
|
|
old = [cmd_read(%W"#{COMMAND} log -1 --format=%H").chomp]
|
|
|
|
old << cmd_read(%W"#{COMMAND} svn reset -r#{rev}")[/^r#{rev} = (\h+)/, 1]
|
|
|
|
3.times do
|
|
|
|
sleep 2
|
|
|
|
system(*%W"#{COMMAND} pull --no-edit --rebase")
|
|
|
|
break unless old.include?(cmd_read(%W"#{COMMAND} log -1 --format=%H").chomp)
|
|
|
|
end
|
2017-07-26 08:44:12 -04:00
|
|
|
end
|
2018-01-26 08:34:09 -05:00
|
|
|
true
|
2017-07-26 08:44:12 -04:00
|
|
|
end
|
2013-11-09 01:20:57 -05:00
|
|
|
end
|
|
|
|
end
|