mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 21:14:00 -07:00 
			
		
		
		
	
		
			Some checks are pending
		
		
	
	EOL CI / EOL Sanity (Ⓐ2.17) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.10) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.12) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.7) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/3/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/3/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/3/) (push) Waiting to run
				
			nox / Run extra sanity tests (push) Waiting to run
				
			* Adjust all __future__ imports: for i in $(grep -REl "__future__.*absolute_import" plugins/ tests/); do sed -e 's/from __future__ import .*/from __future__ import annotations/g' -i $i; done * Remove all UTF-8 encoding specifications for Python source files: for i in $(grep -REl '[-][*]- coding: utf-8 -[*]-' plugins/ tests/); do sed -e '/^# -\*- coding: utf-8 -\*-/d' -i $i; done * Remove __metaclass__ = type: for i in $(grep -REl '__metaclass__ = type' plugins/ tests/); do sed -e '/^__metaclass__ = type/d' -i $i; done
		
			
				
	
	
		
			283 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			283 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/python
 | |
| 
 | |
| # Copyright (c) Ansible project
 | |
| # 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 annotations
 | |
| 
 | |
| DOCUMENTATION = r"""
 | |
| module: locale_gen
 | |
| short_description: Creates or removes locales
 | |
| description:
 | |
|   - Manages locales in Debian and Ubuntu systems.
 | |
| author:
 | |
|   - Augustus Kling (@AugustusKling)
 | |
| extends_documentation_fragment:
 | |
|   - community.general.attributes
 | |
| attributes:
 | |
|   check_mode:
 | |
|     support: full
 | |
|   diff_mode:
 | |
|     support: none
 | |
| options:
 | |
|   name:
 | |
|     type: list
 | |
|     elements: str
 | |
|     description:
 | |
|       - Name and encoding of the locales, such as V(en_GB.UTF-8).
 | |
|       - Before community.general 9.3.0, this was a string. Using a string still works.
 | |
|     required: true
 | |
|   state:
 | |
|     type: str
 | |
|     description:
 | |
|       - Whether the locales shall be present.
 | |
|     choices: [absent, present]
 | |
|     default: present
 | |
| notes:
 | |
|   - If C(/etc/locale.gen) exists, the module assumes to be using the B(glibc) mechanism, else if C(/var/lib/locales/supported.d/)
 | |
|     exists it assumes to be using the B(ubuntu_legacy) mechanism, else it raises an error.
 | |
|   - When using glibc mechanism, it manages locales by editing C(/etc/locale.gen) and running C(locale-gen).
 | |
|   - When using ubuntu_legacy mechanism, it manages locales by editing C(/var/lib/locales/supported.d/local) and then running
 | |
|     C(locale-gen).
 | |
|   - Please note that the code path that uses ubuntu_legacy mechanism has not been tested for a while, because Ubuntu is already
 | |
|     using the glibc mechanism. There is no support for that, given our inability to test it. Therefore, that mechanism is
 | |
|     B(deprecated) and will be removed in community.general 13.0.0.
 | |
|   - Currently the module is B(only supported for Debian and Ubuntu) systems.
 | |
|   - This module requires the package C(locales) installed in Debian and Ubuntu systems.
 | |
| """
 | |
| 
 | |
| EXAMPLES = r"""
 | |
| - name: Ensure a locale exists
 | |
|   community.general.locale_gen:
 | |
|     name: de_CH.UTF-8
 | |
|     state: present
 | |
| 
 | |
| - name: Ensure multiple locales exist
 | |
|   community.general.locale_gen:
 | |
|     name:
 | |
|       - en_GB.UTF-8
 | |
|       - nl_NL.UTF-8
 | |
|     state: present
 | |
| """
 | |
| 
 | |
| RETURN = r"""
 | |
| mechanism:
 | |
|   description: Mechanism used to deploy the locales.
 | |
|   type: str
 | |
|   choices:
 | |
|     - glibc
 | |
|     - ubuntu_legacy
 | |
|   returned: success
 | |
|   sample: glibc
 | |
|   version_added: 10.2.0
 | |
| """
 | |
| 
 | |
| import os
 | |
| import re
 | |
| 
 | |
| from ansible_collections.community.general.plugins.module_utils.module_helper import StateModuleHelper
 | |
| from ansible_collections.community.general.plugins.module_utils.mh.deco import check_mode_skip
 | |
| 
 | |
| from ansible_collections.community.general.plugins.module_utils.locale_gen import locale_runner, locale_gen_runner
 | |
| 
 | |
| 
 | |
| ETC_LOCALE_GEN = "/etc/locale.gen"
 | |
| VAR_LIB_LOCALES = "/var/lib/locales/supported.d"
 | |
| VAR_LIB_LOCALES_LOCAL = os.path.join(VAR_LIB_LOCALES, "local")
 | |
| SUPPORTED_LOCALES = "/usr/share/i18n/SUPPORTED"
 | |
| LOCALE_NORMALIZATION = {
 | |
|     ".utf8": ".UTF-8",
 | |
|     ".eucjp": ".EUC-JP",
 | |
|     ".iso885915": ".ISO-8859-15",
 | |
|     ".cp1251": ".CP1251",
 | |
|     ".koi8r": ".KOI8-R",
 | |
|     ".armscii8": ".ARMSCII-8",
 | |
|     ".euckr": ".EUC-KR",
 | |
|     ".gbk": ".GBK",
 | |
|     ".gb18030": ".GB18030",
 | |
|     ".euctw": ".EUC-TW",
 | |
| }
 | |
| 
 | |
| 
 | |
| class LocaleGen(StateModuleHelper):
 | |
|     output_params = ["name"]
 | |
|     module = dict(
 | |
|         argument_spec=dict(
 | |
|             name=dict(type="list", elements="str", required=True),
 | |
|             state=dict(type='str', default='present', choices=['absent', 'present']),
 | |
|         ),
 | |
|         supports_check_mode=True,
 | |
|     )
 | |
| 
 | |
|     def __init_module__(self):
 | |
|         self.MECHANISMS = dict(
 | |
|             ubuntu_legacy=dict(
 | |
|                 available=SUPPORTED_LOCALES,
 | |
|                 apply_change=self.apply_change_ubuntu_legacy,
 | |
|             ),
 | |
|             glibc=dict(
 | |
|                 available=SUPPORTED_LOCALES,
 | |
|                 apply_change=self.apply_change_glibc,
 | |
|             ),
 | |
|         )
 | |
| 
 | |
|         if os.path.exists(ETC_LOCALE_GEN):
 | |
|             self.vars.ubuntu_mode = False
 | |
|             self.vars.mechanism = "glibc"
 | |
|         elif os.path.exists(VAR_LIB_LOCALES):
 | |
|             self.vars.ubuntu_mode = True
 | |
|             self.vars.mechanism = "ubuntu_legacy"
 | |
|             self.module.deprecate(
 | |
|                 "On this machine mechanism=ubuntu_legacy is used. This mechanism is deprecated and will be removed from"
 | |
|                 " in community.general 13.0.0. If you see this message on a modern Debian or Ubuntu version,"
 | |
|                 " please create an issue in the community.general repository",
 | |
|                 version="13.0.0", collection_name="community.general"
 | |
|             )
 | |
|         else:
 | |
|             self.do_raise('{0} and {1} are missing. Is the package "locales" installed?'.format(
 | |
|                 VAR_LIB_LOCALES, ETC_LOCALE_GEN
 | |
|             ))
 | |
| 
 | |
|         self.runner = locale_runner(self.module)
 | |
| 
 | |
|         self.assert_available()
 | |
|         self.vars.set("is_present", self.is_present(), output=False)
 | |
|         self.vars.set("state_tracking", self._state_name(self.vars.is_present), output=False, change=True)
 | |
| 
 | |
|     def __quit_module__(self):
 | |
|         self.vars.state_tracking = self._state_name(self.is_present())
 | |
| 
 | |
|     @staticmethod
 | |
|     def _state_name(present):
 | |
|         return "present" if present else "absent"
 | |
| 
 | |
|     def assert_available(self):
 | |
|         """Check if the given locales are available on the system. This is done by
 | |
|         checking either :
 | |
|         * if the locale is present in /etc/locales.gen
 | |
|         * or if the locale is present in /usr/share/i18n/SUPPORTED"""
 | |
|         regexp = r'^\s*#?\s*(?P<locale>\S+[\._\S]+) (?P<charset>\S+)\s*$'
 | |
|         locales_available = self.MECHANISMS[self.vars.mechanism]["available"]
 | |
| 
 | |
|         re_compiled = re.compile(regexp)
 | |
|         with open(locales_available, 'r') as fd:
 | |
|             lines = fd.readlines()
 | |
|         res = [re_compiled.match(line) for line in lines]
 | |
|         self.vars.set("available_lines", lines, verbosity=4)
 | |
| 
 | |
|         locales_not_found = []
 | |
|         for locale in self.vars.name:
 | |
|             # Check if the locale is not found in any of the matches
 | |
|             if not any(match and match.group("locale") == locale for match in res):
 | |
|                 locales_not_found.append(locale)
 | |
| 
 | |
|         # locale may be installed but not listed in the file, for example C.UTF-8 in some systems
 | |
|         locales_not_found = self.locale_get_not_present(locales_not_found)
 | |
| 
 | |
|         if locales_not_found:
 | |
|             self.do_raise("The following locales you have entered are not available on your system: {0}".format(', '.join(locales_not_found)))
 | |
| 
 | |
|     def is_present(self):
 | |
|         return not self.locale_get_not_present(self.vars.name)
 | |
| 
 | |
|     def locale_get_not_present(self, locales):
 | |
|         runner = locale_runner(self.module)
 | |
|         with runner() as ctx:
 | |
|             rc, out, err = ctx.run()
 | |
|             if self.verbosity >= 4:
 | |
|                 self.vars.locale_run_info = ctx.run_info
 | |
| 
 | |
|         not_found = []
 | |
|         for locale in locales:
 | |
|             if not any(self.fix_case(locale) == self.fix_case(line) for line in out.splitlines()):
 | |
|                 not_found.append(locale)
 | |
| 
 | |
|         return not_found
 | |
| 
 | |
|     def fix_case(self, name):
 | |
|         """locale -a might return the encoding in either lower or upper case.
 | |
|         Passing through this function makes them uniform for comparisons."""
 | |
|         for s, r in LOCALE_NORMALIZATION.items():
 | |
|             name = name.replace(s, r)
 | |
|         return name
 | |
| 
 | |
|     def set_locale_glibc(self, names, enabled=True):
 | |
|         """ Sets the state of the locale. Defaults to enabled. """
 | |
|         with open(ETC_LOCALE_GEN, 'r') as fr:
 | |
|             lines = fr.readlines()
 | |
| 
 | |
|         locale_regexes = []
 | |
| 
 | |
|         for name in names:
 | |
|             search_string = r'^#?\s*%s (?P<charset>.+)' % re.escape(name)
 | |
|             if enabled:
 | |
|                 new_string = r'%s \g<charset>' % (name)
 | |
|             else:
 | |
|                 new_string = r'# %s \g<charset>' % (name)
 | |
|             re_search = re.compile(search_string)
 | |
|             locale_regexes.append([re_search, new_string])
 | |
| 
 | |
|         for i in range(len(lines)):
 | |
|             for [search, replace] in locale_regexes:
 | |
|                 lines[i] = search.sub(replace, lines[i])
 | |
| 
 | |
|         # Write the modified content back to the file
 | |
|         with open(ETC_LOCALE_GEN, 'w') as fw:
 | |
|             fw.writelines(lines)
 | |
| 
 | |
|     def apply_change_glibc(self, targetState, names):
 | |
|         """Create or remove locale.
 | |
| 
 | |
|         Keyword arguments:
 | |
|         targetState -- Desired state, either present or absent.
 | |
|         names -- Names list including encoding such as de_CH.UTF-8.
 | |
|         """
 | |
| 
 | |
|         self.set_locale_glibc(names, enabled=(targetState == "present"))
 | |
| 
 | |
|         runner = locale_gen_runner(self.module)
 | |
|         with runner() as ctx:
 | |
|             ctx.run()
 | |
| 
 | |
|     def apply_change_ubuntu_legacy(self, targetState, names):
 | |
|         """Create or remove locale.
 | |
| 
 | |
|         Keyword arguments:
 | |
|         targetState -- Desired state, either present or absent.
 | |
|         names -- Name list including encoding such as de_CH.UTF-8.
 | |
|         """
 | |
|         runner = locale_gen_runner(self.module)
 | |
| 
 | |
|         if targetState == "present":
 | |
|             # Create locale.
 | |
|             # Ubuntu's patched locale-gen automatically adds the new locale to /var/lib/locales/supported.d/local
 | |
|             with runner() as ctx:
 | |
|                 ctx.run()
 | |
|         else:
 | |
|             # Delete locale involves discarding the locale from /var/lib/locales/supported.d/local and regenerating all locales.
 | |
|             with open(VAR_LIB_LOCALES_LOCAL, "r") as fr:
 | |
|                 content = fr.readlines()
 | |
|             with open(VAR_LIB_LOCALES_LOCAL, "w") as fw:
 | |
|                 for line in content:
 | |
|                     locale, charset = line.split(' ')
 | |
|                     if locale not in names:
 | |
|                         fw.write(line)
 | |
|             # Purge locales and regenerate.
 | |
|             # Please provide a patch if you know how to avoid regenerating the locales to keep!
 | |
|             with runner("purge") as ctx:
 | |
|                 ctx.run()
 | |
| 
 | |
|     @check_mode_skip
 | |
|     def __state_fallback__(self):
 | |
|         if self.vars.state_tracking == self.vars.state:
 | |
|             return
 | |
|         self.MECHANISMS[self.vars.mechanism]["apply_change"](self.vars.state, self.vars.name)
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     LocaleGen.execute()
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     main()
 |