#!/usr/bin/python # -*- coding: utf-8 -*- # Copyright (c) 2018, Sean Myers # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) # SPDX-License-Identifier: GPL-3.0-or-later from __future__ import (absolute_import, division, print_function) __metaclass__ = type DOCUMENTATION = ''' --- module: rhsm_release short_description: Set or Unset RHSM Release version description: - Sets or unsets the release version used by RHSM repositories. notes: - This module will fail on an unregistered system. Use the M(community.general.redhat_subscription) module to register a system prior to setting the RHSM release. - It is possible to interact with C(subscription-manager) only as root, so root permissions are required to successfully run this module. requirements: - Red Hat Enterprise Linux 6+ with subscription-manager installed extends_documentation_fragment: - community.general.attributes attributes: check_mode: support: full diff_mode: support: none options: release: description: - RHSM release version to use. - To unset either pass V(null) for this option, or omit this option. type: str author: - Sean Myers (@seandst) ''' EXAMPLES = ''' # Set release version to 7.1 - name: Set RHSM release version community.general.rhsm_release: release: "7.1" # Set release version to 6Server - name: Set RHSM release version community.general.rhsm_release: release: "6Server" # Unset release version - name: Unset RHSM release release community.general.rhsm_release: release: null ''' RETURN = ''' current_release: description: The current RHSM release version value returned: success type: str ''' from ansible.module_utils.basic import AnsibleModule import os import re # Matches release-like values such as 7.2, 5.10, 6Server, 8 # but rejects unlikely values, like 100Server, 1.100, 7server etc. release_matcher = re.compile(r'\b\d{1,2}(?:\.\d{1,2}|Server|Client|Workstation|)\b') def _sm_release(module, *args): # pass args to s-m release, e.g. _sm_release(module, '--set', '0.1') becomes # "subscription-manager release --set 0.1" sm_bin = module.get_bin_path('subscription-manager', required=True) cmd = [sm_bin, 'release'] + list(args) # delegate nonzero rc handling to run_command return module.run_command(cmd, check_rc=True, expand_user_and_vars=False) def get_release(module): # Get the current release version, or None if release unset rc, out, err = _sm_release(module, '--show') try: match = release_matcher.findall(out)[0] except IndexError: # 0'th index did not exist; no matches match = None return match def set_release(module, release): # Set current release version, or unset if release is None if release is None: args = ('--unset',) else: args = ('--set', release) return _sm_release(module, *args) def main(): module = AnsibleModule( argument_spec=dict( release=dict(type='str'), ), supports_check_mode=True ) if os.getuid() != 0: module.fail_json( msg="Interacting with subscription-manager requires root permissions ('become: true')" ) target_release = module.params['release'] # sanity check: the target release at least looks like a valid release if target_release and not release_matcher.findall(target_release): module.fail_json(msg='"{0}" does not appear to be a valid release.'.format(target_release)) # Will fail with useful error from s-m if system not subscribed current_release = get_release(module) changed = (target_release != current_release) if not module.check_mode and changed: set_release(module, target_release) # If setting the release fails, then a fail_json would have exited with # the s-m error, e.g. "No releases match '7.20'...". If not, then the # current release is now set to the target release (job's done) current_release = target_release module.exit_json(current_release=current_release, changed=changed) if __name__ == '__main__': main()