blob: 04307a3c863fc13e58a88311356034f55e92c906 [file] [log] [blame]
Simran Basibdb52712015-08-10 13:23:23 -07001#
2# Copyright (C) 2015 The Android Open Source Project
3#
4# Licensed under the Apache License, Version 2.0 (the "License");
5# you may not use this file except in compliance with the License.
6# You may obtain a copy of the License at
7#
8# http://www.apache.org/licenses/LICENSE-2.0
9#
10# Unless required by applicable law or agreed to in writing, software
11# distributed under the License is distributed on an "AS IS" BASIS,
12# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13# See the License for the specific language governing permissions and
14# limitations under the License.
15
16from __future__ import print_function
17import os
Dan Willemsen250303b2015-09-08 13:27:20 -070018import platform
19import re
David Pursehouse46496d82015-08-20 16:37:09 +090020import sys
Simran Basib9a1b732015-08-20 12:19:28 -070021import time
Simran Basibdb52712015-08-10 13:23:23 -070022
23import git_command
24import git_config
Simran Basi8ce50412015-08-28 14:25:44 -070025import wrapper
Simran Basibdb52712015-08-10 13:23:23 -070026
Dan Willemsen250303b2015-09-08 13:27:20 -070027from manifest_xml import GitcManifest
Simran Basibdb52712015-08-10 13:23:23 -070028
Simran Basibdb52712015-08-10 13:23:23 -070029GITC_FS_ROOT_DIR = '/gitc/manifest-rw/'
30NUM_BATCH_RETRIEVE_REVISIONID = 300
31
Simran Basi8ce50412015-08-28 14:25:44 -070032def get_gitc_manifest_dir():
33 return wrapper.Wrapper().get_gitc_manifest_dir()
34
Simran Basib9a1b732015-08-20 12:19:28 -070035def parse_clientdir(gitc_fs_path):
36 """Parse a path in the GITC FS and return its client name.
37
38 @param gitc_fs_path: A subdirectory path within the GITC_FS_ROOT_DIR.
39
40 @returns: The GITC client name
41 """
42 if (gitc_fs_path == GITC_FS_ROOT_DIR or
43 not gitc_fs_path.startswith(GITC_FS_ROOT_DIR)):
44 return None
45 return gitc_fs_path.split(GITC_FS_ROOT_DIR)[1].split('/')[0]
46
Simran Basibdb52712015-08-10 13:23:23 -070047def _set_project_revisions(projects):
48 """Sets the revisionExpr for a list of projects.
49
50 Because of the limit of open file descriptors allowed, length of projects
51 should not be overly large. Recommend calling this function multiple times
52 with each call not exceeding NUM_BATCH_RETRIEVE_REVISIONID projects.
53
54 @param projects: List of project objects to set the revionExpr for.
55 """
56 # Retrieve the commit id for each project based off of it's current
57 # revisionExpr and it is not already a commit id.
58 project_gitcmds = [(
59 project, git_command.GitCommand(None,
60 ['ls-remote',
61 project.remote.url,
62 project.revisionExpr],
63 capture_stdout=True, cwd='/tmp'))
64 for project in projects if not git_config.IsId(project.revisionExpr)]
65 for proj, gitcmd in project_gitcmds:
66 if gitcmd.Wait():
David Pursehouse022a1d42015-08-20 16:41:04 +090067 print('FATAL: Failed to retrieve revisionExpr for %s' % proj)
Simran Basibdb52712015-08-10 13:23:23 -070068 sys.exit(1)
69 proj.revisionExpr = gitcmd.stdout.split('\t')[0]
70
Dan Willemsen250303b2015-09-08 13:27:20 -070071def _manifest_groups(manifest):
72 """Returns the manifest group string that should be synced
73
74 This is the same logic used by Command.GetProjects(), which is used during
75 repo sync
76
77 @param manifest: The XmlManifest object
78 """
79 mp = manifest.manifestProject
80 groups = mp.config.GetString('manifest.groups')
81 if not groups:
82 groups = 'default,platform-' + platform.system().lower()
83 return groups
84
85def generate_gitc_manifest(repodir, client_name, gitc_manifest, repo_manifest_file, paths=None):
Simran Basibdb52712015-08-10 13:23:23 -070086 """Generate a manifest for shafsd to use for this GITC client.
87
Dan Willemsen250303b2015-09-08 13:27:20 -070088 @param repodir: The repo directory
89 @param client_name: The gitc client name
90 @param gitc_manifest: Current gitc manifest, or None if there isn't one yet
91 @param repo_manifest_file: The file used by the main repo manifest
92 @param paths: List of project paths we want to update.
Simran Basibdb52712015-08-10 13:23:23 -070093 """
Dan Willemsen250303b2015-09-08 13:27:20 -070094 manifest = GitcManifest(repodir, client_name)
95 manifest.Override(repo_manifest_file)
96
Simran Basibdb52712015-08-10 13:23:23 -070097 print('Generating GITC Manifest by fetching revision SHAs for each '
98 'project.')
Dan Willemsen250303b2015-09-08 13:27:20 -070099 if paths is None:
100 paths = manifest.paths.keys()
101
102 groups = [x for x in re.split(r'[,\s]+', _manifest_groups(manifest)) if x]
103
104 # Convert the paths to projects, and filter them to the matched groups.
105 projects = [manifest.paths[p] for p in paths]
106 projects = [p for p in projects if p.MatchesGroups(groups)]
107
108 if gitc_manifest is not None:
109 for path, proj in manifest.paths.iteritems():
110 if not proj.MatchesGroups(groups):
111 continue
112
113 if not proj.upstream and not git_config.IsId(proj.revisionExpr):
114 proj.upstream = proj.revisionExpr
115
116 if not path in gitc_manifest.paths:
117 # Any new projects need their first revision, even if we weren't asked
118 # for them.
119 projects.append(proj)
120 elif not path in paths:
121 # And copy revisions from the previous manifest if we're not updating
122 # them now.
123 gitc_proj = gitc_manifest.paths[path]
124 if gitc_proj.old_revision:
125 proj.revisionExpr = None
126 proj.old_revision = gitc_proj.old_revision
127 else:
128 proj.revisionExpr = gitc_proj.revisionExpr
129
Simran Basibdb52712015-08-10 13:23:23 -0700130 index = 0
Simran Basib9a1b732015-08-20 12:19:28 -0700131 while index < len(projects):
Simran Basibdb52712015-08-10 13:23:23 -0700132 _set_project_revisions(
Simran Basib9a1b732015-08-20 12:19:28 -0700133 projects[index:(index+NUM_BATCH_RETRIEVE_REVISIONID)])
Simran Basibdb52712015-08-10 13:23:23 -0700134 index += NUM_BATCH_RETRIEVE_REVISIONID
Dan Willemsen250303b2015-09-08 13:27:20 -0700135
136 if gitc_manifest is not None:
137 for path, proj in gitc_manifest.paths.iteritems():
138 if proj.old_revision and path in paths:
139 # If we updated a project that has been started, keep the old-revision
140 # updated.
141 repo_proj = manifest.paths[path]
142 repo_proj.old_revision = repo_proj.revisionExpr
143 repo_proj.revisionExpr = None
144
145 # Convert URLs from relative to absolute.
146 for name, remote in manifest.remotes.iteritems():
147 remote.fetchUrl = remote.resolvedFetchUrl
148
Simran Basibdb52712015-08-10 13:23:23 -0700149 # Save the manifest.
Dan Willemsen250303b2015-09-08 13:27:20 -0700150 save_manifest(manifest)
Simran Basib9a1b732015-08-20 12:19:28 -0700151
152def save_manifest(manifest, client_dir=None):
153 """Save the manifest file in the client_dir.
154
155 @param client_dir: Client directory to save the manifest in.
156 @param manifest: Manifest object to save.
157 """
158 if not client_dir:
159 client_dir = manifest.gitc_client_dir
Simran Basibdb52712015-08-10 13:23:23 -0700160 with open(os.path.join(client_dir, '.manifest'), 'w') as f:
Dan Willemsen250303b2015-09-08 13:27:20 -0700161 manifest.Save(f, groups=_manifest_groups(manifest))
Simran Basib9a1b732015-08-20 12:19:28 -0700162 # TODO(sbasi/jorg): Come up with a solution to remove the sleep below.
163 # Give the GITC filesystem time to register the manifest changes.
Dan Willemsen250303b2015-09-08 13:27:20 -0700164 time.sleep(3)