2014-09-24 07:28:17 +00:00
|
|
|
module TimeCost
|
|
|
|
class CLI
|
|
|
|
def initialize
|
|
|
|
# FIXME: accept multiple authors
|
|
|
|
@config = {
|
|
|
|
:author_filter_enable => false,
|
|
|
|
:author_filter => ".*?",
|
|
|
|
|
|
|
|
:date_filter_enable => false,
|
2015-02-25 16:15:35 +00:00
|
|
|
:date_filter => [],
|
|
|
|
|
|
|
|
:branches_filter_enable => true,
|
2014-09-24 07:28:17 +00:00
|
|
|
|
|
|
|
:input_dump => [],
|
|
|
|
:output_dump => nil,
|
|
|
|
|
|
|
|
:range_granularity => 0.5, # in decimal hours
|
|
|
|
|
|
|
|
:verbose => false
|
|
|
|
}
|
2014-09-25 07:13:05 +00:00
|
|
|
@rangelist = {}
|
2014-09-24 18:11:05 +00:00
|
|
|
@authorlist = nil
|
2014-09-24 07:28:17 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def parse_cmdline args
|
2015-02-25 16:15:35 +00:00
|
|
|
options = OptionParser.new do |opts|
|
2014-09-24 07:28:17 +00:00
|
|
|
opts.banner = "Usage: #{File.basename $0} [options]"
|
|
|
|
|
|
|
|
opts.on_tail("-v","--verbose", "Run verbosely") do |v|
|
|
|
|
@config[:verbose] = true
|
|
|
|
end
|
|
|
|
|
|
|
|
opts.on_tail("-h","--help", "Show this help") do
|
|
|
|
puts opts
|
|
|
|
exit 0
|
|
|
|
end
|
|
|
|
|
2015-02-25 16:15:35 +00:00
|
|
|
|
2014-09-24 07:28:17 +00:00
|
|
|
opts.on("-i","--input FILE", "Set input dump file") do |file|
|
|
|
|
@config[:input_dump] << file
|
|
|
|
end
|
|
|
|
|
|
|
|
opts.on("-o","--output FILE", "Set output dump file") do |file|
|
|
|
|
@config[:output_dump] = file
|
|
|
|
end
|
|
|
|
|
2015-02-25 16:15:35 +00:00
|
|
|
opts.on("--before DATE", "Keep only commits before DATE") do |date|
|
|
|
|
puts "set date filter to <= #{date}"
|
|
|
|
@config[:date_filter] << lambda { |other|
|
|
|
|
return (other <= DateTime.parse(date))
|
|
|
|
}
|
|
|
|
@config[:date_filter_enable] = true
|
|
|
|
end
|
|
|
|
|
|
|
|
opts.on("--after DATE", "Keep only commits after DATE") do |date|
|
|
|
|
puts "set date filter to >= #{date}"
|
|
|
|
@config[:date_filter] << lambda { |other|
|
|
|
|
return (other >= DateTime.parse(date))
|
|
|
|
}
|
2014-09-24 07:28:17 +00:00
|
|
|
@config[:date_filter_enable] = true
|
|
|
|
end
|
|
|
|
|
2015-02-25 16:15:35 +00:00
|
|
|
opts.on("-t","--time TIME", "Keep only commits on last TIME days") do |time|
|
2014-09-24 07:28:17 +00:00
|
|
|
puts "set time filter to latest #{time} days"
|
|
|
|
@config[:date_filter] = DateTime.now - time.to_f;
|
|
|
|
puts "set date filter to date = #{@config[:date_filter]}"
|
|
|
|
@config[:date_filter_enable] = true
|
|
|
|
end
|
|
|
|
|
|
|
|
opts.on("-a","--author AUTHOR", "Keep only commits by AUTHOR") do |author|
|
|
|
|
puts "set author filter to #{author}"
|
|
|
|
@config[:author_filter] = author
|
|
|
|
@config[:author_filter_enable] = true
|
|
|
|
end
|
|
|
|
|
2015-02-25 16:15:35 +00:00
|
|
|
opts.on_tail("--all", "Collect from all branches and refs") do
|
|
|
|
@config[:branches_filter_enable] = false
|
|
|
|
end
|
|
|
|
|
2014-09-24 07:28:17 +00:00
|
|
|
# overlap :
|
|
|
|
#
|
|
|
|
opts.on("-s","--scotch GRANULARITY", "Use GRANULARITY (decimal hours) to merge ranges") do |granularity|
|
|
|
|
puts "set scotch to #{granularity}"
|
|
|
|
@config[:range_granularity] = granularity.to_f
|
|
|
|
end
|
|
|
|
end
|
2015-02-25 16:15:35 +00:00
|
|
|
options.parse! args
|
2014-09-24 07:28:17 +00:00
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
def analyze_git
|
|
|
|
# git log
|
|
|
|
# foreach, create time range (before) + logs
|
|
|
|
|
2015-02-25 16:15:35 +00:00
|
|
|
cmd = [
|
|
|
|
"git", "log",
|
|
|
|
"--date=iso",
|
|
|
|
"--no-patch"
|
|
|
|
]
|
|
|
|
if not @config[:branches_filter_enable] then
|
|
|
|
cmd << "--all"
|
|
|
|
end
|
|
|
|
cmd.concat ["--", "."]
|
|
|
|
process = IO.popen cmd
|
2014-09-24 07:28:17 +00:00
|
|
|
|
2014-09-25 07:13:05 +00:00
|
|
|
@rangelist = {}
|
2014-09-24 07:28:17 +00:00
|
|
|
commit = nil
|
|
|
|
loop do
|
|
|
|
line = process.gets
|
|
|
|
break if line.nil?
|
|
|
|
# utf-8 fix ?
|
|
|
|
# line.encode!( line.encoding, "binary", :invalid => :replace, :undef => :replace)
|
|
|
|
line.strip!
|
|
|
|
|
|
|
|
case line
|
|
|
|
when /^commit (.*)$/ then
|
|
|
|
id = $1
|
|
|
|
# merge ranges & push
|
|
|
|
unless commit.nil? then
|
2014-09-26 07:42:09 +00:00
|
|
|
range = Range.new commit, granularity: @config[:range_granularity]
|
2014-09-25 07:13:05 +00:00
|
|
|
|
|
|
|
if not @rangelist.include? commit.author then
|
|
|
|
@rangelist[commit.author] = RangeList.new
|
|
|
|
end
|
|
|
|
@rangelist[commit.author].add range
|
2014-09-24 07:28:17 +00:00
|
|
|
end
|
|
|
|
commit = Commit.new id
|
|
|
|
# puts "commit #{id}"
|
|
|
|
|
|
|
|
when /^Author:\s*(.*?)\s*$/ then
|
|
|
|
unless commit.nil? then
|
|
|
|
commit.author = $1
|
|
|
|
|
|
|
|
if @config[:author_filter_enable] and
|
|
|
|
(not commit.author =~ /#{@config[:author_filter]}/) then
|
|
|
|
commit = nil
|
|
|
|
# reject
|
|
|
|
end
|
2014-09-25 07:13:05 +00:00
|
|
|
|
2014-09-24 07:28:17 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
when /^Date:\s*(.*?)\s*$/ then
|
|
|
|
unless commit.nil? then
|
|
|
|
commit.date = $1
|
|
|
|
|
2015-02-25 16:15:35 +00:00
|
|
|
# reject if a some filter does not validate date
|
|
|
|
filter_keep = true
|
|
|
|
filters = @config[:date_filter]
|
|
|
|
filters.each do |f|
|
|
|
|
filter_keep &= f.call(DateTime.parse(commit.date))
|
|
|
|
end
|
|
|
|
|
|
|
|
if not filter_keep then
|
2014-09-24 07:28:17 +00:00
|
|
|
commit = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
when /^\s*$/ then
|
|
|
|
# skip
|
|
|
|
|
|
|
|
else
|
|
|
|
# add as note
|
|
|
|
unless commit.nil? then
|
|
|
|
commit.note = if commit.note.nil? then line
|
|
|
|
else commit.note + "\n" + line
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
def analyze_dumps
|
|
|
|
#read ranges
|
|
|
|
|
|
|
|
@config[:input_dump].each do |filename|
|
2017-09-26 05:49:21 +00:00
|
|
|
filelists = YAML::load(File.open(filename,"r"))
|
|
|
|
# require 'pry'
|
|
|
|
# binding.pry
|
|
|
|
filelists.each do |author, rangelist|
|
|
|
|
# create list if author is new
|
|
|
|
@rangelist[author] ||= RangeList.new
|
|
|
|
|
|
|
|
rangelist.each do |range|
|
|
|
|
@rangelist[author].add range
|
|
|
|
end
|
2014-09-24 07:28:17 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def analyze
|
|
|
|
if @config[:input_dump].empty? then
|
|
|
|
analyze_git
|
|
|
|
else
|
|
|
|
analyze_dumps
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def export
|
|
|
|
return if @config[:output_dump].nil?
|
|
|
|
puts "Exporting to %s" % @config[:output_dump]
|
|
|
|
File.open(@config[:output_dump], "w") do |file|
|
|
|
|
file.puts YAML::dump(@rangelist)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def report
|
|
|
|
return if not @config[:output_dump].nil?
|
|
|
|
|
2014-09-25 07:13:05 +00:00
|
|
|
@rangelist.each do |author,rangelist|
|
|
|
|
rangelist.each do |range|
|
2014-09-26 07:42:09 +00:00
|
|
|
puts range.to_s(!@config[:author_filter_enable]) + "\n"
|
2014-09-25 07:13:05 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
total = 0
|
|
|
|
@rangelist.each do |author,rangelist|
|
|
|
|
puts "SUB-TOTAL for %s: %.2f hours\n" % [author, rangelist.sum]
|
|
|
|
total += rangelist.sum
|
2014-09-24 07:28:17 +00:00
|
|
|
end
|
2014-09-25 07:13:05 +00:00
|
|
|
puts "TOTAL: %.2f hours" % total
|
2014-09-24 07:28:17 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|