5 import xml.etree.ElementTree as ET
9 from common import Colors
10 from common import accept_command
13 SCRIPTDIR = os.path.normpath(os.path.dirname(__file__))
16 def manifest_get_commits(manifest):
18 tree = ET.parse(manifest)
22 if child.tag == 'remote':
23 remotes[child.attrib['name']] = child.attrib['fetch']
24 if child.tag == 'project':
25 name = child.attrib['name']
26 path = child.attrib.get('path', name)
28 remote = child.attrib.get('remote')
30 res[path] = [child.attrib["revision"], [os.path.join(remotes[remote], name), child.attrib['refname']]]
32 res[path] = [child.attrib["revision"], []]
37 def get_branch_name(repo_dir):
38 return git('-C', repo_dir, 'rev-parse', '--symbolic-full-name', 'HEAD').strip()
41 def ensure_revision_if_necessary(repo_dir, revision):
43 Makes sure that @revision is set if the current repo is detached.
46 if get_branch_name(repo_dir) == 'HEAD':
47 revision = git('-C', repo_dir, 'rev-parse', 'HEAD').strip()
52 def update_subprojects(repos_commits, no_interaction=False):
53 subprojects_dir = os.path.join(SCRIPTDIR, "subprojects")
54 for repo_name in os.listdir(subprojects_dir):
55 repo_dir = os.path.normpath(os.path.join(SCRIPTDIR, subprojects_dir, repo_name))
56 if not os.path.exists(os.path.join(repo_dir, '.git')):
59 revision, args = repos_commits.get(repo_name, [None, []])
60 if not update_repo(repo_name, repo_dir, revision, no_interaction, args):
66 def update_repo(repo_name, repo_dir, revision, no_interaction, fetch_args=[], recurse_i=0):
67 revision = ensure_revision_if_necessary(repo_dir, revision)
68 git("config", "rebase.autoStash", "true", repository_path=repo_dir)
71 print("Checking out %s in %s" % (revision, repo_name))
72 git("fetch", *fetch_args, repository_path=repo_dir)
73 git("checkout", "--detach", revision, repository_path=repo_dir)
75 print("Updating branch %s in %s" % (get_branch_name(repo_dir), repo_name))
76 git("pull", "--rebase", repository_path=repo_dir)
77 git("submodule", "update", repository_path=repo_dir)
78 except Exception as e:
79 out = getattr(e, "output", b"").decode()
80 if not no_interaction:
81 print("====================================="
82 "\n%s\nEntering a shell in %s to fix that"
83 " just `exit 0` once done, or `exit 255`"
84 " to skip update for that repository"
85 "\n=====================================" % (
89 shell = os.environ.get("COMSPEC", r"C:\WINDOWS\system32\cmd.exe")
91 shell = os.environ.get("SHELL", os.path.realpath("/bin/sh"))
92 subprocess.check_call(shell, cwd=repo_dir)
93 except subprocess.CalledProcessError as e:
94 if e.returncode == 255:
95 print("Skipping '%s' update" % repo_name)
98 # Result of subshell does not really matter
102 return update_repo(repo_name, repo_dir, revision, no_interaction,
106 print("\nCould not rebase %s, please fix and try again."
107 " Error:\n\n%s %s" % (repo_dir, out, e))
112 commit_message = git("show", "--shortstat", repository_path=repo_dir).split("\n")
113 print(u" -> %s%s%s - %s" % (Colors.HEADER, commit_message[0][7:14], Colors.ENDC,
114 commit_message[4].strip()))
119 if __name__ == "__main__":
120 parser = argparse.ArgumentParser(prog="git-update")
122 parser.add_argument("--no-color",
125 help="Do not output ansi colors.")
126 parser.add_argument("--builddir",
128 help="Specifies the build directory where to"
129 " invoke ninja after updating.")
130 parser.add_argument("--no-interaction",
133 help="Do not allow interaction with the user.")
134 parser.add_argument("--manifest",
136 help="Use a android repo manifest to sync repositories"
137 " Note that it will let all repositories in detached state")
138 options = parser.parse_args()
142 if options.no_interaction:
146 repos_commits = manifest_get_commits(options.manifest)
150 revision, args = repos_commits.get('gst-build', [None, []])
151 if not update_repo('gst-build', SCRIPTDIR, revision, options.no_interaction, args):
154 if not update_subprojects(repos_commits, options.no_interaction):
158 ninja = accept_command(["ninja", "ninja-build"])
160 print("Can't find ninja, other backends are not supported for rebuilding")
163 if not os.path.exists(os.path.join (options.builddir, 'build.ninja')):
164 print("Can't rebuild in %s as no build.ninja file found." % options.builddir)
166 print("Rebuilding all GStreamer modules.")
167 exit(subprocess.call([ninja, '-C', options.builddir]))