diff options
Diffstat (limited to 'tools/git-reindent-branch.py')
-rw-r--r-- | tools/git-reindent-branch.py | 104 |
1 files changed, 104 insertions, 0 deletions
diff --git a/tools/git-reindent-branch.py b/tools/git-reindent-branch.py new file mode 100644 index 0000000..5f821b0 --- /dev/null +++ b/tools/git-reindent-branch.py @@ -0,0 +1,104 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +import sys, os +import subprocess, argparse, tempfile +import indent + + +def run(cmd): + proc = subprocess.Popen(cmd, stdout=subprocess.PIPE) + rv = proc.communicate("")[0].decode("UTF-8") + proc.wait() + return rv + + +clangfmt = run(["git", "show", "master:.clang-format"]) + +argp = argparse.ArgumentParser(description="git whitespace-fixing tool") +argp.add_argument("branch", metavar="BRANCH", type=str, nargs="?", default="HEAD") +args = argp.parse_args() + +branch = args.branch +commit = run(["git", "rev-list", "-n", "1", branch, "--"]).strip() + +# frr-3.1-dev = first commit that is on master but not on stable/3.0 +masterid = run(["git", "rev-list", "-n", "1", "frr-3.1-dev", "--"]).strip() +masterbase = run(["git", "merge-base", commit, masterid]).strip() + +if masterbase == masterid: + refbranch = "master" +else: + refbranch = "3.0" + +sys.stderr.write("autodetected base: %s (can be 3.0 or master)\n" % refbranch) + +beforeid = run( + ["git", "rev-list", "-n", "1", "reindent-%s-before" % refbranch, "--"] +).strip() +afterid = run( + ["git", "rev-list", "-n", "1", "reindent-%s-after" % refbranch, "--"] +).strip() + +beforebase = run(["git", "merge-base", commit, beforeid]).strip() +afterbase = run(["git", "merge-base", commit, afterid]).strip() + +if afterbase == afterid: + sys.stderr.write("this branch was already rebased\n") + sys.exit(1) + +if beforebase != beforeid: + sys.stderr.write( + 'you need to rebase your branch onto the tag "reindent-%s-before"\n' % refbranch + ) + sys.exit(1) + +revs = ( + run(["git", "rev-list", "reindent-%s-before..%s" % (refbranch, commit)]) + .strip() + .split("\n") +) +revs.reverse() + +srcdir = os.getcwd() +tmpdir = tempfile.mkdtemp("frrindent") +os.chdir(tmpdir) + +sys.stderr.write("using temporary directory %s; %d revisions\n" % (tmpdir, len(revs))) +run(["git", "clone", "-s", "-b", "reindent-%s-after" % refbranch, srcdir, "repo"]) +os.chdir("repo") + +with open(".clang-format", "w") as fd: + fd.write(clangfmt) + +prev = beforeid +for rev in revs: + filestat = ( + run(["git", "diff", "-z", "--name-status", prev, rev]).rstrip("\0").split("\0") + ) + changes = zip(filestat[0::2], filestat[1::2]) + sys.stderr.write("%s: %d files\n" % (rev, len(changes))) + + for typ, name in changes: + if typ == "D": + run(["git", "rm", name]) + elif typ in ["A", "M"]: + run(["git", "checkout", rev, "--", name]) + if name.endswith(".c") or name.endswith(".h"): + for d in ["babeld/", "ldpd/", "nhrpd/"]: + if name.startswith(d): + break + else: + sys.stderr.write("\t%s\n" % name) + indent.wrap_file(name) + run(["git", "add", name]) + + run(["git", "commit", "-C", rev]) + prev = rev + +run(["git", "push", "origin", "HEAD:refs/heads/reindented-branch"]) +sys.stderr.write('\n\n"reindented-branch" should now be OK.\n') +sys.stderr.write( + 'you could use "git reset --hard reindented-branch" to set your current branch to the reindented output\n' +) +sys.stderr.write("\033[31;1mplease always double-check the output\033[m\n") |