Fast, insightful and highly customizable Git history analysis.

Vadim Markovtsev 3b85605be1 Remove redundant travis configs %!s(int64=7) %!d(string=hai) anos
cmd ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
contrib 9524bbb9ea Run gofmt -s %!s(int64=7) %!d(string=hai) anos
doc 265962999b Add Jinja shotness img %!s(int64=7) %!d(string=hai) anos
pb 67e185333d Fix golint warnings in files not on the top level %!s(int64=7) %!d(string=hai) anos
rbtree ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
test_data b7f221856b Try to determine EndPosition if it does not exist %!s(int64=7) %!d(string=hai) anos
toposort ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
vendor 67e185333d Fix golint warnings in files not on the top level %!s(int64=7) %!d(string=hai) anos
yaml 67e185333d Fix golint warnings in files not on the top level %!s(int64=7) %!d(string=hai) anos
.gitignore 4e1327c760 Add FileHistory analysis %!s(int64=7) %!d(string=hai) anos
.travis.yml 3b85605be1 Remove redundant travis configs %!s(int64=7) %!d(string=hai) anos
CODE_OF_CONDUCT.md 7ad35117e9 Add the misc docs %!s(int64=7) %!d(string=hai) anos
CONTRIBUTING.md 7ad35117e9 Add the misc docs %!s(int64=7) %!d(string=hai) anos
DCO 7ad35117e9 Add the misc docs %!s(int64=7) %!d(string=hai) anos
Dockerfile 6fe671bbe6 Repair Dockerfile %!s(int64=7) %!d(string=hai) anos
LICENSE.md 7ad35117e9 Add the misc docs %!s(int64=7) %!d(string=hai) anos
MAINTAINERS.md 7ad35117e9 Add the misc docs %!s(int64=7) %!d(string=hai) anos
Makefile 4d8841d6fa Add AppVeyor %!s(int64=7) %!d(string=hai) anos
OCTOPUS.md 6d566a2aa2 Add the plan to resolve #11 %!s(int64=7) %!d(string=hai) anos
PLUGINS.md 6458136f35 Switch from vanilla "flag" to Cobra %!s(int64=7) %!d(string=hai) anos
README.md 280d13007f Finish with the binary packages %!s(int64=7) %!d(string=hai) anos
appveyor.yml 4d8841d6fa Add AppVeyor %!s(int64=7) %!d(string=hai) anos
blob_cache.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
blob_cache_test.go 79ce321f29 Add more tests %!s(int64=7) %!d(string=hai) anos
burndown.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
burndown_test.go 930f63b6c0 Make constants for dependency names %!s(int64=7) %!d(string=hai) anos
couples.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
couples_test.go 44c4e38ac8 Fix golint warnings unrelated to export and docs %!s(int64=7) %!d(string=hai) anos
day.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
day_test.go 930f63b6c0 Make constants for dependency names %!s(int64=7) %!d(string=hai) anos
diff.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
diff_refiner.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
diff_refiner_test.go b7f221856b Try to determine EndPosition if it does not exist %!s(int64=7) %!d(string=hai) anos
diff_test.go 31d564a3e9 go fmt ./... %!s(int64=7) %!d(string=hai) anos
doc.go fabd3116f6 Add godoc badge %!s(int64=7) %!d(string=hai) anos
dummies.go d2a2093e49 Make CountLines() and BlobToString() public %!s(int64=7) %!d(string=hai) anos
dummies_test.go d2a2093e49 Make CountLines() and BlobToString() public %!s(int64=7) %!d(string=hai) anos
file.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
file_history.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
file_history_test.go fac1929bbc Run go fmt %!s(int64=7) %!d(string=hai) anos
file_test.go b86f61d059 Fix golint warnings related to docs for exported types %!s(int64=7) %!d(string=hai) anos
fix_yaml_unicode.py 3c44f04c8a Make labours.py runnable %!s(int64=8) %!d(string=hai) anos
identity.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
identity_test.go b86f61d059 Fix golint warnings related to docs for exported types %!s(int64=7) %!d(string=hai) anos
labours.py 83b52ec7c0 Add shotness couples support to labours %!s(int64=7) %!d(string=hai) anos
mailmap.go 7aa35d12e4 Fix broken mailmap parsing %!s(int64=8) %!d(string=hai) anos
mailmap_test.go 7aa35d12e4 Fix broken mailmap parsing %!s(int64=8) %!d(string=hai) anos
pipeline.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
pipeline_test.go 12a759efb9 Fix ineffassign and misspell warnings %!s(int64=7) %!d(string=hai) anos
registry.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
registry_test.go 5de1987fc4 Move featureFlags from global scope to PipelineRegistry %!s(int64=7) %!d(string=hai) anos
renames.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
renames_test.go 44c4e38ac8 Fix golint warnings unrelated to export and docs %!s(int64=7) %!d(string=hai) anos
requirements.txt 280d13007f Finish with the binary packages %!s(int64=7) %!d(string=hai) anos
shotness.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
shotness_test.go b7f221856b Try to determine EndPosition if it does not exist %!s(int64=7) %!d(string=hai) anos
swivel.py 763ead8089 Add Tensorflow Projector visualisation of couples %!s(int64=8) %!d(string=hai) anos
tree_diff.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
tree_diff_test.go 79ce321f29 Add more tests %!s(int64=7) %!d(string=hai) anos
uast.go ac9cc316dd Pass through golint warnings %!s(int64=7) %!d(string=hai) anos
uast_test.go 930f63b6c0 Make constants for dependency names %!s(int64=7) %!d(string=hai) anos
version.go b86f61d059 Fix golint warnings related to docs for exported types %!s(int64=7) %!d(string=hai) anos

README.md

Hercules GoDoc Build Status Build status Docker Build Status codecov Go Report Card

Amazingly fast and highly customizable Git repository analysis engine written in Go. Batteries included. Powered by go-git and Babelfish.

There are two tools: hercules and labours.py. The first is the program written in Go which takes a Git repository and runs a Directed Acyclic Graph (DAG) of analysis tasks. The second is the Python script which draws some predefined plots. These two tools are normally used together through a pipe. It is possible to write custom analyses using the plugin system. It is also possible to merge several analysis results together. There is a presentation available.

Hercules DAG of Burndown analysis

The DAG of burndown and couples analyses with UAST diff refining. Generated with hercules --burndown --burndown-people --couples --feature=uast --dry-run --dump-dag doc/dag.dot https://github.com/src-d/hercules

git/git image

torvalds/linux line burndown (granularity 30, sampling 30, resampled by year). Generated with hercules --burndown --pb https://github.com/torvalds/linux | python3 labours.py -f pb -m project

Installation

Grab hercules binary from the Releases page. labours.py requires the Python packages listed in requirements.txt:

pip3 install -r requirements.txt

Numpy and Scipy can be installed on Windows using http://www.lfd.uci.edu/~gohlke/pythonlibs/

Build from source

You are going to need Go (>= v1.8) and Python 2 or 3.

go get -d gopkg.in/src-d/hercules.v3/cmd/hercules
cd $GOPATH/src/gopkg.in/src-d/hercules.v3/cmd/hercules
make

Replace $GOPATH with %GOPATH% on Windows.

Contributions

...are welcome! See CONTRIBUTING and code of conduct.

License

Apache 2.0

Usage

# Use "memory" go-git backend and display the burndown plot. "memory" is the fastest but the repository's git data must fit into RAM.
hercules --burndown https://github.com/src-d/go-git | python3 labours.py -m project --resample month
# Use "file system" go-git backend and print some basic information about the repository.
hercules /path/to/cloned/go-git
# Use "file system" go-git backend, cache the cloned repository to /tmp/repo-cache, use Protocol Buffers and display the burndown plot without resampling.
hercules --burndown --pb https://github.com/git/git /tmp/repo-cache | python3 labours.py -m project -f pb --resample raw

# Now something fun
# Get the linear history from git rev-list, reverse it
# Pipe to hercules, produce burndown snapshots for every 30 days grouped by 30 days
# Save the raw data to cache.yaml, so that later is possible to python3 labours.py -i cache.yaml
# Pipe the raw data to labours.py, set text font size to 16pt, use Agg matplotlib backend and save the plot to output.png
git rev-list HEAD | tac | hercules --commits - --burndown https://github.com/git/git | tee cache.yaml | python3 labours.py -m project --font-size 16 --backend Agg --output git.png

labours.py -i /path/to/yaml allows to read the output from hercules which was saved on disk.

Caching

It is possible to store the cloned repository on disk. The subsequent analysis can run on the corresponding directory instead of cloning from scratch:

# First time - cache
hercules https://github.com/git/git /tmp/repo-cache

# Second time - use the cache
hercules --some-analysis /tmp/repo-cache

Docker image

docker run --rm srcd/hercules hercules --burndown --pb https://github.com/git/git | docker run --rm -i -v $(pwd):/io srcd/hercules labours.py -f pb -m project -o /io/git_git.png

Built-in analyses

Project burndown

hercules --burndown
python3 labours.py -m project

Line burndown statistics for the whole repository. Exactly the same what git-of-theseus does but much faster. Blaming is performed efficiently and incrementally using a custom RB tree tracking algorithm, and only the last modification date is recorded while running the analysis.

All burndown analyses depend on the values of granularity and sampling. Granularity is the number of days each band in the stack consists of. Sampling is the frequency with which the burnout state is snapshotted. The smaller the value, the more smooth is the plot but the more work is done.

There is an option to resample the bands inside labours.py, so that you can define a very precise distribution and visualize it different ways. Besides, resampling aligns the bands across periodic boundaries, e.g. months or years. Unresampled bands are apparently not aligned and start from the project's birth date.

Files

hercules --burndown --burndown-files
python3 labours.py -m file

Burndown statistics for every file in the repository which is alive in the latest revision.

Note: it will generate separate graph for every file. You might don't want to run it on repository with many files.

People

hercules --burndown --burndown-people [-people-dict=/path/to/identities]
python3 labours.py -m person

Burndown statistics for the repository's contributors. If -people-dict is not specified, the identities are discovered by the following algorithm:

  1. We start from the root commit towards the HEAD. Emails and names are converted to lower case.
  2. If we process an unknown email and name, record them as a new developer.
  3. If we process a known email but unknown name, match to the developer with the matching email, and add the unknown name to the list of that developer's names.
  4. If we process an unknown email but known name, match to the developer with the matching name, and add the unknown email to the list of that developer's emails.

If -people-dict is specified, it should point to a text file with the custom identities. The format is: every line is a single developer, it contains all the matching emails and names separated by |. The case is ignored.

Churn matrix

Wireshark top 20 churn matrix

Wireshark top 20 devs - churn matrix

hercules --burndown --burndown-people [-people-dict=/path/to/identities]
python3 labours.py -m churn_matrix

Besides the burndown information, -people collects the added and deleted line statistics per developer. It shows how many lines written by developer A are removed by developer B. The format is the matrix with N rows and (N+2) columns, where N is the number of developers.

  1. First column is the number of lines the developer wrote.
  2. Second column is how many lines were written by the developer and deleted by unidentified developers (if -people-dict is not specified, it is always 0).
  3. The rest of the columns show how many lines were written by the developer and deleted by identified developers.

The sequence of developers is stored in people_sequence YAML node.

Code ownership

Ember.js top 20 code ownership

Ember.js top 20 devs - code ownership

hercules --burndown --burndown-people [-people-dict=/path/to/identities]
python3 labours.py -m ownership

-people also allows to draw the code share through time stacked area plot. That is, how many lines are alive at the sampled moments in time for each identified developer.

Couples

Linux kernel file couples

torvalds/linux files' coupling in Tensorflow Projector

hercules --couples [-people-dict=/path/to/identities]
python3 labours.py -m couples -o <name> [--couples-tmp-dir=/tmp]

Important: it requires Tensorflow to be installed, please follow official instructions.

The files are coupled if they are changed in the same commit. The developers are coupled if they change the same file. hercules records the number of couples throught the whole commit history and outputs the two corresponding co-occurrence matrices. labours.py then trains Swivel embeddings - dense vectors which reflect the co-occurrence probability through the Euclidean distance. The training requires a working Tensorflow installation. The intermediate files are stored in the system temporary directory or --couples-tmp-dir if it is specified. The trained embeddings are written to the current working directory with the name depending on -o. The output format is TSV and matches Tensorflow Projector so that the files and people can be visualized with t-SNE implemented in TF Projector.

Structural hotness

      46  jinja2/compiler.py:visit_Template [FunctionDef]
      42  jinja2/compiler.py:visit_For [FunctionDef]
      34  jinja2/compiler.py:visit_Output [FunctionDef]
      29  jinja2/environment.py:compile [FunctionDef]
      27  jinja2/compiler.py:visit_Include [FunctionDef]
      22  jinja2/compiler.py:visit_Macro [FunctionDef]
      22  jinja2/compiler.py:visit_FromImport [FunctionDef]
      21  jinja2/compiler.py:visit_Filter [FunctionDef]
      21  jinja2/runtime.py:__call__ [FunctionDef]
      20  jinja2/compiler.py:visit_Block [FunctionDef]

Thanks to Babelfish, hercules is able to measure how many times each structural unit has been modified. By default, it looks at functions; refer to UAST XPath manual to set an other query.

hercules --shotness [--shotness-xpath-*]
python3 labours.py -m shotness

Couples analysis automatically loads "shotness" data if available.

Jinja2 functions grouped by structural hotness

hercules --shotness --pb https://github.com/pallets/jinja | python3 labours.py -m couples -f pb

Everything in a single pass

hercules --burndown --burndown-files --burndown-people --couples --shotness [-people-dict=/path/to/identities]
python3 labours.py -m all

Plugins

Hercules has a plugin system and allows to run custom analyses. See PLUGINS.md.

Merging

hercules combine is the command which joins several analysis results in Protocol Buffers format together.

hercules --burndown --pb https://github.com/src-d/go-git > go-git.pb
hercules --burndown --pb https://github.com/src-d/hercules > hercules.pb
hercules combine go-git.pb hercules.pb | python3 labours.py -f pb -m project --resample M

Bad unicode errors

YAML does not support the whole range of Unicode characters and the parser on labours.py side may raise exceptions. Filter the output from hercules through fix_yaml_unicode.py to discard such offending characters.

hercules --burndown --burndown-people https://github.com/... | python3 fix_yaml_unicode.py | python3 labours.py -m people

Plotting

These options affects all plots:

python3 labours.py [--style=white|black] [--backend=] [--size=Y,X]

--style changes the background to be either white ("black" foreground) or black ("white" foreground). --backend chooses the Matplotlib backend. --size sets the size of the figure in inches. The default is 12,9.

(required in macOS) you can pin the default Matplotlib backend with

echo "backend: TkAgg" > ~/.matplotlib/matplotlibrc

These options are effective in burndown charts only:

python3 labours.py [--text-size] [--relative]

--text-size changes the font size, --relative activate the stretched burndown layout.

Custom plotting backend

It is possible to output all the information needed to draw the plots in JSON format. Simply append .json to the output (-o) and you are done. The data format is not fully specified and depends on the Python code which generates it. Each JSON file should contain "type" which reflects the plot kind.

Caveats

  1. Currently, go-git's file system storage backend is considerably slower than the in-memory one, so you should clone repos instead of reading them from disk whenever possible. Please note that the in-memory storage may require much RAM, for example, the Linux kernel takes over 200GB in 2017.
  2. Parsing YAML in Python is slow when the number of internal objects is big. hercules' output for the Linux kernel in "couples" mode is 1.5 GB and takes more than an hour / 180GB RAM to be parsed. However, most of the repositories are parsed within a minute. Try using Protocol Buffers instead (hercules --pb and labours.py -f pb).
  3. To speed-up yaml parsing ```

    Debian, Ubuntu

    apt install libyaml-dev

    macOS

    brew install yaml-cpp libyaml

# you might need to re-install pyyaml for changes to make effect pip uninstall pyyaml pip --no-cache-dir install pyyaml ```