mirror of
https://gerrit.googlesource.com/git-repo
synced 2025-07-04 20:17:16 +00:00
Compare commits
15 Commits
Author | SHA1 | Date | |
---|---|---|---|
454fdaf119 | |||
f7f9dd4deb | |||
70ee4dd313 | |||
cfe3095e50 | |||
621de7ed12 | |||
d7ebdf56be | |||
fabab4e245 | |||
b577444a90 | |||
1e19f7dd61 | |||
d8b4101eae | |||
1c53b0fa44 | |||
e5ae870a2f | |||
e59e2ae757 | |||
c44ad09309 | |||
4592a63de5 |
1
color.py
1
color.py
@ -210,6 +210,7 @@ class Coloring:
|
|||||||
if have_fg:
|
if have_fg:
|
||||||
bg = a
|
bg = a
|
||||||
else:
|
else:
|
||||||
|
have_fg = True
|
||||||
fg = a
|
fg = a
|
||||||
elif is_attr(a):
|
elif is_attr(a):
|
||||||
attr = a
|
attr = a
|
||||||
|
1
constraints.txt
Normal file
1
constraints.txt
Normal file
@ -0,0 +1 @@
|
|||||||
|
black<24
|
4
error.py
4
error.py
@ -107,6 +107,10 @@ class GitError(RepoError):
|
|||||||
return self.message
|
return self.message
|
||||||
|
|
||||||
|
|
||||||
|
class GitAuthError(RepoExitError):
|
||||||
|
"""Cannot talk to remote due to auth issue."""
|
||||||
|
|
||||||
|
|
||||||
class GitcUnsupportedError(RepoExitError):
|
class GitcUnsupportedError(RepoExitError):
|
||||||
"""Gitc no longer supported."""
|
"""Gitc no longer supported."""
|
||||||
|
|
||||||
|
@ -313,12 +313,15 @@ class GitCommand:
|
|||||||
cwd = None
|
cwd = None
|
||||||
command_name = cmdv[0]
|
command_name = cmdv[0]
|
||||||
command.append(command_name)
|
command.append(command_name)
|
||||||
# Need to use the --progress flag for fetch/clone so output will be
|
|
||||||
# displayed as by default git only does progress output if stderr is a
|
if command_name in ("fetch", "clone"):
|
||||||
# TTY.
|
env["GIT_TERMINAL_PROMPT"] = "0"
|
||||||
if sys.stderr.isatty() and command_name in ("fetch", "clone"):
|
# Need to use the --progress flag for fetch/clone so output will be
|
||||||
if "--progress" not in cmdv and "--quiet" not in cmdv:
|
# displayed as by default git only does progress output if stderr is
|
||||||
command.append("--progress")
|
# a TTY.
|
||||||
|
if sys.stderr.isatty():
|
||||||
|
if "--progress" not in cmdv and "--quiet" not in cmdv:
|
||||||
|
command.append("--progress")
|
||||||
command.extend(cmdv[1:])
|
command.extend(cmdv[1:])
|
||||||
|
|
||||||
event_log = (
|
event_log = (
|
||||||
|
@ -307,8 +307,6 @@ class Superproject:
|
|||||||
)
|
)
|
||||||
return SyncResult(False, False)
|
return SyncResult(False, False)
|
||||||
|
|
||||||
_PrintBetaNotice()
|
|
||||||
|
|
||||||
should_exit = True
|
should_exit = True
|
||||||
if not self._remote_url:
|
if not self._remote_url:
|
||||||
self._LogWarning(
|
self._LogWarning(
|
||||||
@ -452,16 +450,6 @@ class Superproject:
|
|||||||
return UpdateProjectsResult(manifest_path, False)
|
return UpdateProjectsResult(manifest_path, False)
|
||||||
|
|
||||||
|
|
||||||
@functools.lru_cache(maxsize=10)
|
|
||||||
def _PrintBetaNotice():
|
|
||||||
"""Print the notice of beta status."""
|
|
||||||
print(
|
|
||||||
"NOTICE: --use-superproject is in beta; report any issues to the "
|
|
||||||
"address described in `repo version`",
|
|
||||||
file=sys.stderr,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@functools.lru_cache(maxsize=None)
|
@functools.lru_cache(maxsize=None)
|
||||||
def _UseSuperprojectFromConfiguration():
|
def _UseSuperprojectFromConfiguration():
|
||||||
"""Returns the user choice of whether to use superproject."""
|
"""Returns the user choice of whether to use superproject."""
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
.\" DO NOT MODIFY THIS FILE! It was generated by help2man.
|
.\" DO NOT MODIFY THIS FILE! It was generated by help2man.
|
||||||
.TH REPO "1" "October 2022" "repo init" "Repo Manual"
|
.TH REPO "1" "September 2024" "repo init" "Repo Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
repo \- repo init - manual page for repo init
|
repo \- repo init - manual page for repo init
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
@ -28,6 +28,11 @@ manifest repository location
|
|||||||
\fB\-b\fR REVISION, \fB\-\-manifest\-branch\fR=\fI\,REVISION\/\fR
|
\fB\-b\fR REVISION, \fB\-\-manifest\-branch\fR=\fI\,REVISION\/\fR
|
||||||
manifest branch or revision (use HEAD for default)
|
manifest branch or revision (use HEAD for default)
|
||||||
.TP
|
.TP
|
||||||
|
\fB\-\-manifest\-upstream\-branch\fR=\fI\,BRANCH\/\fR
|
||||||
|
when a commit is provided to \fB\-\-manifest\-branch\fR, this
|
||||||
|
is the name of the git ref in which the commit can be
|
||||||
|
found
|
||||||
|
.TP
|
||||||
\fB\-m\fR NAME.xml, \fB\-\-manifest\-name\fR=\fI\,NAME\/\fR.xml
|
\fB\-m\fR NAME.xml, \fB\-\-manifest\-name\fR=\fI\,NAME\/\fR.xml
|
||||||
initial manifest file
|
initial manifest file
|
||||||
.TP
|
.TP
|
||||||
@ -163,6 +168,10 @@ The optional \fB\-b\fR argument can be used to select the manifest branch to che
|
|||||||
and use. If no branch is specified, the remote's default branch is used. This is
|
and use. If no branch is specified, the remote's default branch is used. This is
|
||||||
equivalent to using \fB\-b\fR HEAD.
|
equivalent to using \fB\-b\fR HEAD.
|
||||||
.PP
|
.PP
|
||||||
|
The optional \fB\-\-manifest\-upstream\-branch\fR argument can be used when a commit is
|
||||||
|
provided to \fB\-\-manifest\-branch\fR (or \fB\-b\fR), to specify the name of the git ref in
|
||||||
|
which the commit can be found.
|
||||||
|
.PP
|
||||||
The optional \fB\-m\fR argument can be used to specify an alternate manifest to be
|
The optional \fB\-m\fR argument can be used to specify an alternate manifest to be
|
||||||
used. If no manifest is specified, the manifest default.xml will be used.
|
used. If no manifest is specified, the manifest default.xml will be used.
|
||||||
.PP
|
.PP
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
.\" DO NOT MODIFY THIS FILE! It was generated by help2man.
|
.\" DO NOT MODIFY THIS FILE! It was generated by help2man.
|
||||||
.TH REPO "1" "April 2024" "repo smartsync" "Repo Manual"
|
.TH REPO "1" "September 2024" "repo smartsync" "Repo Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
repo \- repo smartsync - manual page for repo smartsync
|
repo \- repo smartsync - manual page for repo smartsync
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
@ -47,6 +47,10 @@ force remove projects with uncommitted modifications
|
|||||||
if projects no longer exist in the manifest. WARNING:
|
if projects no longer exist in the manifest. WARNING:
|
||||||
this may cause loss of data
|
this may cause loss of data
|
||||||
.TP
|
.TP
|
||||||
|
\fB\-\-rebase\fR
|
||||||
|
rebase local commits regardless of whether they are
|
||||||
|
published
|
||||||
|
.TP
|
||||||
\fB\-l\fR, \fB\-\-local\-only\fR
|
\fB\-l\fR, \fB\-\-local\-only\fR
|
||||||
only update working tree, don't fetch
|
only update working tree, don't fetch
|
||||||
.TP
|
.TP
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
.\" DO NOT MODIFY THIS FILE! It was generated by help2man.
|
.\" DO NOT MODIFY THIS FILE! It was generated by help2man.
|
||||||
.TH REPO "1" "April 2024" "repo sync" "Repo Manual"
|
.TH REPO "1" "September 2024" "repo sync" "Repo Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
repo \- repo sync - manual page for repo sync
|
repo \- repo sync - manual page for repo sync
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
@ -47,6 +47,10 @@ force remove projects with uncommitted modifications
|
|||||||
if projects no longer exist in the manifest. WARNING:
|
if projects no longer exist in the manifest. WARNING:
|
||||||
this may cause loss of data
|
this may cause loss of data
|
||||||
.TP
|
.TP
|
||||||
|
\fB\-\-rebase\fR
|
||||||
|
rebase local commits regardless of whether they are
|
||||||
|
published
|
||||||
|
.TP
|
||||||
\fB\-l\fR, \fB\-\-local\-only\fR
|
\fB\-l\fR, \fB\-\-local\-only\fR
|
||||||
only update working tree, don't fetch
|
only update working tree, don't fetch
|
||||||
.TP
|
.TP
|
||||||
|
71
project.py
71
project.py
@ -32,6 +32,7 @@ import urllib.parse
|
|||||||
|
|
||||||
from color import Coloring
|
from color import Coloring
|
||||||
from error import DownloadError
|
from error import DownloadError
|
||||||
|
from error import GitAuthError
|
||||||
from error import GitError
|
from error import GitError
|
||||||
from error import ManifestInvalidPathError
|
from error import ManifestInvalidPathError
|
||||||
from error import ManifestInvalidRevisionError
|
from error import ManifestInvalidRevisionError
|
||||||
@ -749,7 +750,7 @@ class Project:
|
|||||||
|
|
||||||
def _git(*args):
|
def _git(*args):
|
||||||
# Ignore return code, in case there was no rebase in progress.
|
# Ignore return code, in case there was no rebase in progress.
|
||||||
GitCommand(self, *args, log_as_error=False).Wait()
|
GitCommand(self, args, log_as_error=False).Wait()
|
||||||
|
|
||||||
_git("cherry-pick", "--abort")
|
_git("cherry-pick", "--abort")
|
||||||
_git("rebase", "--abort")
|
_git("rebase", "--abort")
|
||||||
@ -1535,6 +1536,7 @@ class Project:
|
|||||||
syncbuf,
|
syncbuf,
|
||||||
force_sync=False,
|
force_sync=False,
|
||||||
force_checkout=False,
|
force_checkout=False,
|
||||||
|
force_rebase=False,
|
||||||
submodules=False,
|
submodules=False,
|
||||||
errors=None,
|
errors=None,
|
||||||
verbose=False,
|
verbose=False,
|
||||||
@ -1680,18 +1682,21 @@ class Project:
|
|||||||
if pub:
|
if pub:
|
||||||
not_merged = self._revlist(not_rev(revid), pub)
|
not_merged = self._revlist(not_rev(revid), pub)
|
||||||
if not_merged:
|
if not_merged:
|
||||||
if upstream_gain:
|
if upstream_gain and not force_rebase:
|
||||||
# The user has published this branch and some of those
|
# The user has published this branch and some of those
|
||||||
# commits are not yet merged upstream. We do not want
|
# commits are not yet merged upstream. We do not want
|
||||||
# to rewrite the published commits so we punt.
|
# to rewrite the published commits so we punt.
|
||||||
fail(
|
fail(
|
||||||
LocalSyncFail(
|
LocalSyncFail(
|
||||||
"branch %s is published (but not merged) and is "
|
"branch %s is published (but not merged) and is "
|
||||||
"now %d commits behind"
|
"now %d commits behind. Fix this manually or rerun "
|
||||||
|
"with the --rebase option to force a rebase."
|
||||||
% (branch.name, len(upstream_gain)),
|
% (branch.name, len(upstream_gain)),
|
||||||
project=self.name,
|
project=self.name,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
return
|
||||||
|
syncbuf.later1(self, _doff, not verbose)
|
||||||
return
|
return
|
||||||
elif pub == head:
|
elif pub == head:
|
||||||
# All published commits are merged, and thus we are a
|
# All published commits are merged, and thus we are a
|
||||||
@ -2391,26 +2396,25 @@ class Project:
|
|||||||
try:
|
try:
|
||||||
# if revision (sha or tag) is not present then following function
|
# if revision (sha or tag) is not present then following function
|
||||||
# throws an error.
|
# throws an error.
|
||||||
|
revs = [f"{self.revisionExpr}^0"]
|
||||||
|
upstream_rev = None
|
||||||
|
if self.upstream:
|
||||||
|
upstream_rev = self.GetRemote().ToLocal(self.upstream)
|
||||||
|
revs.append(upstream_rev)
|
||||||
|
|
||||||
self.bare_git.rev_list(
|
self.bare_git.rev_list(
|
||||||
"-1",
|
"-1",
|
||||||
"--missing=allow-any",
|
"--missing=allow-any",
|
||||||
"%s^0" % self.revisionExpr,
|
*revs,
|
||||||
"--",
|
"--",
|
||||||
log_as_error=False,
|
log_as_error=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
if self.upstream:
|
if self.upstream:
|
||||||
rev = self.GetRemote().ToLocal(self.upstream)
|
|
||||||
self.bare_git.rev_list(
|
|
||||||
"-1",
|
|
||||||
"--missing=allow-any",
|
|
||||||
"%s^0" % rev,
|
|
||||||
"--",
|
|
||||||
log_as_error=False,
|
|
||||||
)
|
|
||||||
self.bare_git.merge_base(
|
self.bare_git.merge_base(
|
||||||
"--is-ancestor",
|
"--is-ancestor",
|
||||||
self.revisionExpr,
|
self.revisionExpr,
|
||||||
rev,
|
upstream_rev,
|
||||||
log_as_error=False,
|
log_as_error=False,
|
||||||
)
|
)
|
||||||
return True
|
return True
|
||||||
@ -2657,6 +2661,20 @@ class Project:
|
|||||||
# Fallthru to sleep+retry logic at the bottom.
|
# Fallthru to sleep+retry logic at the bottom.
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
# TODO(b/360889369#comment24): git may gc commits incorrectly.
|
||||||
|
# Until the root cause is fixed, retry fetch with --refetch which
|
||||||
|
# will bring the repository into a good state.
|
||||||
|
elif gitcmd.stdout and (
|
||||||
|
"could not parse commit" in gitcmd.stdout
|
||||||
|
or "unable to parse commit" in gitcmd.stdout
|
||||||
|
):
|
||||||
|
cmd.insert(1, "--refetch")
|
||||||
|
print(
|
||||||
|
"could not parse commit error, retrying with refetch",
|
||||||
|
file=output_redir,
|
||||||
|
)
|
||||||
|
continue
|
||||||
|
|
||||||
# Try to prune remote branches once in case there are conflicts.
|
# Try to prune remote branches once in case there are conflicts.
|
||||||
# For example, if the remote had refs/heads/upstream, but deleted
|
# For example, if the remote had refs/heads/upstream, but deleted
|
||||||
# that and now has refs/heads/upstream/foo.
|
# that and now has refs/heads/upstream/foo.
|
||||||
@ -2682,6 +2700,33 @@ class Project:
|
|||||||
)
|
)
|
||||||
# Continue right away so we don't sleep as we shouldn't need to.
|
# Continue right away so we don't sleep as we shouldn't need to.
|
||||||
continue
|
continue
|
||||||
|
elif (
|
||||||
|
ret == 128
|
||||||
|
and gitcmd.stdout
|
||||||
|
and "fatal: could not read Username" in gitcmd.stdout
|
||||||
|
):
|
||||||
|
# User needs to be authenticated, and Git wants to prompt for
|
||||||
|
# username and password.
|
||||||
|
print(
|
||||||
|
"git requires authentication, but repo cannot perform "
|
||||||
|
"interactive authentication. Check git credentials.",
|
||||||
|
file=output_redir,
|
||||||
|
)
|
||||||
|
break
|
||||||
|
elif (
|
||||||
|
ret == 128
|
||||||
|
and gitcmd.stdout
|
||||||
|
and "remote helper 'sso' aborted session" in gitcmd.stdout
|
||||||
|
):
|
||||||
|
# User needs to be authenticated, and Git wants to prompt for
|
||||||
|
# username and password.
|
||||||
|
print(
|
||||||
|
"git requires authentication, but repo cannot perform "
|
||||||
|
"interactive authentication.",
|
||||||
|
file=output_redir,
|
||||||
|
)
|
||||||
|
raise GitAuthError(gitcmd.stdout)
|
||||||
|
break
|
||||||
elif current_branch_only and is_sha1 and ret == 128:
|
elif current_branch_only and is_sha1 and ret == 128:
|
||||||
# Exit code 128 means "couldn't find the ref you asked for"; if
|
# Exit code 128 means "couldn't find the ref you asked for"; if
|
||||||
# we're in sha1 mode, we just tried sync'ing from the upstream
|
# we're in sha1 mode, we just tried sync'ing from the upstream
|
||||||
|
8
repo
8
repo
@ -124,7 +124,7 @@ if not REPO_REV:
|
|||||||
BUG_URL = "https://issues.gerritcodereview.com/issues/new?component=1370071"
|
BUG_URL = "https://issues.gerritcodereview.com/issues/new?component=1370071"
|
||||||
|
|
||||||
# increment this whenever we make important changes to this script
|
# increment this whenever we make important changes to this script
|
||||||
VERSION = (2, 45)
|
VERSION = (2, 48)
|
||||||
|
|
||||||
# increment this if the MAINTAINER_KEYS block is modified
|
# increment this if the MAINTAINER_KEYS block is modified
|
||||||
KEYRING_VERSION = (2, 3)
|
KEYRING_VERSION = (2, 3)
|
||||||
@ -282,6 +282,12 @@ def InitParser(parser):
|
|||||||
metavar="REVISION",
|
metavar="REVISION",
|
||||||
help="manifest branch or revision (use HEAD for default)",
|
help="manifest branch or revision (use HEAD for default)",
|
||||||
)
|
)
|
||||||
|
group.add_option(
|
||||||
|
"--manifest-upstream-branch",
|
||||||
|
help="when a commit is provided to --manifest-branch, this "
|
||||||
|
"is the name of the git ref in which the commit can be found",
|
||||||
|
metavar="BRANCH",
|
||||||
|
)
|
||||||
group.add_option(
|
group.add_option(
|
||||||
"-m",
|
"-m",
|
||||||
"--manifest-name",
|
"--manifest-name",
|
||||||
|
@ -52,6 +52,10 @@ The optional -b argument can be used to select the manifest branch
|
|||||||
to checkout and use. If no branch is specified, the remote's default
|
to checkout and use. If no branch is specified, the remote's default
|
||||||
branch is used. This is equivalent to using -b HEAD.
|
branch is used. This is equivalent to using -b HEAD.
|
||||||
|
|
||||||
|
The optional --manifest-upstream-branch argument can be used when a commit is
|
||||||
|
provided to --manifest-branch (or -b), to specify the name of the git ref in
|
||||||
|
which the commit can be found.
|
||||||
|
|
||||||
The optional -m argument can be used to specify an alternate manifest
|
The optional -m argument can be used to specify an alternate manifest
|
||||||
to be used. If no manifest is specified, the manifest default.xml
|
to be used. If no manifest is specified, the manifest default.xml
|
||||||
will be used.
|
will be used.
|
||||||
@ -135,6 +139,7 @@ to update the working directory files.
|
|||||||
# manifest project is special and is created when instantiating the
|
# manifest project is special and is created when instantiating the
|
||||||
# manifest which happens before we parse options.
|
# manifest which happens before we parse options.
|
||||||
self.manifest.manifestProject.clone_depth = opt.manifest_depth
|
self.manifest.manifestProject.clone_depth = opt.manifest_depth
|
||||||
|
self.manifest.manifestProject.upstream = opt.manifest_upstream_branch
|
||||||
clone_filter_for_depth = (
|
clone_filter_for_depth = (
|
||||||
"blob:none" if (_REPO_ALLOW_SHALLOW == "0") else None
|
"blob:none" if (_REPO_ALLOW_SHALLOW == "0") else None
|
||||||
)
|
)
|
||||||
@ -317,6 +322,12 @@ to update the working directory files.
|
|||||||
" be used with --standalone-manifest."
|
" be used with --standalone-manifest."
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if opt.manifest_upstream_branch and opt.manifest_branch is None:
|
||||||
|
self.OptionParser.error(
|
||||||
|
"--manifest-upstream-branch cannot be used without "
|
||||||
|
"--manifest-branch."
|
||||||
|
)
|
||||||
|
|
||||||
if args:
|
if args:
|
||||||
if opt.manifest_url:
|
if opt.manifest_url:
|
||||||
self.OptionParser.error(
|
self.OptionParser.error(
|
||||||
|
@ -131,6 +131,11 @@ def _SafeCheckoutOrder(checkouts: List[Project]) -> List[List[Project]]:
|
|||||||
return res
|
return res
|
||||||
|
|
||||||
|
|
||||||
|
def _chunksize(projects: int, jobs: int) -> int:
|
||||||
|
"""Calculate chunk size for the given number of projects and jobs."""
|
||||||
|
return min(max(1, projects // jobs), WORKER_BATCH_SIZE)
|
||||||
|
|
||||||
|
|
||||||
class _FetchOneResult(NamedTuple):
|
class _FetchOneResult(NamedTuple):
|
||||||
"""_FetchOne return value.
|
"""_FetchOne return value.
|
||||||
|
|
||||||
@ -400,6 +405,13 @@ later is required to fix a server side protocol bug.
|
|||||||
"projects no longer exist in the manifest. "
|
"projects no longer exist in the manifest. "
|
||||||
"WARNING: this may cause loss of data",
|
"WARNING: this may cause loss of data",
|
||||||
)
|
)
|
||||||
|
p.add_option(
|
||||||
|
"--rebase",
|
||||||
|
dest="rebase",
|
||||||
|
action="store_true",
|
||||||
|
help="rebase local commits regardless of whether they are "
|
||||||
|
"published",
|
||||||
|
)
|
||||||
p.add_option(
|
p.add_option(
|
||||||
"-l",
|
"-l",
|
||||||
"--local-only",
|
"--local-only",
|
||||||
@ -812,7 +824,6 @@ later is required to fix a server side protocol bug.
|
|||||||
def _Fetch(self, projects, opt, err_event, ssh_proxy, errors):
|
def _Fetch(self, projects, opt, err_event, ssh_proxy, errors):
|
||||||
ret = True
|
ret = True
|
||||||
|
|
||||||
jobs = opt.jobs_network
|
|
||||||
fetched = set()
|
fetched = set()
|
||||||
remote_fetched = set()
|
remote_fetched = set()
|
||||||
pm = Progress(
|
pm = Progress(
|
||||||
@ -842,6 +853,8 @@ later is required to fix a server side protocol bug.
|
|||||||
objdir_project_map.setdefault(project.objdir, []).append(project)
|
objdir_project_map.setdefault(project.objdir, []).append(project)
|
||||||
projects_list = list(objdir_project_map.values())
|
projects_list = list(objdir_project_map.values())
|
||||||
|
|
||||||
|
jobs = min(opt.jobs_network, len(projects_list))
|
||||||
|
|
||||||
def _ProcessResults(results_sets):
|
def _ProcessResults(results_sets):
|
||||||
ret = True
|
ret = True
|
||||||
for results in results_sets:
|
for results in results_sets:
|
||||||
@ -881,35 +894,22 @@ later is required to fix a server side protocol bug.
|
|||||||
Sync.ssh_proxy = None
|
Sync.ssh_proxy = None
|
||||||
|
|
||||||
# NB: Multiprocessing is heavy, so don't spin it up for one job.
|
# NB: Multiprocessing is heavy, so don't spin it up for one job.
|
||||||
if len(projects_list) == 1 or jobs == 1:
|
if jobs == 1:
|
||||||
self._FetchInitChild(ssh_proxy)
|
self._FetchInitChild(ssh_proxy)
|
||||||
if not _ProcessResults(
|
if not _ProcessResults(
|
||||||
self._FetchProjectList(opt, x) for x in projects_list
|
self._FetchProjectList(opt, x) for x in projects_list
|
||||||
):
|
):
|
||||||
ret = False
|
ret = False
|
||||||
else:
|
else:
|
||||||
# Favor throughput over responsiveness when quiet. It seems that
|
if not opt.quiet:
|
||||||
# imap() will yield results in batches relative to chunksize, so
|
|
||||||
# even as the children finish a sync, we won't see the result until
|
|
||||||
# one child finishes ~chunksize jobs. When using a large --jobs
|
|
||||||
# with large chunksize, this can be jarring as there will be a large
|
|
||||||
# initial delay where repo looks like it isn't doing anything and
|
|
||||||
# sits at 0%, but then suddenly completes a lot of jobs all at once.
|
|
||||||
# Since this code is more network bound, we can accept a bit more
|
|
||||||
# CPU overhead with a smaller chunksize so that the user sees more
|
|
||||||
# immediate & continuous feedback.
|
|
||||||
if opt.quiet:
|
|
||||||
chunksize = WORKER_BATCH_SIZE
|
|
||||||
else:
|
|
||||||
pm.update(inc=0, msg="warming up")
|
pm.update(inc=0, msg="warming up")
|
||||||
chunksize = 4
|
|
||||||
with multiprocessing.Pool(
|
with multiprocessing.Pool(
|
||||||
jobs, initializer=self._FetchInitChild, initargs=(ssh_proxy,)
|
jobs, initializer=self._FetchInitChild, initargs=(ssh_proxy,)
|
||||||
) as pool:
|
) as pool:
|
||||||
results = pool.imap_unordered(
|
results = pool.imap_unordered(
|
||||||
functools.partial(self._FetchProjectList, opt),
|
functools.partial(self._FetchProjectList, opt),
|
||||||
projects_list,
|
projects_list,
|
||||||
chunksize=chunksize,
|
chunksize=_chunksize(len(projects_list), jobs),
|
||||||
)
|
)
|
||||||
if not _ProcessResults(results):
|
if not _ProcessResults(results):
|
||||||
ret = False
|
ret = False
|
||||||
@ -1009,7 +1009,13 @@ later is required to fix a server side protocol bug.
|
|||||||
return _FetchMainResult(all_projects)
|
return _FetchMainResult(all_projects)
|
||||||
|
|
||||||
def _CheckoutOne(
|
def _CheckoutOne(
|
||||||
self, detach_head, force_sync, force_checkout, verbose, project
|
self,
|
||||||
|
detach_head,
|
||||||
|
force_sync,
|
||||||
|
force_checkout,
|
||||||
|
force_rebase,
|
||||||
|
verbose,
|
||||||
|
project,
|
||||||
):
|
):
|
||||||
"""Checkout work tree for one project
|
"""Checkout work tree for one project
|
||||||
|
|
||||||
@ -1019,6 +1025,7 @@ later is required to fix a server side protocol bug.
|
|||||||
existing git directory that was previously linked to a different
|
existing git directory that was previously linked to a different
|
||||||
object directory).
|
object directory).
|
||||||
force_checkout: Force checking out of the repo content.
|
force_checkout: Force checking out of the repo content.
|
||||||
|
force_rebase: Force rebase.
|
||||||
verbose: Whether to show verbose messages.
|
verbose: Whether to show verbose messages.
|
||||||
project: Project object for the project to checkout.
|
project: Project object for the project to checkout.
|
||||||
|
|
||||||
@ -1036,6 +1043,7 @@ later is required to fix a server side protocol bug.
|
|||||||
syncbuf,
|
syncbuf,
|
||||||
force_sync=force_sync,
|
force_sync=force_sync,
|
||||||
force_checkout=force_checkout,
|
force_checkout=force_checkout,
|
||||||
|
force_rebase=force_rebase,
|
||||||
errors=errors,
|
errors=errors,
|
||||||
verbose=verbose,
|
verbose=verbose,
|
||||||
)
|
)
|
||||||
@ -1109,6 +1117,7 @@ later is required to fix a server side protocol bug.
|
|||||||
opt.detach_head,
|
opt.detach_head,
|
||||||
opt.force_sync,
|
opt.force_sync,
|
||||||
opt.force_checkout,
|
opt.force_checkout,
|
||||||
|
opt.rebase,
|
||||||
opt.verbose,
|
opt.verbose,
|
||||||
),
|
),
|
||||||
projects,
|
projects,
|
||||||
@ -1480,6 +1489,19 @@ later is required to fix a server side protocol bug.
|
|||||||
[success, manifest_str] = server.GetApprovedManifest(
|
[success, manifest_str] = server.GetApprovedManifest(
|
||||||
branch, target
|
branch, target
|
||||||
)
|
)
|
||||||
|
elif (
|
||||||
|
"TARGET_PRODUCT" in os.environ
|
||||||
|
and "TARGET_BUILD_VARIANT" in os.environ
|
||||||
|
and "TARGET_RELEASE" in os.environ
|
||||||
|
):
|
||||||
|
target = "%s-%s-%s" % (
|
||||||
|
os.environ["TARGET_PRODUCT"],
|
||||||
|
os.environ["TARGET_RELEASE"],
|
||||||
|
os.environ["TARGET_BUILD_VARIANT"],
|
||||||
|
)
|
||||||
|
[success, manifest_str] = server.GetApprovedManifest(
|
||||||
|
branch, target
|
||||||
|
)
|
||||||
elif (
|
elif (
|
||||||
"TARGET_PRODUCT" in os.environ
|
"TARGET_PRODUCT" in os.environ
|
||||||
and "TARGET_BUILD_VARIANT" in os.environ
|
and "TARGET_BUILD_VARIANT" in os.environ
|
||||||
|
8
tests/fixtures/test.gitconfig
vendored
8
tests/fixtures/test.gitconfig
vendored
@ -11,3 +11,11 @@
|
|||||||
intk = 10k
|
intk = 10k
|
||||||
intm = 10m
|
intm = 10m
|
||||||
intg = 10g
|
intg = 10g
|
||||||
|
|
||||||
|
[color "status"]
|
||||||
|
one = yellow
|
||||||
|
two = magenta cyan
|
||||||
|
three = black red ul
|
||||||
|
reset = reset
|
||||||
|
none
|
||||||
|
empty =
|
||||||
|
74
tests/test_color.py
Normal file
74
tests/test_color.py
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
# Copyright (C) 2024 The Android Open Source Project
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
"""Unittests for the color.py module."""
|
||||||
|
|
||||||
|
import os
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
import color
|
||||||
|
import git_config
|
||||||
|
|
||||||
|
|
||||||
|
def fixture(*paths):
|
||||||
|
"""Return a path relative to test/fixtures."""
|
||||||
|
return os.path.join(os.path.dirname(__file__), "fixtures", *paths)
|
||||||
|
|
||||||
|
|
||||||
|
class ColoringTests(unittest.TestCase):
|
||||||
|
"""tests of the Coloring class."""
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
"""Create a GitConfig object using the test.gitconfig fixture."""
|
||||||
|
config_fixture = fixture("test.gitconfig")
|
||||||
|
self.config = git_config.GitConfig(config_fixture)
|
||||||
|
color.SetDefaultColoring("true")
|
||||||
|
self.color = color.Coloring(self.config, "status")
|
||||||
|
|
||||||
|
def test_Color_Parse_all_params_none(self):
|
||||||
|
"""all params are None"""
|
||||||
|
val = self.color._parse(None, None, None, None)
|
||||||
|
self.assertEqual("", val)
|
||||||
|
|
||||||
|
def test_Color_Parse_first_parameter_none(self):
|
||||||
|
"""check fg & bg & attr"""
|
||||||
|
val = self.color._parse(None, "black", "red", "ul")
|
||||||
|
self.assertEqual("\x1b[4;30;41m", val)
|
||||||
|
|
||||||
|
def test_Color_Parse_one_entry(self):
|
||||||
|
"""check fg"""
|
||||||
|
val = self.color._parse("one", None, None, None)
|
||||||
|
self.assertEqual("\033[33m", val)
|
||||||
|
|
||||||
|
def test_Color_Parse_two_entry(self):
|
||||||
|
"""check fg & bg"""
|
||||||
|
val = self.color._parse("two", None, None, None)
|
||||||
|
self.assertEqual("\033[35;46m", val)
|
||||||
|
|
||||||
|
def test_Color_Parse_three_entry(self):
|
||||||
|
"""check fg & bg & attr"""
|
||||||
|
val = self.color._parse("three", None, None, None)
|
||||||
|
self.assertEqual("\033[4;30;41m", val)
|
||||||
|
|
||||||
|
def test_Color_Parse_reset_entry(self):
|
||||||
|
"""check reset entry"""
|
||||||
|
val = self.color._parse("reset", None, None, None)
|
||||||
|
self.assertEqual("\033[m", val)
|
||||||
|
|
||||||
|
def test_Color_Parse_empty_entry(self):
|
||||||
|
"""check empty entry"""
|
||||||
|
val = self.color._parse("none", "blue", "white", "dim")
|
||||||
|
self.assertEqual("\033[2;34;47m", val)
|
||||||
|
val = self.color._parse("empty", "green", "white", "bold")
|
||||||
|
self.assertEqual("\033[1;32;47m", val)
|
@ -355,6 +355,30 @@ class SafeCheckoutOrder(unittest.TestCase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class Chunksize(unittest.TestCase):
|
||||||
|
"""Tests for _chunksize."""
|
||||||
|
|
||||||
|
def test_single_project(self):
|
||||||
|
"""Single project."""
|
||||||
|
self.assertEqual(sync._chunksize(1, 1), 1)
|
||||||
|
|
||||||
|
def test_low_project_count(self):
|
||||||
|
"""Multiple projects, low number of projects to sync."""
|
||||||
|
self.assertEqual(sync._chunksize(10, 1), 10)
|
||||||
|
self.assertEqual(sync._chunksize(10, 2), 5)
|
||||||
|
self.assertEqual(sync._chunksize(10, 4), 2)
|
||||||
|
self.assertEqual(sync._chunksize(10, 8), 1)
|
||||||
|
self.assertEqual(sync._chunksize(10, 16), 1)
|
||||||
|
|
||||||
|
def test_high_project_count(self):
|
||||||
|
"""Multiple projects, high number of projects to sync."""
|
||||||
|
self.assertEqual(sync._chunksize(2800, 1), 32)
|
||||||
|
self.assertEqual(sync._chunksize(2800, 16), 32)
|
||||||
|
self.assertEqual(sync._chunksize(2800, 32), 32)
|
||||||
|
self.assertEqual(sync._chunksize(2800, 64), 32)
|
||||||
|
self.assertEqual(sync._chunksize(2800, 128), 21)
|
||||||
|
|
||||||
|
|
||||||
class GetPreciousObjectsState(unittest.TestCase):
|
class GetPreciousObjectsState(unittest.TestCase):
|
||||||
"""Tests for _GetPreciousObjectsState."""
|
"""Tests for _GetPreciousObjectsState."""
|
||||||
|
|
||||||
|
7
tox.ini
7
tox.ini
@ -30,6 +30,7 @@ python =
|
|||||||
|
|
||||||
[testenv]
|
[testenv]
|
||||||
deps =
|
deps =
|
||||||
|
-c constraints.txt
|
||||||
black
|
black
|
||||||
flake8
|
flake8
|
||||||
isort
|
isort
|
||||||
@ -44,17 +45,19 @@ setenv =
|
|||||||
[testenv:lint]
|
[testenv:lint]
|
||||||
skip_install = true
|
skip_install = true
|
||||||
deps =
|
deps =
|
||||||
|
-c constraints.txt
|
||||||
black
|
black
|
||||||
flake8
|
flake8
|
||||||
commands =
|
commands =
|
||||||
black --check {posargs:.}
|
black --check {posargs:. repo run_tests release/update-hooks release/update-manpages}
|
||||||
flake8
|
flake8
|
||||||
|
|
||||||
[testenv:format]
|
[testenv:format]
|
||||||
skip_install = true
|
skip_install = true
|
||||||
deps =
|
deps =
|
||||||
|
-c constraints.txt
|
||||||
black
|
black
|
||||||
flake8
|
flake8
|
||||||
commands =
|
commands =
|
||||||
black {posargs:.}
|
black {posargs:. repo run_tests release/update-hooks release/update-manpages}
|
||||||
flake8
|
flake8
|
||||||
|
Reference in New Issue
Block a user