Blame dnf/cli/commands/group.py

Packit Service 21c75c
# group.py
Packit Service 21c75c
# Group CLI command.
Packit Service 21c75c
#
Packit Service 21c75c
# Copyright (C) 2012-2016 Red Hat, Inc.
Packit Service 21c75c
#
Packit Service 21c75c
# This copyrighted material is made available to anyone wishing to use,
Packit Service 21c75c
# modify, copy, or redistribute it subject to the terms and conditions of
Packit Service 21c75c
# the GNU General Public License v.2, or (at your option) any later version.
Packit Service 21c75c
# This program is distributed in the hope that it will be useful, but WITHOUT
Packit Service 21c75c
# ANY WARRANTY expressed or implied, including the implied warranties of
Packit Service 21c75c
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General
Packit Service 21c75c
# Public License for more details.  You should have received a copy of the
Packit Service 21c75c
# GNU General Public License along with this program; if not, write to the
Packit Service 21c75c
# Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
Packit Service 21c75c
# 02110-1301, USA.  Any Red Hat trademarks that are incorporated in the
Packit Service 21c75c
# source code or documentation are not subject to the GNU General Public
Packit Service 21c75c
# License and may only be used or replicated with the express permission of
Packit Service 21c75c
# Red Hat, Inc.
Packit Service 21c75c
#
Packit Service 21c75c
Packit Service 21c75c
from __future__ import absolute_import
Packit Service 21c75c
from __future__ import unicode_literals
Packit Service 21c75c
from dnf.comps import CompsQuery
Packit Service 21c75c
from dnf.cli import commands
Packit Service 21c75c
from dnf.i18n import _, ucd
Packit Service 21c75c
Packit Service 21c75c
import libdnf.transaction
Packit Service 21c75c
Packit Service 21c75c
import dnf.cli
Packit Service 21c75c
import dnf.exceptions
Packit Service 21c75c
import dnf.util
Packit Service 21c75c
import logging
Packit Service 21c75c
Packit Service 21c75c
logger = logging.getLogger("dnf")
Packit Service 21c75c
Packit Service 21c75c
class GroupCommand(commands.Command):
Packit Service 21c75c
    """ Single sub-command interface for most groups interaction. """
Packit Service 21c75c
Packit Service 21c75c
    direct_commands = {'grouplist'    : 'list',
Packit Service 21c75c
                       'groupinstall' : 'install',
Packit Service 21c75c
                       'groupupdate'  : 'install',
Packit Service 21c75c
                       'groupremove'  : 'remove',
Packit Service 21c75c
                       'grouperase'   : 'remove',
Packit Service 21c75c
                       'groupinfo'    : 'info'}
Packit Service 21c75c
    aliases = ('group', 'groups', 'grp') + tuple(direct_commands.keys())
Packit Service 21c75c
    summary = _('display, or use, the groups information')
Packit Service 21c75c
Packit Service 21c75c
    _CMD_ALIASES = {'update'     : 'upgrade',
Packit Service 21c75c
                    'erase'      : 'remove'}
Packit Service 21c75c
    _MARK_CMDS = ('install', 'remove')
Packit Service 21c75c
    _GROUP_SUBCOMMANDS = ('summary', 'list', 'info', 'remove', 'install', 'upgrade', 'mark')
Packit Service 21c75c
Packit Service 21c75c
Packit Service 21c75c
    def _canonical(self):
Packit Service 21c75c
        # were we called with direct command?
Packit Service 21c75c
        direct = self.direct_commands.get(self.opts.command)
Packit Service 21c75c
        if direct:
Packit Service 21c75c
            # canonize subcmd and args
Packit Service 21c75c
            if self.opts.subcmd is not None:
Packit Service 21c75c
                self.opts.args.insert(0, self.opts.subcmd)
Packit Service 21c75c
            self.opts.subcmd = direct
Packit Service 21c75c
        if self.opts.subcmd is None:
Packit Service 21c75c
            self.opts.subcmd = 'summary'
Packit Service 21c75c
        self.opts.subcmd = self._CMD_ALIASES.get(self.opts.subcmd,
Packit Service 21c75c
                                                 self.opts.subcmd)
Packit Service 21c75c
Packit Service 21c75c
    def __init__(self, cli):
Packit Service 21c75c
        super(GroupCommand, self).__init__(cli)
Packit Service 21c75c
        self._remark = False
Packit Service 21c75c
Packit Service 21c75c
    def _assert_comps(self):
Packit Service 21c75c
        msg = _('No group data available for configured repositories.')
Packit Service 21c75c
        if not len(self.base.comps):
Packit Service 21c75c
            raise dnf.exceptions.CompsError(msg)
Packit Service 21c75c
Packit Service 21c75c
    def _environment_lists(self, patterns):
Packit Service 21c75c
        def available_pred(env):
Packit Service 21c75c
            env_found = self.base.history.env.get(env.id)
Packit Service 21c75c
            return not(env_found)
Packit Service 21c75c
Packit Service 21c75c
        self._assert_comps()
Packit Service 21c75c
        if patterns is None:
Packit Service 21c75c
            envs = self.base.comps.environments
Packit Service 21c75c
        else:
Packit Service 21c75c
            envs = self.base.comps.environments_by_pattern(",".join(patterns))
Packit Service 21c75c
Packit Service 21c75c
        return dnf.util.mapall(list, dnf.util.partition(available_pred, envs))
Packit Service 21c75c
Packit Service 21c75c
    def _group_lists(self, uservisible, patterns):
Packit Service 21c75c
        def installed_pred(group):
Packit Service 21c75c
            group_found = self.base.history.group.get(group.id)
Packit Service 21c75c
            if group_found:
Packit Service 21c75c
                return True
Packit Service 21c75c
            return False
Packit Service 21c75c
        installed = []
Packit Service 21c75c
        available = []
Packit Service 21c75c
Packit Service 21c75c
        self._assert_comps()
Packit Service 21c75c
Packit Service 21c75c
        if patterns is None:
Packit Service 21c75c
            grps = self.base.comps.groups
Packit Service 21c75c
        else:
Packit Service 21c75c
            grps = self.base.comps.groups_by_pattern(",".join(patterns))
Packit Service 21c75c
        for grp in grps:
Packit Service 21c75c
            tgt_list = available
Packit Service 21c75c
            if installed_pred(grp):
Packit Service 21c75c
                tgt_list = installed
Packit Service 21c75c
            if not uservisible or grp.uservisible:
Packit Service 21c75c
                tgt_list.append(grp)
Packit Service 21c75c
Packit Service 21c75c
        return installed, available
Packit Service 21c75c
Packit Service 21c75c
    def _info(self, userlist):
Packit Service 21c75c
        for strng in userlist:
Packit Service 21c75c
            group_matched = False
Packit Service 21c75c
Packit Service 21c75c
            for env in self.base.comps.environments_by_pattern(strng):
Packit Service 21c75c
                self.output.display_groups_in_environment(env)
Packit Service 21c75c
                group_matched = True
Packit Service 21c75c
Packit Service 21c75c
            for group in self.base.comps.groups_by_pattern(strng):
Packit Service 21c75c
                self.output.display_pkgs_in_groups(group)
Packit Service 21c75c
                group_matched = True
Packit Service 21c75c
Packit Service 21c75c
            if not group_matched:
Packit Service 21c75c
                logger.error(_('Warning: Group %s does not exist.'), strng)
Packit Service 21c75c
Packit Service 21c75c
        return 0, []
Packit Service 21c75c
Packit Service 21c75c
    def _list(self, userlist):
Packit Service 21c75c
        uservisible = 1
Packit Service 21c75c
        showinstalled = 0
Packit Service 21c75c
        showavailable = 0
Packit Service 21c75c
        print_ids = self.base.conf.verbose or self.opts.ids
Packit Service 21c75c
Packit Service 21c75c
        while userlist:
Packit Service 21c75c
            if userlist[0] == 'hidden':
Packit Service 21c75c
                uservisible = 0
Packit Service 21c75c
                userlist.pop(0)
Packit Service 21c75c
            elif userlist[0] == 'installed':
Packit Service 21c75c
                showinstalled = 1
Packit Service 21c75c
                userlist.pop(0)
Packit Service 21c75c
            elif userlist[0] == 'available':
Packit Service 21c75c
                showavailable = 1
Packit Service 21c75c
                userlist.pop(0)
Packit Service 21c75c
            elif userlist[0] == 'ids':
Packit Service 21c75c
                print_ids = True
Packit Service 21c75c
                userlist.pop(0)
Packit Service 21c75c
            else:
Packit Service 21c75c
                break
Packit Service 21c75c
        if self.opts.hidden:
Packit Service 21c75c
            uservisible = 0
Packit Service 21c75c
        if self.opts.installed:
Packit Service 21c75c
            showinstalled = 1
Packit Service 21c75c
        if self.opts.available:
Packit Service 21c75c
            showavailable = 1
Packit Service 21c75c
        if not userlist:
Packit Service 21c75c
            userlist = None # Match everything...
Packit Service 21c75c
Packit Service 21c75c
        errs = False
Packit Service 21c75c
        if userlist is not None:
Packit Service 21c75c
            for group in userlist:
Packit Service 21c75c
                comps = self.base.comps
Packit Service 21c75c
                in_group = len(comps.groups_by_pattern(group)) > 0
Packit Service 21c75c
                in_environment = len(comps.environments_by_pattern(group)) > 0
Packit Service 21c75c
                if not in_group and not in_environment:
Packit Service 21c75c
                    logger.error(_('Warning: No groups match:') + '\n   %s',
Packit Service 21c75c
                                 group)
Packit Service 21c75c
                    errs = True
Packit Service 21c75c
            if errs:
Packit Service 21c75c
                return 0, []
Packit Service 21c75c
Packit Service 21c75c
        env_inst, env_avail = self._environment_lists(userlist)
Packit Service 21c75c
        installed, available = self._group_lists(uservisible, userlist)
Packit Service 21c75c
Packit Service 21c75c
        def _out_grp(sect, group):
Packit Service 21c75c
            if not done:
Packit Service 21c75c
                print(sect)
Packit Service 21c75c
            msg = '   %s' % (group.ui_name if group.ui_name is not None else _("<name-unset>"))
Packit Service 21c75c
            if print_ids:
Packit Service 21c75c
                msg += ' (%s)' % group.id
Packit Service 21c75c
            if group.lang_only:
Packit Service 21c75c
                msg += ' [%s]' % group.lang_only
Packit Service 21c75c
            print('{}'.format(msg))
Packit Service 21c75c
Packit Service 21c75c
        def _out_env(sect, envs):
Packit Service 21c75c
            if envs:
Packit Service 21c75c
                print(sect)
Packit Service 21c75c
            for e in envs:
Packit Service 21c75c
                msg = '   %s' % (e.ui_name if e.ui_name is not None else _("<name-unset>"))
Packit Service 21c75c
                if print_ids:
Packit Service 21c75c
                    msg += ' (%s)' % e.id
Packit Service 21c75c
                print(msg)
Packit Service 21c75c
Packit Service 21c75c
        if not showinstalled:
Packit Service 21c75c
            _out_env(_('Available Environment Groups:'), env_avail)
Packit Service 21c75c
        if not showavailable:
Packit Service 21c75c
            _out_env(_('Installed Environment Groups:'), env_inst)
Packit Service 21c75c
Packit Service 21c75c
        if not showavailable:
Packit Service 21c75c
            done = False
Packit Service 21c75c
            for group in installed:
Packit Service 21c75c
                if group.lang_only:
Packit Service 21c75c
                    continue
Packit Service 21c75c
                _out_grp(_('Installed Groups:'), group)
Packit Service 21c75c
                done = True
Packit Service 21c75c
Packit Service 21c75c
            done = False
Packit Service 21c75c
            for group in installed:
Packit Service 21c75c
                if not group.lang_only:
Packit Service 21c75c
                    continue
Packit Service 21c75c
                _out_grp(_('Installed Language Groups:'), group)
Packit Service 21c75c
                done = True
Packit Service 21c75c
Packit Service 21c75c
        if showinstalled:
Packit Service 21c75c
            return 0, []
Packit Service 21c75c
Packit Service 21c75c
        done = False
Packit Service 21c75c
        for group in available:
Packit Service 21c75c
            if group.lang_only:
Packit Service 21c75c
                continue
Packit Service 21c75c
            _out_grp(_('Available Groups:'), group)
Packit Service 21c75c
            done = True
Packit Service 21c75c
Packit Service 21c75c
        done = False
Packit Service 21c75c
        for group in available:
Packit Service 21c75c
            if not group.lang_only:
Packit Service 21c75c
                continue
Packit Service 21c75c
            _out_grp(_('Available Language Groups:'), group)
Packit Service 21c75c
            done = True
Packit Service 21c75c
Packit Service 21c75c
        return 0, []
Packit Service 21c75c
Packit Service 21c75c
    def _mark_install(self, patterns):
Packit Service 21c75c
        q = CompsQuery(self.base.comps, self.base.history,
Packit Service 21c75c
                       CompsQuery.GROUPS | CompsQuery.ENVIRONMENTS,
Packit Service 21c75c
                       CompsQuery.AVAILABLE | CompsQuery.INSTALLED)
Packit Service 21c75c
        solver = self.base._build_comps_solver()
Packit Service 21c75c
        res = q.get(*patterns)
Packit Service 21c75c
Packit Service 21c75c
        if self.opts.with_optional:
Packit Service 21c75c
            types = tuple(self.base.conf.group_package_types + ['optional'])
Packit Service 21c75c
        else:
Packit Service 21c75c
            types = tuple(self.base.conf.group_package_types)
Packit Service 21c75c
        pkg_types = libdnf.transaction.listToCompsPackageType(types)
Packit Service 21c75c
        for env_id in res.environments:
Packit Service 21c75c
            dnf.comps.install_or_skip(solver._environment_install, env_id, pkg_types)
Packit Service 21c75c
        for group_id in res.groups:
Packit Service 21c75c
            dnf.comps.install_or_skip(solver._group_install, group_id, pkg_types)
Packit Service 21c75c
Packit Service 21c75c
    def _mark_remove(self, patterns):
Packit Service 21c75c
        q = CompsQuery(self.base.comps, self.base.history,
Packit Service 21c75c
                       CompsQuery.GROUPS | CompsQuery.ENVIRONMENTS,
Packit Service 21c75c
                       CompsQuery.INSTALLED)
Packit Service 21c75c
        solver = self.base._build_comps_solver()
Packit Service 21c75c
        res = q.get(*patterns)
Packit Service 21c75c
        for env_id in res.environments:
Packit Service 21c75c
            assert dnf.util.is_string_type(env_id)
Packit Service 21c75c
            solver._environment_remove(env_id)
Packit Service 21c75c
        for grp_id in res.groups:
Packit Service 21c75c
            assert dnf.util.is_string_type(grp_id)
Packit Service 21c75c
            solver._group_remove(grp_id)
Packit Service 21c75c
Packit Service 21c75c
    def _mark_subcmd(self, extcmds):
Packit Service 21c75c
        if extcmds[0] in self._MARK_CMDS:
Packit Service 21c75c
            return extcmds[0], extcmds[1:]
Packit Service 21c75c
        return 'install', extcmds
Packit Service 21c75c
Packit Service 21c75c
    def _summary(self, userlist):
Packit Service 21c75c
        uservisible = 1
Packit Service 21c75c
        if len(userlist) > 0:
Packit Service 21c75c
            if userlist[0] == 'hidden':
Packit Service 21c75c
                uservisible = 0
Packit Service 21c75c
                userlist.pop(0)
Packit Service 21c75c
        if self.opts.hidden:
Packit Service 21c75c
            uservisible = 0
Packit Service 21c75c
        if not userlist:
Packit Service 21c75c
            userlist = None # Match everything...
Packit Service 21c75c
Packit Service 21c75c
        installed, available = self._group_lists(uservisible, userlist)
Packit Service 21c75c
Packit Service 21c75c
        def _out_grp(sect, num):
Packit Service 21c75c
            if not num:
Packit Service 21c75c
                return
Packit Service 21c75c
            logger.info('%s %u', sect, num)
Packit Service 21c75c
        done = 0
Packit Service 21c75c
        for group in installed:
Packit Service 21c75c
            if group.lang_only:
Packit Service 21c75c
                continue
Packit Service 21c75c
            done += 1
Packit Service 21c75c
        _out_grp(_('Installed Groups:'), done)
Packit Service 21c75c
Packit Service 21c75c
        done = 0
Packit Service 21c75c
        for group in installed:
Packit Service 21c75c
            if not group.lang_only:
Packit Service 21c75c
                continue
Packit Service 21c75c
            done += 1
Packit Service 21c75c
        _out_grp(_('Installed Language Groups:'), done)
Packit Service 21c75c
Packit Service 21c75c
        done = False
Packit Service 21c75c
        for group in available:
Packit Service 21c75c
            if group.lang_only:
Packit Service 21c75c
                continue
Packit Service 21c75c
            done += 1
Packit Service 21c75c
        _out_grp(_('Available Groups:'), done)
Packit Service 21c75c
Packit Service 21c75c
        done = False
Packit Service 21c75c
        for group in available:
Packit Service 21c75c
            if not group.lang_only:
Packit Service 21c75c
                continue
Packit Service 21c75c
            done += 1
Packit Service 21c75c
        _out_grp(_('Available Language Groups:'), done)
Packit Service 21c75c
Packit Service 21c75c
        return 0, []
Packit Service 21c75c
Packit Service 21c75c
    @staticmethod
Packit Service 21c75c
    def set_argparser(parser):
Packit Service 21c75c
        parser.add_argument('--with-optional', action='store_true',
Packit Service 21c75c
                            help=_("include optional packages from group"))
Packit Service 21c75c
        grpparser = parser.add_mutually_exclusive_group()
Packit Service 21c75c
        grpparser.add_argument('--hidden', action='store_true',
Packit Service 21c75c
                               help=_("show also hidden groups"))
Packit Service 21c75c
        grpparser.add_argument('--installed', action='store_true',
Packit Service 21c75c
                               help=_("show only installed groups"))
Packit Service 21c75c
        grpparser.add_argument('--available', action='store_true',
Packit Service 21c75c
                               help=_("show only available groups"))
Packit Service 21c75c
        grpparser.add_argument('--ids', action='store_true',
Packit Service 21c75c
                               help=_("show also ID of groups"))
Packit Service 21c75c
        parser.add_argument('subcmd', nargs='?', metavar='COMMAND',
Packit Service 21c75c
                            help=_('available subcommands: {} (default), {}').format(
Packit Service 21c75c
                                GroupCommand._GROUP_SUBCOMMANDS[0],
Packit Service 21c75c
                                ', '.join(GroupCommand._GROUP_SUBCOMMANDS[1:])))
Packit Service 21c75c
        parser.add_argument('args', nargs='*', metavar='COMMAND_ARG',
Packit Service 21c75c
                            help=_('argument for group subcommand'))
Packit Service 21c75c
Packit Service 21c75c
    def configure(self):
Packit Service 21c75c
        self._canonical()
Packit Service 21c75c
Packit Service 21c75c
        cmd = self.opts.subcmd
Packit Service 21c75c
        args = self.opts.args
Packit Service 21c75c
Packit Service 21c75c
        if cmd not in self._GROUP_SUBCOMMANDS:
Packit Service 21c75c
            logger.critical(_('Invalid groups sub-command, use: %s.'),
Packit Service 21c75c
                            ", ".join(self._GROUP_SUBCOMMANDS))
Packit Service 21c75c
            raise dnf.cli.CliError
Packit Service 21c75c
        if cmd in ('install', 'remove', 'mark', 'info') and not args:
Packit Service 21c75c
            self.cli.optparser.print_help(self)
Packit Service 21c75c
            raise dnf.cli.CliError
Packit Service 21c75c
Packit Service 21c75c
        demands = self.cli.demands
Packit Service 21c75c
        demands.sack_activation = True
Packit Service 21c75c
        if cmd in ('install', 'mark', 'remove', 'upgrade'):
Packit Service 21c75c
            demands.root_user = True
Packit Service 21c75c
            demands.resolving = True
Packit Service 21c75c
        if cmd == 'remove':
Packit Service 21c75c
            demands.allow_erasing = True
Packit Service 21c75c
            demands.available_repos = False
Packit Service 21c75c
        else:
Packit Service 21c75c
            demands.available_repos = True
Packit Service 21c75c
Packit Service 21c75c
        commands._checkEnabledRepo(self.base)
Packit Service 21c75c
Packit Service 21c75c
        if cmd in ('install', 'upgrade'):
Packit Service 21c75c
            commands._checkGPGKey(self.base, self.cli)
Packit Service 21c75c
Packit Service 21c75c
    def run(self):
Packit Service 21c75c
        cmd = self.opts.subcmd
Packit Service 21c75c
        extcmds = self.opts.args
Packit Service 21c75c
Packit Service 21c75c
        if cmd == 'summary':
Packit Service 21c75c
            return self._summary(extcmds)
Packit Service 21c75c
        if cmd == 'list':
Packit Service 21c75c
            return self._list(extcmds)
Packit Service 21c75c
        if cmd == 'info':
Packit Service 21c75c
            return self._info(extcmds)
Packit Service 21c75c
        if cmd == 'mark':
Packit Service 21c75c
            (subcmd, extcmds) = self._mark_subcmd(extcmds)
Packit Service 21c75c
            if subcmd == 'remove':
Packit Service 21c75c
                return self._mark_remove(extcmds)
Packit Service 21c75c
            else:
Packit Service 21c75c
                assert subcmd == 'install'
Packit Service 21c75c
                return self._mark_install(extcmds)
Packit Service 21c75c
Packit Service 21c75c
        if cmd == 'install':
Packit Service 21c75c
            if self.opts.with_optional:
Packit Service 21c75c
                types = tuple(self.base.conf.group_package_types + ['optional'])
Packit Service 21c75c
            else:
Packit Service 21c75c
                types = tuple(self.base.conf.group_package_types)
Packit Service 21c75c
Packit Service 21c75c
            self._remark = True
Packit Service 21c75c
            try:
Packit Service 21c75c
                return self.base.env_group_install(extcmds, types,
Packit Service 21c75c
                                                   self.base.conf.strict)
Packit Service 21c75c
            except dnf.exceptions.MarkingError as e:
Packit Service 21c75c
                msg = _('No package %s available.')
Packit Service 21c75c
                logger.info(msg, self.base.output.term.bold(e))
Packit Service 21c75c
                raise dnf.exceptions.PackagesNotAvailableError(
Packit Service 21c75c
                    _("Unable to find a mandatory group package."))
Packit Service 21c75c
        if cmd == 'upgrade':
Packit Service 21c75c
            return self.base.env_group_upgrade(extcmds)
Packit Service 21c75c
        if cmd == 'remove':
Packit Service 21c75c
            for arg in extcmds:
Packit Service 21c75c
                try:
Packit Service 21c75c
                    self.base.env_group_remove([arg])
Packit Service 21c75c
                except dnf.exceptions.Error:
Packit Service 21c75c
                    pass
Packit Service 21c75c
Packit Service 21c75c
    def run_transaction(self):
Packit Service 21c75c
        if not self._remark:
Packit Service 21c75c
            return
Packit Service 21c75c
        goal = self.base._goal
Packit Service 21c75c
        history = self.base.history
Packit Service 21c75c
        names = goal.group_members
Packit Service 21c75c
        for pkg in self.base.sack.query().installed().filterm(name=names):
Packit Service 21c75c
            reason = history.rpm.get_reason(pkg)
Packit Service 21c75c
            history.set_reason(pkg, goal.group_reason(pkg, reason))