2008-10-21 14:00:00 +00:00
|
|
|
#
|
|
|
|
# Copyright (C) 2008 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.
|
|
|
|
|
|
|
|
import os
|
|
|
|
import optparse
|
2012-04-16 18:02:21 +00:00
|
|
|
import platform
|
2012-03-29 03:15:45 +00:00
|
|
|
import re
|
2008-10-21 14:00:00 +00:00
|
|
|
import sys
|
|
|
|
|
|
|
|
from error import NoSuchProjectError
|
2012-03-29 03:15:45 +00:00
|
|
|
from error import InvalidProjectGroupsError
|
2008-10-21 14:00:00 +00:00
|
|
|
|
|
|
|
class Command(object):
|
|
|
|
"""Base class for any command line action in repo.
|
|
|
|
"""
|
|
|
|
|
|
|
|
common = False
|
|
|
|
manifest = None
|
|
|
|
_optparse = None
|
|
|
|
|
2009-04-18 20:49:13 +00:00
|
|
|
def WantPager(self, opt):
|
|
|
|
return False
|
|
|
|
|
2008-10-21 14:00:00 +00:00
|
|
|
@property
|
|
|
|
def OptionParser(self):
|
|
|
|
if self._optparse is None:
|
|
|
|
try:
|
|
|
|
me = 'repo %s' % self.NAME
|
|
|
|
usage = self.helpUsage.strip().replace('%prog', me)
|
|
|
|
except AttributeError:
|
|
|
|
usage = 'repo %s' % self.NAME
|
|
|
|
self._optparse = optparse.OptionParser(usage = usage)
|
|
|
|
self._Options(self._optparse)
|
|
|
|
return self._optparse
|
|
|
|
|
|
|
|
def _Options(self, p):
|
|
|
|
"""Initialize the option parser.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def Usage(self):
|
|
|
|
"""Display usage and terminate.
|
|
|
|
"""
|
|
|
|
self.OptionParser.print_usage()
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
def Execute(self, opt, args):
|
|
|
|
"""Perform the action, after option parsing is complete.
|
|
|
|
"""
|
|
|
|
raise NotImplementedError
|
2012-04-16 17:36:08 +00:00
|
|
|
|
2008-10-21 14:00:00 +00:00
|
|
|
def GetProjects(self, args, missing_ok=False):
|
|
|
|
"""A list of projects that match the arguments.
|
|
|
|
"""
|
2012-09-24 03:15:13 +00:00
|
|
|
all_projects = self.manifest.projects
|
2008-10-21 14:00:00 +00:00
|
|
|
result = []
|
|
|
|
|
2012-03-29 03:15:45 +00:00
|
|
|
mp = self.manifest.manifestProject
|
|
|
|
|
|
|
|
groups = mp.config.GetString('manifest.groups')
|
2012-04-23 20:41:58 +00:00
|
|
|
if not groups:
|
2012-08-13 20:11:18 +00:00
|
|
|
groups = 'all,-notdefault,platform-' + platform.system().lower()
|
2012-04-16 17:36:08 +00:00
|
|
|
groups = [x for x in re.split('[,\s]+', groups) if x]
|
2012-03-29 03:15:45 +00:00
|
|
|
|
2008-10-21 14:00:00 +00:00
|
|
|
if not args:
|
2012-10-26 19:18:00 +00:00
|
|
|
for project in all_projects.values():
|
2012-03-29 03:15:45 +00:00
|
|
|
if ((missing_ok or project.Exists) and
|
|
|
|
project.MatchesGroups(groups)):
|
2008-10-21 14:00:00 +00:00
|
|
|
result.append(project)
|
|
|
|
else:
|
2012-10-26 19:18:00 +00:00
|
|
|
by_path = None
|
2008-10-21 14:00:00 +00:00
|
|
|
|
|
|
|
for arg in args:
|
2012-09-24 03:15:13 +00:00
|
|
|
project = all_projects.get(arg)
|
2008-10-21 14:00:00 +00:00
|
|
|
|
|
|
|
if not project:
|
2011-01-10 01:31:57 +00:00
|
|
|
path = os.path.abspath(arg).replace('\\', '/')
|
2012-10-26 19:18:00 +00:00
|
|
|
|
|
|
|
if not by_path:
|
|
|
|
by_path = dict()
|
|
|
|
for p in all_projects.values():
|
|
|
|
by_path[p.worktree] = p
|
|
|
|
|
|
|
|
if os.path.exists(path):
|
|
|
|
oldpath = None
|
|
|
|
while path \
|
|
|
|
and path != oldpath \
|
|
|
|
and path != self.manifest.topdir:
|
|
|
|
try:
|
|
|
|
project = by_path[path]
|
|
|
|
break
|
|
|
|
except KeyError:
|
|
|
|
oldpath = path
|
|
|
|
path = os.path.dirname(path)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
project = by_path[path]
|
|
|
|
except KeyError:
|
|
|
|
pass
|
2008-10-21 14:00:00 +00:00
|
|
|
|
|
|
|
if not project:
|
|
|
|
raise NoSuchProjectError(arg)
|
|
|
|
if not missing_ok and not project.Exists:
|
|
|
|
raise NoSuchProjectError(arg)
|
2012-03-29 03:15:45 +00:00
|
|
|
if not project.MatchesGroups(groups):
|
|
|
|
raise InvalidProjectGroupsError(arg)
|
2008-10-21 14:00:00 +00:00
|
|
|
|
|
|
|
result.append(project)
|
|
|
|
|
|
|
|
def _getpath(x):
|
|
|
|
return x.relpath
|
|
|
|
result.sort(key=_getpath)
|
|
|
|
return result
|
|
|
|
|
2012-10-22 03:50:15 +00:00
|
|
|
# pylint: disable=W0223
|
2012-10-11 07:44:48 +00:00
|
|
|
# Pylint warns that the `InteractiveCommand` and `PagedCommand` classes do not
|
|
|
|
# override method `Execute` which is abstract in `Command`. Since that method
|
|
|
|
# is always implemented in classes derived from `InteractiveCommand` and
|
|
|
|
# `PagedCommand`, this warning can be suppressed.
|
2008-10-21 14:00:00 +00:00
|
|
|
class InteractiveCommand(Command):
|
|
|
|
"""Command which requires user interaction on the tty and
|
|
|
|
must not run within a pager, even if the user asks to.
|
|
|
|
"""
|
2009-04-18 20:49:13 +00:00
|
|
|
def WantPager(self, opt):
|
|
|
|
return False
|
2008-10-21 14:00:00 +00:00
|
|
|
|
|
|
|
class PagedCommand(Command):
|
|
|
|
"""Command which defaults to output in a pager, as its
|
|
|
|
display tends to be larger than one screen full.
|
|
|
|
"""
|
2009-04-18 20:49:13 +00:00
|
|
|
def WantPager(self, opt):
|
|
|
|
return True
|
2009-03-04 01:47:06 +00:00
|
|
|
|
2012-10-22 03:50:15 +00:00
|
|
|
# pylint: enable=W0223
|
2012-10-11 07:44:48 +00:00
|
|
|
|
2009-03-04 01:47:06 +00:00
|
|
|
class MirrorSafeCommand(object):
|
|
|
|
"""Command permits itself to run within a mirror,
|
|
|
|
and does not require a working directory.
|
|
|
|
"""
|