forked from ungleich-public/cdist
Merge branch 'feature/type/__localedef' into 'master'
__localedef: Add new type to replace __locale See merge request ungleich-public/cdist!951
This commit is contained in:
commit
a234445e85
9 changed files with 354 additions and 0 deletions
1
cdist/conf/type/__locale/deprecated
Normal file
1
cdist/conf/type/__locale/deprecated
Normal file
|
@ -0,0 +1 @@
|
||||||
|
This type is deprecated. Please use __localedef instead.
|
100
cdist/conf/type/__localedef/explorer/state
Executable file
100
cdist/conf/type/__localedef/explorer/state
Executable file
|
@ -0,0 +1,100 @@
|
||||||
|
#!/bin/sh -e
|
||||||
|
#
|
||||||
|
# 2020 Dennis Camera (dennis.camera at ssrq-sds-fds.ch)
|
||||||
|
#
|
||||||
|
# This file is part of cdist.
|
||||||
|
#
|
||||||
|
# cdist is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# cdist is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with cdist. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
#
|
||||||
|
# This explorer determines if the locale is defined on the target system.
|
||||||
|
# Will print nothing on error.
|
||||||
|
#
|
||||||
|
# Possible output:
|
||||||
|
# present:
|
||||||
|
# the main locale (and possibly aliases) is present
|
||||||
|
# absent:
|
||||||
|
# neither the main locale nor any aliases are present
|
||||||
|
# alias-present:
|
||||||
|
# the main locale is absent, but at least one of its aliases is present
|
||||||
|
#
|
||||||
|
|
||||||
|
# Hardcoded, create a pull request in case it is at another location for
|
||||||
|
# some other distro. (cf. gencode-remote)
|
||||||
|
aliasfile='/usr/share/locale/locale.alias'
|
||||||
|
|
||||||
|
command -v locale >/dev/null 2>&1 || exit 0
|
||||||
|
|
||||||
|
locales=$(locale -a)
|
||||||
|
|
||||||
|
parse_locale() {
|
||||||
|
# This function will split locales into their parts. Locale strings are
|
||||||
|
# usually of the form: [language[_territory][.codeset][@modifier]]
|
||||||
|
# For simplicity, language and territory are not separated by this function.
|
||||||
|
# Old Linux systems were also using "english" or "german" as locale strings.
|
||||||
|
# Usage: parse_locale locale_str lang_var codeset_var modifier_var
|
||||||
|
eval "${2:?}"="$(expr "$1" : '\([^.@]*\)')"
|
||||||
|
eval "${3:?}"="$(expr "$1" : '[^.]*\.\([^@]*\)')"
|
||||||
|
eval "${4:?}"="$(expr "$1" : '.*@\(.*\)$')"
|
||||||
|
}
|
||||||
|
|
||||||
|
format_locale() {
|
||||||
|
# Usage: format_locale language codeset modifier
|
||||||
|
printf '%s' "$1"
|
||||||
|
test -z "$2" || printf '.%s' "$2"
|
||||||
|
test -z "$3" || printf '@%s' "$3"
|
||||||
|
printf '\n'
|
||||||
|
}
|
||||||
|
|
||||||
|
gnu_normalize_codeset() {
|
||||||
|
# reimplementation of glibc/locale/programs/localedef.c normalize_codeset()
|
||||||
|
echo "$*" | tr '[:upper:]' '[:lower:]' | tr -cd '[:alnum:]'
|
||||||
|
}
|
||||||
|
|
||||||
|
locale_available() (
|
||||||
|
echo "${locales}" | grep -qxF "$1" || {
|
||||||
|
# glibc uses "normalized" locale names in archives.
|
||||||
|
# If a locale is stored in an archive, the normalized name will be
|
||||||
|
# printed by locale, so that needs to be checked, too.
|
||||||
|
localename=$(
|
||||||
|
parse_locale "$1" _lang _codeset _modifier \
|
||||||
|
&& format_locale "${_lang:?}" "$(gnu_normalize_codeset "${_codeset?}")" \
|
||||||
|
"${_modifier?}")
|
||||||
|
echo "${locales}" | grep -qxF "${localename}"
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
if locale_available "${__object_id:?}"
|
||||||
|
then
|
||||||
|
echo present
|
||||||
|
else
|
||||||
|
# NOTE: locale.alias can be symlinked.
|
||||||
|
if test -e "${aliasfile}"
|
||||||
|
then
|
||||||
|
# Check if one of the aliases of the locale is defined
|
||||||
|
baselocale=$(
|
||||||
|
parse_locale "${__object_id:?}" _lang _codeset _modifiers \
|
||||||
|
&& format_locale "${_lang}" "${_codeset}")
|
||||||
|
while read -r _alias _localename
|
||||||
|
do
|
||||||
|
if test "${_localename}" = "${baselocale}" \
|
||||||
|
&& echo "${locales}" | grep -qxF "${_alias}"
|
||||||
|
then
|
||||||
|
echo alias-present
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
done <"${aliasfile}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo absent
|
||||||
|
fi
|
5
cdist/conf/type/__localedef/files/lib/glibc.sh
Normal file
5
cdist/conf/type/__localedef/files/lib/glibc.sh
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
# -*- mode: sh; indent-tabs-mode: t -*-
|
||||||
|
|
||||||
|
gnu_normalize_codeset() {
|
||||||
|
echo "$*" | tr -cd '[:alnum:]' | tr '[:upper:]' '[:lower:]'
|
||||||
|
}
|
20
cdist/conf/type/__localedef/files/lib/locale.sh
Normal file
20
cdist/conf/type/__localedef/files/lib/locale.sh
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
# -*- mode: sh; indent-tabs-mode:t -*-
|
||||||
|
|
||||||
|
parse_locale() {
|
||||||
|
# This function will split locales into their parts. Locale strings are
|
||||||
|
# usually of the form: [language[_territory][.codeset][@modifier]]
|
||||||
|
# For simplicity, language and territory are not separated by this function.
|
||||||
|
# Old Linux systems were also using "english" or "german" as locale strings.
|
||||||
|
# Usage: parse_locale locale_str lang_var codeset_var modifier_var
|
||||||
|
eval "${2:?}"="$(expr "$1" : '\([^.@]*\)')"
|
||||||
|
eval "${3:?}"="$(expr "$1" : '[^.]*\.\([^@]*\)')"
|
||||||
|
eval "${4:?}"="$(expr "$1" : '.*@\(.*\)$')"
|
||||||
|
}
|
||||||
|
|
||||||
|
format_locale() {
|
||||||
|
# Usage: format_locale language codeset modifier
|
||||||
|
printf '%s' "$1"
|
||||||
|
test -z "$2" || printf '.%s' "$2"
|
||||||
|
test -z "$3" || printf '@%s' "$3"
|
||||||
|
printf '\n'
|
||||||
|
}
|
136
cdist/conf/type/__localedef/gencode-remote
Executable file
136
cdist/conf/type/__localedef/gencode-remote
Executable file
|
@ -0,0 +1,136 @@
|
||||||
|
#!/bin/sh -e
|
||||||
|
#
|
||||||
|
# 2013-2019 Nico Schottelius (nico-cdist at schottelius.org)
|
||||||
|
# 2020 Dennis Camera (dennis.camera at ssrq-sds-fds.ch)
|
||||||
|
#
|
||||||
|
# This file is part of cdist.
|
||||||
|
#
|
||||||
|
# cdist is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# cdist is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with cdist. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
#
|
||||||
|
# Manage system locales using localedef(1).
|
||||||
|
#
|
||||||
|
|
||||||
|
# shellcheck source=cdist/conf/type/__localedef/files/lib/locale.sh
|
||||||
|
. "${__type:?}/files/lib/locale.sh"
|
||||||
|
# shellcheck source=cdist/conf/type/__localedef/files/lib/glibc.sh
|
||||||
|
. "${__type:?}/files/lib/glibc.sh"
|
||||||
|
|
||||||
|
state_is=$(cat "${__object:?}/explorer/state")
|
||||||
|
state_should=$(cat "${__object:?}/parameter/state")
|
||||||
|
|
||||||
|
test "${state_should}" = 'present' -o "${state_should}" = 'absent' || {
|
||||||
|
printf 'Invalid state: %s\n' "${state_should}" >&2
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# NOTE: If state explorer fails (e.g. locale(1) missing), the following check
|
||||||
|
# will always fail and let definition/removal run.
|
||||||
|
if test "${state_is}" = "${state_should}"
|
||||||
|
then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
locale=${__object_id:?}
|
||||||
|
os=$(cat "${__global:?}/explorer/os")
|
||||||
|
|
||||||
|
if expr "${locale}" : '.*/' >/dev/null
|
||||||
|
then
|
||||||
|
printf 'Paths as locales are not supported.\n' >&2
|
||||||
|
printf '__object_id is: %s\n' "${locale}" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
: "${lang=}" "${codeset=}" "${modifier=}" # declare variables for shellcheck
|
||||||
|
parse_locale "${locale}" lang codeset modifier
|
||||||
|
|
||||||
|
|
||||||
|
case ${os}
|
||||||
|
in
|
||||||
|
(alpine|openwrt)
|
||||||
|
printf '%s does not support locales.\n' "${os}" >&2
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
(archlinux|debian|devuan|ubuntu|suse|centos|fedora|redhat|scientific)
|
||||||
|
# FIXME: The code below only works for glibc-based installations.
|
||||||
|
|
||||||
|
# NOTE: Hardcoded, create a pull request in case it is at another
|
||||||
|
# location for some opther distro.
|
||||||
|
# NOTE: locale.alias can be symlinked (e.g. Debian)
|
||||||
|
aliasfile='/usr/share/locale/locale.alias'
|
||||||
|
|
||||||
|
case ${state_should}
|
||||||
|
in
|
||||||
|
(present)
|
||||||
|
input=$(format_locale "${lang}" '' "${modifier}")
|
||||||
|
cat <<-EOF
|
||||||
|
set --
|
||||||
|
if test -e '${aliasfile}'
|
||||||
|
then
|
||||||
|
set -- -A '${aliasfile}'
|
||||||
|
fi
|
||||||
|
|
||||||
|
localedef -i '${input}' -f '${codeset}' "\$@" '${locale}'
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
(absent)
|
||||||
|
main_localename=$(format_locale "${lang}" "$(gnu_normalize_codeset "${codeset}")" "${modifier}")
|
||||||
|
|
||||||
|
cat <<-EOF
|
||||||
|
while read -r _alias _localename
|
||||||
|
do
|
||||||
|
if test "\${_localename}" = '$(format_locale "${lang}" "${codeset}")'
|
||||||
|
then
|
||||||
|
localedef --delete-from-archive "\${_alias}"
|
||||||
|
fi
|
||||||
|
done <'${aliasfile}'
|
||||||
|
EOF
|
||||||
|
|
||||||
|
if test "${state_is}" = present
|
||||||
|
then
|
||||||
|
printf "localedef --delete-from-archive '%s'\n" "${main_localename}"
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
;;
|
||||||
|
(freebsd)
|
||||||
|
case ${state_should}
|
||||||
|
in
|
||||||
|
(present)
|
||||||
|
if expr "$(grep -oe '^[0-9]*' "${__global:?}/explorer/os_version")" '>=' 11 >/dev/null
|
||||||
|
then
|
||||||
|
# localedef(1) is available with FreeBSD >= 11
|
||||||
|
printf "localedef -i '%s' -f '%s' '%s'\n" "${input}" "${codeset}" "${locale}"
|
||||||
|
else
|
||||||
|
printf 'localedef(1) was added to FreeBSD starting with version 11.\n' >&2
|
||||||
|
printf 'Please upgrade your FreeBSD installation to use %s.\n' "${__type##*/}" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
(absent)
|
||||||
|
printf "rm -R '/usr/share/locale/%s'\n" "${locale}"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
;;
|
||||||
|
(netbsd|openbsd)
|
||||||
|
# NetBSD/OpenBSD are missing localedef(1).
|
||||||
|
# We also do not delete defined locales because they can't be recreated.
|
||||||
|
echo "${os} is lacking localedef(1). Locale management unavailable." >&2
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
(*)
|
||||||
|
echo "Your operating system (${os}) is currently not supported by this type (${__type##*/})." >&2
|
||||||
|
echo "Please contribute an implementation for it if you can." >&2
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
esac
|
60
cdist/conf/type/__localedef/man.rst
Normal file
60
cdist/conf/type/__localedef/man.rst
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
cdist-type__localedef(7)
|
||||||
|
========================
|
||||||
|
|
||||||
|
NAME
|
||||||
|
----
|
||||||
|
cdist-type__localedef - Define and remove system locales
|
||||||
|
|
||||||
|
|
||||||
|
DESCRIPTION
|
||||||
|
-----------
|
||||||
|
This cdist type allows you to define locales on the system using
|
||||||
|
:strong:`localedef`\ (1) or remove them.
|
||||||
|
On systems that don't support definition of new locales, the type will raise an
|
||||||
|
error.
|
||||||
|
|
||||||
|
**NB:** This type respects the glibc ``locale.alias`` file,
|
||||||
|
i.e. it defines alias locales or deletes aliases of a locale when it is removed.
|
||||||
|
It is not possible, however, to use alias names to define locales or only remove
|
||||||
|
certain aliases of a locale.
|
||||||
|
|
||||||
|
|
||||||
|
OPTIONAL PARAMETERS
|
||||||
|
-------------------
|
||||||
|
state
|
||||||
|
``present`` or ``absent``. Defaults to ``present``.
|
||||||
|
|
||||||
|
|
||||||
|
EXAMPLES
|
||||||
|
--------
|
||||||
|
|
||||||
|
.. code-block:: sh
|
||||||
|
|
||||||
|
# Add locale de_CH.UTF-8
|
||||||
|
__localedef de_CH.UTF-8
|
||||||
|
|
||||||
|
# Same as above, but more explicit
|
||||||
|
__localedef de_CH.UTF-8 --state present
|
||||||
|
|
||||||
|
# Remove colourful British English
|
||||||
|
__localedef en_GB.UTF-8 --state absent
|
||||||
|
|
||||||
|
|
||||||
|
SEE ALSO
|
||||||
|
--------
|
||||||
|
:strong:`locale`\ (1),
|
||||||
|
:strong:`localedef`\ (1),
|
||||||
|
:strong:`cdist-type__locale_system`\ (7)
|
||||||
|
|
||||||
|
|
||||||
|
AUTHORS
|
||||||
|
-------
|
||||||
|
| Dennis Camera <dennis.camera--@--ssrq-sds-fds.ch>
|
||||||
|
| Nico Schottelius <nico-cdist--@--schottelius.org>
|
||||||
|
|
||||||
|
|
||||||
|
COPYING
|
||||||
|
-------
|
||||||
|
Copyright \(C) 2013-2019 Nico Schottelius, 2020 Dennis Camera. Free use of this
|
||||||
|
software is granted under the terms of the GNU General Public License version 3
|
||||||
|
or later (GPLv3+).
|
30
cdist/conf/type/__localedef/manifest
Executable file
30
cdist/conf/type/__localedef/manifest
Executable file
|
@ -0,0 +1,30 @@
|
||||||
|
#!/bin/sh -e
|
||||||
|
#
|
||||||
|
# 2013-2019 Nico Schottelius (nico-cdist at schottelius.org)
|
||||||
|
# 2015 David Hürlimann (david at ungleich.ch)
|
||||||
|
# 2020 Dennis Camera (dennis.camera at ssrq-sds-fds.ch)
|
||||||
|
#
|
||||||
|
# This file is part of cdist.
|
||||||
|
#
|
||||||
|
# cdist is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# cdist is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with cdist. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
#
|
||||||
|
# Install required packages.
|
||||||
|
#
|
||||||
|
|
||||||
|
case $(cat "${__global:?}/explorer/os")
|
||||||
|
in
|
||||||
|
(debian|devuan)
|
||||||
|
__package_apt locales --state present
|
||||||
|
;;
|
||||||
|
esac
|
1
cdist/conf/type/__localedef/parameter/default/state
Normal file
1
cdist/conf/type/__localedef/parameter/default/state
Normal file
|
@ -0,0 +1 @@
|
||||||
|
present
|
1
cdist/conf/type/__localedef/parameter/optional
Normal file
1
cdist/conf/type/__localedef/parameter/optional
Normal file
|
@ -0,0 +1 @@
|
||||||
|
state
|
Loading…
Reference in a new issue