2015-10-30 23:54:04 +01:00
|
|
|
# coding: utf-8
|
2012-05-04 11:40:30 +02:00
|
|
|
#
|
2015-10-05 06:17:10 +02:00
|
|
|
# Copyright © 2012-2015 Ejwa Software. All rights reserved.
|
2012-05-04 11:40:30 +02:00
|
|
|
#
|
|
|
|
# This file is part of gitinspector.
|
|
|
|
#
|
|
|
|
# gitinspector is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# gitinspector is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with gitinspector. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2012-05-25 16:42:29 +02:00
|
|
|
from __future__ import print_function
|
2013-06-25 16:37:18 +02:00
|
|
|
from __future__ import unicode_literals
|
2014-01-21 12:14:13 +01:00
|
|
|
import atexit
|
2013-07-14 00:07:36 +02:00
|
|
|
import getopt
|
2012-05-20 22:02:08 +02:00
|
|
|
import os
|
2015-10-20 18:40:08 +02:00
|
|
|
import sys
|
2015-10-31 05:10:00 +01:00
|
|
|
from .blame import Blame
|
2015-10-29 03:09:46 +01:00
|
|
|
from .changes import Changes
|
2015-10-30 01:22:15 +01:00
|
|
|
from .config import GitConfig
|
2015-11-01 03:36:40 +01:00
|
|
|
from .metrics import MetricsLogic
|
2015-10-30 01:22:15 +01:00
|
|
|
from . import (basedir, clone, extensions, filtering, format, help, interval,
|
2015-10-21 05:26:20 +02:00
|
|
|
localization, optval, terminal, version)
|
2015-10-20 18:40:08 +02:00
|
|
|
from .output import outputable
|
|
|
|
from .output.blameoutput import BlameOutput
|
|
|
|
from .output.changesoutput import ChangesOutput
|
|
|
|
from .output.extensionsoutput import ExtensionsOutput
|
|
|
|
from .output.filteringoutput import FilteringOutput
|
|
|
|
from .output.metricsoutput import MetricsOutput
|
|
|
|
from .output.responsibilitiesoutput import ResponsibilitiesOutput
|
|
|
|
from .output.timelineoutput import TimelineOutput
|
|
|
|
|
|
|
|
localization.init()
|
|
|
|
|
2015-10-12 03:15:30 +02:00
|
|
|
class Runner(object):
|
2013-07-14 00:07:36 +02:00
|
|
|
def __init__(self):
|
|
|
|
self.hard = False
|
|
|
|
self.include_metrics = False
|
|
|
|
self.list_file_types = False
|
|
|
|
self.localize_output = False
|
|
|
|
self.responsibilities = False
|
|
|
|
self.grading = False
|
|
|
|
self.timeline = False
|
|
|
|
self.useweeks = False
|
2012-05-04 11:40:30 +02:00
|
|
|
|
2015-10-30 23:46:13 +01:00
|
|
|
def process(self, repos):
|
2015-10-09 03:02:06 +02:00
|
|
|
localization.check_compatibility(version.__version__)
|
|
|
|
|
2013-07-14 00:07:36 +02:00
|
|
|
if not self.localize_output:
|
|
|
|
localization.disable()
|
|
|
|
|
2012-05-05 10:26:02 +02:00
|
|
|
terminal.skip_escapes(not sys.stdout.isatty())
|
2013-05-20 01:48:17 +02:00
|
|
|
terminal.set_stdout_encoding()
|
2012-05-20 22:02:08 +02:00
|
|
|
previous_directory = os.getcwd()
|
2015-11-01 03:36:40 +01:00
|
|
|
|
2015-10-31 05:10:00 +01:00
|
|
|
summed_blames = None
|
|
|
|
summed_changes = None
|
2015-11-01 03:36:40 +01:00
|
|
|
summed_metrics = None
|
2013-07-10 14:23:11 +02:00
|
|
|
|
2015-10-30 23:46:13 +01:00
|
|
|
for repo in repos:
|
|
|
|
os.chdir(previous_directory)
|
|
|
|
os.chdir(repo)
|
|
|
|
absolute_path = basedir.get_basedir_git()
|
2015-10-31 05:10:00 +01:00
|
|
|
changes = Changes(self.hard)
|
|
|
|
summed_blames = Blame(self.hard, self.useweeks, changes) + summed_blames
|
|
|
|
summed_changes = changes + summed_changes
|
2015-10-31 03:44:00 +01:00
|
|
|
|
2015-11-01 03:36:40 +01:00
|
|
|
if self.include_metrics:
|
|
|
|
summed_metrics = MetricsLogic() + summed_metrics
|
|
|
|
|
2015-10-31 03:44:00 +01:00
|
|
|
if sys.stdout.isatty() and format.is_interactive_format():
|
|
|
|
terminal.clear_row()
|
|
|
|
|
2015-10-30 23:46:13 +01:00
|
|
|
os.chdir(absolute_path)
|
2012-05-04 11:40:30 +02:00
|
|
|
|
2015-10-30 23:29:03 +01:00
|
|
|
format.output_header(absolute_path)
|
2015-10-29 03:09:46 +01:00
|
|
|
outputable.output(ChangesOutput(changes))
|
|
|
|
|
|
|
|
if changes.get_commits():
|
2015-10-31 05:10:00 +01:00
|
|
|
outputable.output(BlameOutput(summed_changes, summed_blames))
|
2012-05-04 11:40:30 +02:00
|
|
|
|
2013-07-14 00:07:36 +02:00
|
|
|
if self.timeline:
|
2015-10-31 05:10:00 +01:00
|
|
|
outputable.output(TimelineOutput(summed_changes, self.useweeks))
|
2012-05-04 11:40:30 +02:00
|
|
|
|
2013-07-14 00:07:36 +02:00
|
|
|
if self.include_metrics:
|
2015-11-01 03:36:40 +01:00
|
|
|
outputable.output(MetricsOutput(summed_metrics))
|
2012-05-14 15:24:45 +02:00
|
|
|
|
2013-07-14 00:07:36 +02:00
|
|
|
if self.responsibilities:
|
2015-10-31 05:10:00 +01:00
|
|
|
outputable.output(ResponsibilitiesOutput(summed_changes, summed_blames))
|
2012-05-25 14:27:28 +02:00
|
|
|
|
2015-10-12 03:03:07 +02:00
|
|
|
outputable.output(FilteringOutput())
|
2012-05-07 09:38:11 +02:00
|
|
|
|
2013-07-14 00:07:36 +02:00
|
|
|
if self.list_file_types:
|
2015-10-12 03:03:07 +02:00
|
|
|
outputable.output(ExtensionsOutput())
|
2012-05-04 11:40:30 +02:00
|
|
|
|
2012-10-18 16:59:52 +02:00
|
|
|
format.output_footer()
|
2012-05-20 22:02:08 +02:00
|
|
|
os.chdir(previous_directory)
|
|
|
|
|
2013-06-05 01:50:29 +02:00
|
|
|
def __check_python_version__():
|
2013-06-06 03:05:30 +02:00
|
|
|
if sys.version_info < (2, 6):
|
2013-06-05 01:50:29 +02:00
|
|
|
python_version = str(sys.version_info[0]) + "." + str(sys.version_info[1])
|
2013-07-25 08:52:52 +02:00
|
|
|
sys.exit(_("gitinspector requires at least Python 2.6 to run (version {0} was found).").format(python_version))
|
2013-06-05 01:50:29 +02:00
|
|
|
|
2013-06-14 03:11:56 +02:00
|
|
|
def main():
|
2014-01-08 06:05:10 +01:00
|
|
|
terminal.check_terminal_encoding()
|
|
|
|
terminal.set_stdin_encoding()
|
2013-07-14 03:10:00 +02:00
|
|
|
argv = terminal.convert_command_line_to_utf8()
|
2015-10-30 03:41:09 +01:00
|
|
|
run = Runner()
|
2015-10-30 23:46:13 +01:00
|
|
|
repos = []
|
2012-05-04 11:40:30 +02:00
|
|
|
|
|
|
|
try:
|
2015-10-30 03:41:09 +01:00
|
|
|
opts, args = optval.gnu_getopt(argv[1:], "f:F:hHlLmrTwx:", ["exclude=", "file-types=", "format=",
|
|
|
|
"hard:true", "help", "list-file-types:true", "localize-output:true",
|
|
|
|
"metrics:true", "responsibilities:true", "since=", "grading:true",
|
|
|
|
"timeline:true", "until=", "version", "weeks:true"])
|
2013-07-01 01:38:19 +02:00
|
|
|
|
2015-10-30 23:46:13 +01:00
|
|
|
#Try to clone the repos or return the same directory and bail out.
|
|
|
|
for repo in args:
|
|
|
|
repos.append(clone.create(repo))
|
2014-01-21 12:14:13 +01:00
|
|
|
|
2013-07-01 01:38:19 +02:00
|
|
|
#We need the repo above to be set before we read the git config.
|
2015-10-30 23:46:13 +01:00
|
|
|
GitConfig(run, repos[-1] if len(repos) > 0 else ".", len(repos) > 1).read()
|
2013-07-14 04:29:00 +02:00
|
|
|
clear_x_on_next_pass = True
|
2013-07-01 01:38:19 +02:00
|
|
|
|
2015-10-30 03:41:09 +01:00
|
|
|
for o, a in opts:
|
2013-08-03 10:30:43 +02:00
|
|
|
if o in("-h", "--help"):
|
2013-07-14 00:07:36 +02:00
|
|
|
help.output()
|
|
|
|
sys.exit(0)
|
|
|
|
elif o in("-f", "--file-types"):
|
|
|
|
extensions.define(a)
|
|
|
|
elif o in("-F", "--format"):
|
|
|
|
if not format.select(a):
|
|
|
|
raise format.InvalidFormatError(_("specified output format not supported."))
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "-H":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.hard = True
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--hard":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.hard = optval.get_boolean_argument(a)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "-l":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.list_file_types = True
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--list-file-types":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.list_file_types = optval.get_boolean_argument(a)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "-L":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.localize_output = True
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--localize-output":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.localize_output = optval.get_boolean_argument(a)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "-m":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.include_metrics = True
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--metrics":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.include_metrics = optval.get_boolean_argument(a)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "-r":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.responsibilities = True
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--responsibilities":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.responsibilities = optval.get_boolean_argument(a)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--since":
|
2013-07-14 00:07:36 +02:00
|
|
|
interval.set_since(a)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--version":
|
2013-07-14 00:07:36 +02:00
|
|
|
version.output()
|
|
|
|
sys.exit(0)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--grading":
|
2013-07-14 00:07:36 +02:00
|
|
|
grading = optval.get_boolean_argument(a)
|
2015-10-30 03:41:09 +01:00
|
|
|
run.include_metrics = grading
|
|
|
|
run.list_file_types = grading
|
|
|
|
run.responsibilities = grading
|
|
|
|
run.grading = grading
|
|
|
|
run.hard = grading
|
|
|
|
run.timeline = grading
|
|
|
|
run.useweeks = grading
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "-T":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.timeline = True
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--timeline":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.timeline = optval.get_boolean_argument(a)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--until":
|
2013-07-14 00:07:36 +02:00
|
|
|
interval.set_until(a)
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "-w":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.useweeks = True
|
2013-07-24 09:06:30 +02:00
|
|
|
elif o == "--weeks":
|
2015-10-30 03:41:09 +01:00
|
|
|
run.useweeks = optval.get_boolean_argument(a)
|
2013-07-14 00:07:36 +02:00
|
|
|
elif o in("-x", "--exclude"):
|
2013-07-14 04:29:00 +02:00
|
|
|
if clear_x_on_next_pass:
|
|
|
|
clear_x_on_next_pass = False
|
|
|
|
filtering.clear()
|
2013-07-14 00:07:36 +02:00
|
|
|
filtering.add(a)
|
|
|
|
|
2013-07-14 12:19:31 +02:00
|
|
|
__check_python_version__()
|
2015-10-30 23:46:13 +01:00
|
|
|
run.process(["."] if repos == [] else repos)
|
2013-07-14 12:19:31 +02:00
|
|
|
|
|
|
|
except (filtering.InvalidRegExpError, format.InvalidFormatError, optval.InvalidOptionArgument, getopt.error) as exception:
|
2013-07-15 04:48:25 +02:00
|
|
|
print(sys.argv[0], "\b:", exception.msg, file=sys.stderr)
|
|
|
|
print(_("Try `{0} --help' for more information.").format(sys.argv[0]), file=sys.stderr)
|
2012-05-04 11:40:30 +02:00
|
|
|
sys.exit(2)
|
|
|
|
|
2014-01-21 12:14:13 +01:00
|
|
|
@atexit.register
|
|
|
|
def cleanup():
|
|
|
|
clone.delete()
|
|
|
|
|
2013-06-14 03:11:56 +02:00
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|