waf: git_submodule: don't update submodule if non-fastforward

Don't cause trouble to developers working on submodules.
This commit is contained in:
Gustavo Jose de Sousa 2016-07-22 20:33:56 -03:00 committed by Lucas De Marchi
parent 65e19fd9af
commit 4cd70842d9
1 changed files with 45 additions and 3 deletions

View File

@ -37,23 +37,65 @@ from waflib.Configure import conf
from waflib.TaskGen import before_method, feature, taskgen_method from waflib.TaskGen import before_method, feature, taskgen_method
import os.path import os.path
import re
class update_submodule(Task.Task): class update_submodule(Task.Task):
color = 'BLUE' color = 'BLUE'
run_str = '${GIT} submodule update --recursive --init -- ${SUBMODULE_PATH}' run_str = '${GIT} submodule update --recursive --init -- ${SUBMODULE_PATH}'
fast_forward_diff_re = dict(
removed=re.compile(r'-Subproject commit ([0-9a-f]+)'),
added=re.compile(r'\+Subproject commit ([0-9a-f]+)')
)
def is_fast_forward(self, path):
bld = self.generator.bld
git = self.env.get_flat('GIT')
cmd = git, 'diff', '--submodule=short', '--', os.path.basename(path)
cwd = self.cwd.make_node(os.path.dirname(path))
out = bld.cmd_and_log(cmd, quiet=Context.BOTH, cwd=cwd)
m = self.fast_forward_diff_re['removed'].search(out)
n = self.fast_forward_diff_re['added'].search(out)
if not m or not n:
bld.fatal('git_submodule: failed to parse diff')
head = n.group(1)
wanted = m.group(1)
cmd = git, 'merge-base', head, wanted
cwd = self.cwd.make_node(path)
out = bld.cmd_and_log(cmd, quiet=Context.BOTH, cwd=cwd)
return out.strip() == head
def runnable_status(self): def runnable_status(self):
e = self.env.get_flat e = self.env.get_flat
cmd = e('GIT'), 'submodule', 'status', '--recursive', '--', e('SUBMODULE_PATH') cmd = e('GIT'), 'submodule', 'status', '--recursive', '--', e('SUBMODULE_PATH')
out = self.generator.bld.cmd_and_log(cmd, quiet=Context.BOTH, cwd=self.cwd) out = self.generator.bld.cmd_and_log(cmd, quiet=Context.BOTH, cwd=self.cwd)
self.non_fast_forward = []
# git submodule status uses a blank prefix for submodules that are up # git submodule status uses a blank prefix for submodules that are up
# to date # to date
r = Task.SKIP_ME
for line in out.splitlines(): for line in out.splitlines():
if line[0] != ' ': prefix = line[0]
return Task.RUN_ME path = line[1:].split()[1]
if prefix == ' ':
continue
if prefix == '-':
r = Task.RUN_ME
if prefix == '+':
if not self.is_fast_forward(path):
self.non_fast_forward.append(path)
else:
r = Task.RUN_ME
return Task.SKIP_ME if self.non_fast_forward:
r = Task.SKIP_ME
return r
def uid(self): def uid(self):
if not hasattr(self, 'uid_'): if not hasattr(self, 'uid_'):