Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(646)

Unified Diff: git_common.py

Issue 184253003: Add git-reup and friends (Closed) Base URL: https://chromium.googlesource.com/chromium/tools/depot_tools.git@freeze_thaw
Patch Set: one more argparse Created 6 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: git_common.py
diff --git a/git_common.py b/git_common.py
index 1296286b42e076c2c5aa265d88eb8b16422a6284..7313507d5069c01eef4a1ca98d10f648b37eedc0 100644
--- a/git_common.py
+++ b/git_common.py
@@ -1,4 +1,4 @@
-# Copyright 2013 The Chromium Authors. All rights reserved.
+# Copyright 2014 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
@@ -16,6 +16,7 @@ IMapIterator.__next__ = IMapIterator.next
import binascii
+import collections
import contextlib
import functools
import logging
@@ -28,6 +29,7 @@ import threading
import subprocess2
+MERGE_BASE_FMT = 'branch.%s.base'
GIT_EXE = 'git.bat' if sys.platform.startswith('win') else 'git'
@@ -200,14 +202,33 @@ class ProgressPrinter(object):
del self._thread
+def once(function):
+ """@Decorates |function| so that it only performs its action once, no matter
+ how many times the decorated |function| is called."""
+ def _inner_gen():
+ yield function()
+ while True:
+ yield
+ return _inner_gen().next
+
+
+## Git functions
+
def branches(*args):
- NO_BRANCH = ('* (no branch)', '* (detached from ')
+ NO_BRANCH = ('* (no branch', '* (detached from ')
for line in run('branch', *args).splitlines():
if line.startswith(NO_BRANCH):
continue
yield line.split()[-1]
+def config(option, default=None):
+ try:
+ return run('config', '--get', option) or default
+ except subprocess2.CalledProcessError:
+ return default
+
+
def config_list(option):
try:
return run('config', '--get-all', option).split()
@@ -219,6 +240,29 @@ def current_branch():
return run('rev-parse', '--abbrev-ref', 'HEAD')
+def del_config(option, scope='local'):
+ run('config', '--' + scope, '--unset', option)
+
+
+def get_branch_tree():
+ """Get the dictionary of {branch: parent}, compatible with topo_iter.
+
+ Returns a tuple of (skipped, <branch_tree dict>) where skipped is a list of
+ branches without upstream branches defined.
+ """
+ skipped = []
agable 2014/03/21 01:14:21 set?
iannucci 2014/03/22 04:17:35 Done.
+ branch_tree = {}
+
+ for branch in branches():
+ parent = upstream(branch)
+ if not parent:
+ skipped.append(branch)
+ continue
+ branch_tree[branch] = parent
+
+ return skipped, branch_tree
+
+
def parse_commitrefs(*commitrefs):
"""Returns binary encoded commit hashes for one or more commitrefs.
@@ -233,41 +277,77 @@ def parse_commitrefs(*commitrefs):
raise BadCommitRefException(commitrefs)
+def root():
+ return config('depot-tools.upstream', 'origin/master')
+
+
def run(*cmd, **kwargs):
- """Runs a git command. Returns stdout as a string.
+ """The same as run_both, except it only returns stdout."""
+ return run_both(*cmd, **kwargs)[0]
+
+
+def run_both(*cmd, **kwargs):
agable 2014/03/21 01:14:21 run_both? really? least descriptive name evar.
iannucci 2014/03/22 04:17:35 Done.
+ """Runs a git command.
- If logging is DEBUG, we'll print the command before we run it.
+ Returns (stdout, stderr) as a pair of strings.
kwargs
autostrip (bool) - Strip the output. Defaults to True.
+ indata (str) - Specifies stdin data for the process.
"""
+ kwargs.setdefault('stdin', subprocess2.PIPE)
+ kwargs.setdefault('stdout', subprocess2.PIPE)
+ kwargs.setdefault('stderr', subprocess2.PIPE)
autostrip = kwargs.pop('autostrip', True)
+ indata = kwargs.pop('indata', None)
- retstream, proc = stream_proc(*cmd, **kwargs)
- ret = retstream.read()
+ cmd = (GIT_EXE,) + cmd
+ proc = subprocess2.Popen(cmd, **kwargs)
+ ret, err = proc.communicate(indata)
retcode = proc.wait()
if retcode != 0:
- raise subprocess2.CalledProcessError(retcode, cmd, os.getcwd(), ret, None)
+ raise subprocess2.CalledProcessError(retcode, cmd, os.getcwd(), ret, err)
if autostrip:
ret = (ret or '').strip()
- return ret
+ err = (err or '').strip()
+ return ret, err
-def stream_proc(*cmd, **kwargs):
- """Runs a git command. Returns stdout as a file.
- If logging is DEBUG, we'll print the command before we run it.
- """
+def stream(*cmd, **kwargs):
agable 2014/03/21 01:14:21 Also poorly named. Since it's executing a command
iannucci 2014/03/22 04:17:35 Done.
+ """Runs a git command. Returns stdout as a file."""
+ kwargs.setdefault('stderr', subprocess2.VOID)
+ kwargs.setdefault('stdout', subprocess2.PIPE)
cmd = (GIT_EXE,) + cmd
- logging.debug('Running %s', ' '.join(repr(tok) for tok in cmd))
- proc = subprocess2.Popen(cmd, stderr=subprocess2.VOID,
- stdout=subprocess2.PIPE, **kwargs)
- return proc.stdout, proc
+ proc = subprocess2.Popen(cmd, **kwargs)
+ return proc.stdout
-def stream(*cmd, **kwargs):
- return stream_proc(*cmd, **kwargs)[0]
+def set_config(option, value, scope='local'):
+ run('config', '--' + scope, option, value)
+
+
+def get_or_create_merge_base(branch, parent=None):
+ """Finds the configured merge base for branch.
+
+ If parent is supplied, it's used instead of calling upstream(branch).
+ """
+ option = MERGE_BASE_FMT % branch
+ base = config(option)
+ if base:
+ try:
+ run('merge-base', '--is-ancestor', base, branch)
+ logging.debug('Found pre-set merge-base for %s: %s', branch, base)
+ except subprocess2.CalledProcessError:
+ logging.debug('Found WRONG pre-set merge-base for %s: %s', branch, base)
+ base = None
+
+ if not base:
+ base = run('merge-base', parent or upstream(branch), branch)
+ set_config(option, base)
+
+ return base
def hash_one(reflike):
@@ -292,10 +372,134 @@ def intern_f(f, kind='blob'):
return ret
+def in_rebase():
+ git_dir = run('rev-parse', '--git-dir')
+ return (
+ os.path.exists(os.path.join(git_dir, 'rebase-merge')) or
+ os.path.exists(os.path.join(git_dir, 'rebase-apply')))
+
+
+def manual_merge_base(branch, base):
+ set_config(MERGE_BASE_FMT % branch, base)
+
+
+def mktree(treedict):
+ """Makes a git tree object and returns its hash.
+
+ See |tree()| for the values of mode, type, and ref.
+
+ Args:
+ treedict - { name: (mode, type, ref) }
+ """
+ with tempfile.TemporaryFile() as f:
+ for name, (mode, typ, ref) in treedict.iteritems():
+ f.write('%s %s %s\t%s\0' % (mode, typ, ref, name))
+ f.seek(0)
+ return run('mktree', '-z', stdin=f)
+
+
+def remove_merge_base(branch):
+ del_config(MERGE_BASE_FMT % branch)
+
+
+RebaseRet = collections.namedtuple('RebaseRet', 'success message')
+
+
+def rebase(parent, start, branch, abort=False, ignore_date=False):
+ """Rebases |start|..|branch| onto the branch |parent|.
agable 2014/03/21 01:14:21 start..branch doesn't actually include start, in g
iannucci 2014/03/22 04:17:35 Er... no it doesn't include start, which is why I
+
+ Args:
+ parent - The new parent ref for the rebased commits.
+ start - The commit to start from
+ branch - The branch to rebase
+ abort - If True, will call git-rebase --abort in the event that the rebase
+ doesn't complete successfully.
+ ignore_date - If True, will cause commit timestamps to match the timestamps
+ of the original commits. Mostly used for getting deterministic
+ timestamps in tests.
+
+ Returns a namedtuple with fields:
+ success - a boolean indicating that the rebase command completed
+ successfully.
+ message - if the rebase failed, this contains the stdout of the failed
+ rebase.
+ """
+ try:
+ args = ['--committer-date-is-author-date'] if ignore_date else []
+ args.extend(('--onto', parent, start, branch))
+ run('rebase', *args)
+ return RebaseRet(True, '')
+ except subprocess2.CalledProcessError as cpe:
+ if abort:
+ run('rebase', '--abort')
+ return RebaseRet(False, cpe.output)
+
+
+def squash_current_branch(header=None, merge_base=None):
+ header = header or 'git squash commit.'
+ merge_base = merge_base or get_or_create_merge_base(current_branch())
+ log_msg = header
+ if log_msg:
+ log_msg += '\n'
+ log_msg += run('log', '--reverse', '--format=%H%n%B', '%s..HEAD' % merge_base)
+ run('reset', '--soft', merge_base)
+ run('commit', '-a', '-F', '-', indata=log_msg)
+
+
def tags(*args):
return run('tag', *args).splitlines()
+def topo_iter(branch_tree, top_down=True):
+ """Generates (branch, parent) in topographical order for a branch tree.
+
+ Given a tree:
+
+ A1
+ B1 B2
+ C1 C2 C3
+ D1
+
+ branch_tree would look like: {
+ 'D1': 'C3',
+ 'C3': 'B2',
+ 'B2': 'A1',
+ 'C1': 'B1',
+ 'C2': 'B1',
+ 'B1': 'A1',
+ }
+
+ It is OK to have multiple 'root' nodes in your graph.
+
+ if top_down is True, items are yielded from A->D. Otherwise they're yielded
+ from D->A. There is no specified ordering within a layer.
+ """
+ branch_tree = branch_tree.copy()
+
+ # TODO(iannucci): There is probably a more efficient way to do these.
+ if top_down:
+ while branch_tree:
+ this_pass = [(b, p) for b, p in branch_tree.iteritems()
+ if p not in branch_tree]
+ assert this_pass, "Branch tree has cycles: %r" % branch_tree
+ for branch, parent in this_pass:
+ yield branch, parent
+ del branch_tree[branch]
+ else:
+ parent_to_branches = collections.defaultdict(set)
+ for branch, parent in branch_tree.iteritems():
+ parent_to_branches[parent].add(branch)
+
+ while branch_tree:
+ this_pass = [(b, p) for b, p in branch_tree.iteritems()
+ if not parent_to_branches[b]]
+ assert this_pass, "Branch tree has cycles: %r" % branch_tree
+ for branch, parent in this_pass:
+ yield branch, parent
+ parent_to_branches[parent].discard(branch)
+ del branch_tree[branch]
+
+
def tree(treeref, recurse=False):
"""Returns a dict representation of a git tree object.
@@ -339,18 +543,3 @@ def upstream(branch):
branch+'@{upstream}')
except subprocess2.CalledProcessError:
return None
-
-
-def mktree(treedict):
- """Makes a git tree object and returns its hash.
-
- See |tree()| for the values of mode, type, and ref.
-
- Args:
- treedict - { name: (mode, type, ref) }
- """
- with tempfile.TemporaryFile() as f:
- for name, (mode, typ, ref) in treedict.iteritems():
- f.write('%s %s %s\t%s\0' % (mode, typ, ref, name))
- f.seek(0)
- return run('mktree', '-z', stdin=f)

Powered by Google App Engine
This is Rietveld 408576698