Created
December 10, 2021 07:28
-
-
Save jwkenney/55f0b540b84b575ee47cb0237c039afc to your computer and use it in GitHub Desktop.
Custom Ansible fact for local Linux users
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/usr/bin/env bash | |
# Requires Bash v4+ | |
# Gathers Ansible facts for local Linux users from /etc/passwd, and optionally /etc/shadow. | |
# Place this under /etc/ansible/facts.d/ on your remote hosts, | |
# and make it executable. Ansible will save user info under the 'ansible_local' fact, | |
# whenever a playbook gathers facts. | |
# You can toggle whether to also include password expiration data from /etc/shadow. | |
# THIS REQUIRES ROOT PRIVILEGES TO WORK, or else empty / '0' results will be returned. | |
gather_shadow=true | |
readarray -t passwd_list < <(getent passwd) | |
readarray -t passwd_users < <(getent passwd | cut -d: -f1) | |
[ -r /etc/shadow ] && file_readable="true" || file_readable="false" | |
let "lastitem = ${#passwd_list[@]} - 1" | |
if ${gather_shadow}; then | |
echo "{" | |
for index in "${!passwd_list[@]}"; do | |
# Parse the standard passwd data | |
IFS=: read -r user pass uid gid description homedir shell <<< "${passwd_list[$index]}" | |
# If user is not found in shadow DB, or if we don't have permissions to read, we return empty data. | |
shadow_data=$(getent shadow "${user}" || echo '::::::::') | |
IFS=: read -r shadowuser pwhash lastchg minage maxage warn inactive expires misc <<< "${shadow_data}" | |
# 'present' is our clue to the user that we couldn't fetch the data from shadow, whatever the reason. | |
[ -n "${shadowuser}" ] && present="true" || present="false" | |
# Last element in the list should skip the comma at the end | |
[ ${index} -eq ${lastitem} ] && delim='' || delim=',' | |
# Don't mess with the heredoc EOF spacing or Bash will cry | |
cat << EOF | |
"${user}": { | |
"name": "${user}", | |
"uid": ${uid}, | |
"gid": ${gid}, | |
"description": "${description}", | |
"home": "${homedir}", | |
"shell": "${shell}", | |
"shadow": { | |
"db_readable": ${file_readable}, | |
"user_found": ${present}, | |
"last_change": ${lastchg:-0}, | |
"min_age": ${minage:-0}, | |
"max_age": ${maxage:-0}, | |
"warn": ${warn:-0}, | |
"inactive": ${inactive:-0}, | |
"expires": ${expires:-0} | |
} | |
}${delim} | |
EOF | |
done | |
echo "}" | |
else | |
echo "{" | |
for index in "${!passwd_list[@]}"; do | |
IFS=: read -r user pass uid gid description homedir shell <<< "${passwd_list[$index]}" | |
[ ${index} -eq ${lastitem} ] && delim='' || delim=',' | |
# Don't mess with the heredoc EOF spacing or Bash will cry | |
cat << EOF | |
"${user}": { | |
"name": "${user}", | |
"uid": ${uid}, | |
"gid": ${gid}, | |
"description": "${description}", | |
"home": "${homedir}", | |
"shell": "${shell}" | |
}${delim} | |
EOF | |
done | |
echo "}" | |
fi |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
This is an example of a custom/local Ansible fact for local Linux users on a node. It can optionally gather password expiration info from
/etc/shadow
, assuming the connecting user has the proper permissions.Requirements:
Usage:
gather_shadow
variable at the top of the script, and decide whether to toggle ittrue
orfalse
/etc/ansible/facts.d/
on all of your remote Linux nodes, and copy the file in asusers.fact
chmod +x /etc/ansible/facts.d/users.fact
ansible -i your_host, -m setup your_host
ansible_local.users
for the node.Gathering additional data on password expiration (gather_shadow):
gather_shadow
in the script, which allows you to gather extended info about password expiration for each user. However, Ansible must connect to the node with root/sudo privileges in order for this data to be accessible.db_readable
: This tells you whether the script had permissions to read the /etc/shadow file. Iffalse
, then none of the shadow data for this host will be valid.user_found
: This will betrue
if results were found for a user when the shadow DB was queried. If false, then either the shadow DB was not accessible, or the user is missing an entry in the shadow DB (perhaps a defunct or stale user).With
gather_shadow=true
, a user entry should look like this:With
gather_shadow=false
, a user entry should look like this:Troubleshooting:
bash --version
ansible_local.users
fact.chmod +x /etc/ansible/facts.d/users.fact
gather_facts: true
is set in your playsdb_readable
fact shows asfalse
, then it means Ansible's connecting user did not have sufficient root permissions to retrieve expiration info from the shadow database. Either add the required sudo/root privileges for the connecting user, or disable gathering shadow data by settinggather_shadow=false
in the script.db_readable
istrue
butuser_found
isfalse
, then it means the user is missing from the shadow database. This may be due to an improperly-created or defunct user on the system.