repo-bisect: Multiple fixes and enhancements
* Add hour granularity. Bisection proceeds on date boundaries until the delta reaches one day, then continues with hours. * Sync the main manifest after "repo sync -l -m ..." because repo resets it. * Use internal git repos (under .repo/projects) to fetch revisions because a working tree may not be available (eg. if a project is added to the main manifest during the bisect timeline). * Several other misc fixes and cleanups.
This commit is contained in:
parent
18a8952483
commit
a6764580b5
110
repo-bisect
110
repo-bisect
|
@ -14,13 +14,56 @@ cfg = dict()
|
||||||
cfg['verbose'] = 0
|
cfg['verbose'] = 0
|
||||||
cfg['nosync'] = False
|
cfg['nosync'] = False
|
||||||
|
|
||||||
|
cwd = os.getcwd()
|
||||||
|
if not os.path.exists('.repo'):
|
||||||
|
sys.stderr.write("Not a repo\n")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
def verbose(s):
|
def verbose(s):
|
||||||
if cfg['verbose'] > 0:
|
if cfg['verbose'] > 0:
|
||||||
sys.stderr.write(s)
|
sys.stderr.write(s)
|
||||||
|
|
||||||
def date_from_string(s):
|
# Valid formats:
|
||||||
ymd = s.split('-')
|
# YYYY-MM-DD-HH-MM-SS
|
||||||
return datetime.datetime(int(ymd[0]), int(ymd[1]), int(ymd[2]))
|
# YYYY-MM-DD-HH
|
||||||
|
# YYYY-MM-DD
|
||||||
|
def string_to_datetime(s):
|
||||||
|
fields = s.split('-')
|
||||||
|
if len(fields) == 6:
|
||||||
|
return datetime.datetime(int(fields[0]), int(fields[1]), int(fields[2]),
|
||||||
|
int(fields[3]), int(fields[4]), int(fields[5]))
|
||||||
|
if len(fields) == 4:
|
||||||
|
return datetime.datetime(int(fields[0]), int(fields[1]), int(fields[2]),
|
||||||
|
int(fields[3]))
|
||||||
|
if len(fields) == 3:
|
||||||
|
return datetime.datetime(int(fields[0]), int(fields[1]), int(fields[2]))
|
||||||
|
return None
|
||||||
|
|
||||||
|
def datetime_to_string(dt):
|
||||||
|
if dt.hour == 0 and dt.minute == 0 and dt.second == 0:
|
||||||
|
return dt.strftime("%Y-%m-%d")
|
||||||
|
return dt.strftime("%Y-%m-%d-%H-%M-%S")
|
||||||
|
|
||||||
|
# Find mid point between two datetime objects.
|
||||||
|
#
|
||||||
|
# Granularity is 1 hour.
|
||||||
|
#
|
||||||
|
# If the objects differ by less than 2 hours, exit.
|
||||||
|
#
|
||||||
|
# If the objects differ by at least 2 days and both hour components are
|
||||||
|
# zero, the resulting hour component will be zero.
|
||||||
|
def datetime_mid(d1, d2):
|
||||||
|
delta = (d2 - d1) / 2
|
||||||
|
if delta.total_seconds() < 3600:
|
||||||
|
print "No more dates left to test"
|
||||||
|
sys.exit(0)
|
||||||
|
mid = d1 + delta
|
||||||
|
if mid.minute != 0 or mid.second != 0:
|
||||||
|
mid = mid.replace(minute=0, second=0)
|
||||||
|
if delta.total_seconds() > 86400 and d1.hour == 0 and d2.hour == 0:
|
||||||
|
if mid.hour != 0:
|
||||||
|
mid = mid.replace(hour=0)
|
||||||
|
return mid
|
||||||
|
|
||||||
def git_config(key):
|
def git_config(key):
|
||||||
args = ['git', 'config', key]
|
args = ['git', 'config', key]
|
||||||
|
@ -36,7 +79,7 @@ def git_reset_hard(rev):
|
||||||
child = subprocess.Popen(args, stdin=None, stdout=subprocess.PIPE, stderr=None)
|
child = subprocess.Popen(args, stdin=None, stdout=subprocess.PIPE, stderr=None)
|
||||||
out, err = child.communicate()
|
out, err = child.communicate()
|
||||||
if child.returncode != 0:
|
if child.returncode != 0:
|
||||||
sys.stderr.write('Failed to read git log\n')
|
sys.stderr.write('Failed to reset git tree\n')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
def git_checkout(rev):
|
def git_checkout(rev):
|
||||||
|
@ -44,7 +87,7 @@ def git_checkout(rev):
|
||||||
child = subprocess.Popen(args, stdin=None, stdout=subprocess.PIPE, stderr=None)
|
child = subprocess.Popen(args, stdin=None, stdout=subprocess.PIPE, stderr=None)
|
||||||
out, err = child.communicate()
|
out, err = child.communicate()
|
||||||
if child.returncode != 0:
|
if child.returncode != 0:
|
||||||
sys.stderr.write('Failed to read git log\n')
|
sys.stderr.write('Failed to checkout git tree\n')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
def git_rev_by_date(date, branch):
|
def git_rev_by_date(date, branch):
|
||||||
|
@ -73,15 +116,15 @@ def repo_manifest():
|
||||||
return ElementTree.fromstring(out)
|
return ElementTree.fromstring(out)
|
||||||
|
|
||||||
def repo_sync_to_date(date):
|
def repo_sync_to_date(date):
|
||||||
print "bisect: sync to %s" % (state['start'])
|
print "bisect: sync to %s" % (date)
|
||||||
|
|
||||||
# Set manifest revision
|
# Set manifest revision
|
||||||
os.chdir('.repo/manifests')
|
os.chdir('.repo/manifests')
|
||||||
remote = git_config('branch.default.remote')
|
remote = git_config('branch.default.remote')
|
||||||
branch = git_config('branch.default.merge')
|
branch = git_config('branch.default.merge')
|
||||||
rev = git_rev_by_date(date, '%s/%s' % (remote, branch))
|
manifest_rev = git_rev_by_date(date, '%s/%s' % (remote, branch))
|
||||||
print "rev at %s = %s" % (date, rev)
|
verbose("manifest revision %s\n" % (manifest_rev))
|
||||||
git_reset_hard(rev)
|
git_reset_hard(manifest_rev)
|
||||||
os.chdir(cwd)
|
os.chdir(cwd)
|
||||||
|
|
||||||
# Fetch the manifest
|
# Fetch the manifest
|
||||||
|
@ -99,23 +142,31 @@ def repo_sync_to_date(date):
|
||||||
for elem in manifest.findall('project'):
|
for elem in manifest.findall('project'):
|
||||||
project_name = elem.get('name')
|
project_name = elem.get('name')
|
||||||
project_path = elem.get('path', project_name)
|
project_path = elem.get('path', project_name)
|
||||||
os.chdir(project_path)
|
|
||||||
project_remote = elem.get('remote', def_remote)
|
project_remote = elem.get('remote', def_remote)
|
||||||
project_revision = elem.get('revision', def_revision)
|
project_revision = elem.get('revision', def_revision)
|
||||||
if project_revision.startswith('refs/tags'):
|
if project_revision.startswith('refs/tags'):
|
||||||
manifest_revision = project_revision.split('/')[2]
|
manifest_revision = project_revision.split('/')[2]
|
||||||
else:
|
else:
|
||||||
manifest_revision = "%s/%s" % (project_remote, project_revision)
|
manifest_revision = "%s/%s" % (project_remote, project_revision)
|
||||||
|
os.chdir(".repo/projects/%s.git" % (project_path))
|
||||||
rev = git_rev_by_date(date, manifest_revision)
|
rev = git_rev_by_date(date, manifest_revision)
|
||||||
elem.set('revision', rev)
|
|
||||||
os.chdir(cwd)
|
os.chdir(cwd)
|
||||||
|
elem.set('revision', rev)
|
||||||
|
|
||||||
# Write new manifest
|
# Write new manifest
|
||||||
pathname = "%s/.repo/manifests/bisect-%s.xml" % (cwd, date)
|
pathname = "%s/.repo/manifests/bisect-%s.xml" % (cwd, date)
|
||||||
|
|
||||||
ElementTree.ElementTree(manifest).write(pathname)
|
ElementTree.ElementTree(manifest).write(pathname)
|
||||||
|
|
||||||
# Sync
|
# Sync the working tree. Note:
|
||||||
|
#
|
||||||
|
# - Both "repo manifest" and "repo sync -m" include local manifests,
|
||||||
|
# so we must move the local manifests out of the way temporarily
|
||||||
|
# to avoid duplicate project errors.
|
||||||
|
#
|
||||||
|
# - "repo sync" always syncs up the main manifest (even with -m), so
|
||||||
|
# we must reset the main manifest after we sync.
|
||||||
|
|
||||||
have_local = os.path.exists('.repo/local_manifests')
|
have_local = os.path.exists('.repo/local_manifests')
|
||||||
if have_local:
|
if have_local:
|
||||||
os.rename('.repo/local_manifests', '.repo/local_manifests.hide')
|
os.rename('.repo/local_manifests', '.repo/local_manifests.hide')
|
||||||
|
@ -129,6 +180,10 @@ def repo_sync_to_date(date):
|
||||||
if have_local:
|
if have_local:
|
||||||
os.rename('.repo/local_manifests.hide', '.repo/local_manifests')
|
os.rename('.repo/local_manifests.hide', '.repo/local_manifests')
|
||||||
|
|
||||||
|
os.chdir('.repo/manifests')
|
||||||
|
git_reset_hard(manifest_rev)
|
||||||
|
os.chdir(cwd)
|
||||||
|
|
||||||
def state_read():
|
def state_read():
|
||||||
s = dict()
|
s = dict()
|
||||||
f = open('.repo/bisect', 'r')
|
f = open('.repo/bisect', 'r')
|
||||||
|
@ -154,6 +209,8 @@ def usage():
|
||||||
print " --verbose Increase verbose level"
|
print " --verbose Increase verbose level"
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
|
### Main code ###
|
||||||
|
|
||||||
optargs, argv = getopt.getopt(sys.argv[1:], 'v',
|
optargs, argv = getopt.getopt(sys.argv[1:], 'v',
|
||||||
['verbose', 'nosync'])
|
['verbose', 'nosync'])
|
||||||
for k, v in optargs:
|
for k, v in optargs:
|
||||||
|
@ -162,11 +219,6 @@ for k, v in optargs:
|
||||||
if k in ('-v', '--verbose'):
|
if k in ('-v', '--verbose'):
|
||||||
cfg['verbose'] += 1
|
cfg['verbose'] += 1
|
||||||
|
|
||||||
cwd = os.getcwd()
|
|
||||||
if not os.path.exists('.repo'):
|
|
||||||
sys.stderr.write("Not a repo\n")
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
if len(argv) < 1:
|
if len(argv) < 1:
|
||||||
usage()
|
usage()
|
||||||
|
|
||||||
|
@ -182,28 +234,20 @@ if action == 'start':
|
||||||
repo_sync_to_date(state['start'])
|
repo_sync_to_date(state['start'])
|
||||||
if action == 'good':
|
if action == 'good':
|
||||||
state = state_read()
|
state = state_read()
|
||||||
d1 = date_from_string(state['start'])
|
d1 = string_to_datetime(state['start'])
|
||||||
d2 = date_from_string(state['end'])
|
d2 = string_to_datetime(state['end'])
|
||||||
delta = (d2 - d1) / 2
|
mid = datetime_mid(d1, d2)
|
||||||
if delta.days < 1:
|
state['start'] = datetime_to_string(mid)
|
||||||
print "No more dates left to test"
|
|
||||||
sys.exit(0)
|
|
||||||
mid = d1 + delta
|
|
||||||
state['start'] = "%s-%s-%s" % (mid.year, mid.month, mid.day)
|
|
||||||
state_write(state)
|
state_write(state)
|
||||||
print "bisect: start=%s, end=%s" % (state['start'], state['end'])
|
print "bisect: start=%s, end=%s" % (state['start'], state['end'])
|
||||||
if not cfg['nosync']:
|
if not cfg['nosync']:
|
||||||
repo_sync_to_date(state['start'])
|
repo_sync_to_date(state['start'])
|
||||||
if action == 'bad':
|
if action == 'bad':
|
||||||
state = state_read()
|
state = state_read()
|
||||||
d1 = date_from_string(state['start'])
|
d1 = string_to_datetime(state['start'])
|
||||||
d2 = date_from_string(state['end'])
|
d2 = string_to_datetime(state['end'])
|
||||||
delta = (d2 - d1) / 2
|
mid = datetime_mid(d1, d2)
|
||||||
if delta.days < 1:
|
state['end'] = datetime_to_string(mid)
|
||||||
print "No more dates left to test"
|
|
||||||
sys.exit(0)
|
|
||||||
mid = d1 + delta
|
|
||||||
state['end'] = "%s-%s-%s" % (mid.year, mid.month, mid.day)
|
|
||||||
state_write(state)
|
state_write(state)
|
||||||
print "bisect: start=%s, end=%s" % (state['start'], state['end'])
|
print "bisect: start=%s, end=%s" % (state['start'], state['end'])
|
||||||
if not cfg['nosync']:
|
if not cfg['nosync']:
|
||||||
|
|
Loading…
Reference in New Issue