mirror of
https://github.com/ansible-collections/community.general.git
synced 2025-04-26 20:31:27 -07:00
* made adding package managers easier added portage support * moar pkg mgrs and moar info - added 'pkg' pkg manager (freebsd) - added pip - more apt info * updated clgo * Updates from feedback Co-Authored-By: bcoca <bcoca@users.noreply.github.com> * incorporated more feedback and added docstrings * moar from feedback - made manager list dynamic and names based on class - better not found msg - made abstract metaclass again - test is now init exception - module to global - better dedupe comments * more targetted errors/warnings * added strategy, reordered to conserve priority * rpm > apt * move break to top * fix trate * piping it * lines and meta * refactored common functions - moved pip into it's own module - cleaned up base clases - ensure 'lower' match in package_facts * missing license * avoid facts * update clog * addressed feedback * fix clog * cleanup * upd * removed pip as that was removed * renamed cpan * added a single line since 2 lines are needed to be readabnle instead of just 1 line, it is a huge problem otherwise * fix internal ref * not intended in this round * updated as per fb
83 lines
2.5 KiB
Python
83 lines
2.5 KiB
Python
# (c) 2018, Ansible Project
|
|
# Simplified BSD License (see licenses/simplified_bsd.txt or https://opensource.org/licenses/BSD-2-Clause)
|
|
|
|
from __future__ import absolute_import, division, print_function
|
|
__metaclass__ = type
|
|
|
|
from abc import ABCMeta, abstractmethod
|
|
|
|
from ansible.module_utils.six import with_metaclass
|
|
from ansible.module_utils.basic import get_all_subclasses
|
|
from ansible.module_utils.common.process import get_bin_path
|
|
|
|
|
|
def get_all_pkg_managers():
|
|
|
|
return dict([(obj.__name__.lower(), obj) for obj in get_all_subclasses(PkgMgr) if obj not in (CLIMgr, LibMgr)])
|
|
|
|
|
|
class PkgMgr(with_metaclass(ABCMeta, object)):
|
|
|
|
@abstractmethod
|
|
def is_available(self):
|
|
# This method is supposed to return True/False if the package manager is currently installed/usable
|
|
# It can also 'prep' the required systems in the process of detecting availability
|
|
pass
|
|
|
|
@abstractmethod
|
|
def list_installed(self):
|
|
# This method should return a list of installed packages, each list item will be passed to get_package_details
|
|
pass
|
|
|
|
@abstractmethod
|
|
def get_package_details(self, package):
|
|
# This takes a 'package' item and returns a dictionary with the package information, name and version are minimal requirements
|
|
pass
|
|
|
|
def get_packages(self):
|
|
# Take all of the above and return a dictionary of lists of dictionaries (package = list of installed versions)
|
|
|
|
installed_packages = {}
|
|
for package in self.list_installed():
|
|
package_details = self.get_package_details(package)
|
|
if 'source' not in package_details:
|
|
package_details['source'] = self.__class__.__name__.lower()
|
|
name = package_details['name']
|
|
if name not in installed_packages:
|
|
installed_packages[name] = [package_details]
|
|
else:
|
|
installed_packages[name].append(package_details)
|
|
return installed_packages
|
|
|
|
|
|
class LibMgr(PkgMgr):
|
|
|
|
LIB = None
|
|
|
|
def __init__(self):
|
|
|
|
self._lib = None
|
|
super(LibMgr, self).__init__()
|
|
|
|
def is_available(self):
|
|
found = False
|
|
try:
|
|
self._lib = __import__(self.LIB)
|
|
found = True
|
|
except ImportError:
|
|
pass
|
|
return found
|
|
|
|
|
|
class CLIMgr(PkgMgr):
|
|
|
|
CLI = None
|
|
|
|
def __init__(self):
|
|
|
|
self._cli = None
|
|
super(CLIMgr, self).__init__()
|
|
|
|
def is_available(self):
|
|
self._cli = get_bin_path(self.CLI, False)
|
|
return bool(self._cli)
|