-
Notifications
You must be signed in to change notification settings - Fork 22
/
Copy pathldap-auth.sh
executable file
·237 lines (209 loc) · 6.88 KB
/
ldap-auth.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
#!/bin/sh
#
# ldap-auth.sh - Simple shell script to authenticate users against LDAP
#
# This script should be able to run on any POSIX-compliant shell with
# cat, grep and sed available (even works on BusyBox).
# It expects to get two environment variables: username and password of
# the user to be authenticated.
#
# It then rewrites the username into a DN and tries to bind with it,
# using the supplied password.
#
# When binding succeded, it can optionally execute a search to e.g. check
# for group memberships or, alternatively, grant access straight away.
#
# When access is granted, it exits with an exit code of 0, non-zero
# otherwise. Status messages are only written to stderr, not stdout,
# so you can use stdout to pass whatever you want back to the caller.
#
# In order to run ldap-auth.sh, you need to create a configuration file
# first. Either take a config from the examples directory or copy the
# available settings below to a new file.
# NOTE: The configuration file will be sourced, hence you can use regular
# shell script syntax inside. Since the config file is in fact executed,
# make sure it's content is safe.
#
# Finally, configure the service that wants to do LDAP authentication to
# execute ldap-auth.sh and pass the configuration file as its only argument.
#
# Additionally, you may define the following functions in your config file,
# which get then called at the corresponding events, before ldap-auth.sh
# exits. You could print some additional info from these functions,
# for instance.
#
# on_auth_success() {
# ...
# }
#
# on_auth_failure() {
# ...
# }
#
# In these functions, you have access to the $output variable,
# which will hold the raw LDIF output produced by the LDAP client.
# It won't include comments, but curl produces a somewhat invalid LDIF
# with blank lines between different attributes and deep indentation.
#
########## CONFIGURATION ##########
# Some settings (those not commented out) have default values set. You
# don't need to include them in your configuration file unless you want
# to change these defaults.
# Uncomment to enable debugging to stderr (prints full client output
# and more).
#DEBUG=1
# Must be one of "curl" and "ldapsearch".
# NOTE:
# - When choosing "curl", make sure "curl --version | grep ldap" outputs
# something. Otherwise, curl was compiled without LDAP support.
# - When choosing "ldapsearch", make sure the ldapwhoami command is
# available as well, as that might be needed in some cases.
#CLIENT="curl"
# Usernames should be validated using a regular expression to be of
# a known format. Special characters will be escaped anyway, but it is
# generally not recommended to allow more than necessary.
# This pattern is set by default. In your config file, you can either
# overwrite it with a different one or use "unset USERNAME_PATTERN" to
# disable validation completely.
USERNAME_PATTERN='^[a-z|A-Z|0-9|_|-|.]+$'
# Adapt to your needs.
#SERVER="ldap://ldap-server:389"
# Will try binding as this user.
# ldap_dn_escape escapes special characters in strings to make them
# usable within LDAP DN components.
#USERDN="uid=$(ldap_dn_escape "$username"),ou=people,dc=example,dc=com"
# If you want to take additional checks like requiring group memberships
# or fetch specific user attributes, you can execute a custom search, which
# has to return exactly one result in order for authentication to succeed.
# Uncomment the following lines to enable search query execution.
#BASEDN="$USERDN"
#SCOPE="base"
#FILTER="(&(objectClass=person)(memberOf=cn=some-group,ou=groups,dc=example,dc=com))"
# Space-separated list of additional LDAP attributes to query.
# You could process them in your own on_auth_success hook.
#ATTRS="cn"
# When the timeout (in seconds) is exceeded (e.g. due to slow networking),
# authentication fails.
TIMEOUT=3
########## END OF CONFIGURATION ##########
########## SCRIPT CODE FOLLOWS, DON'T TOUCH! ##########
# Log messages to stderr.
log() {
echo "$1" >&2
}
# Escape string to be safely usable in LDAP DN components and URIs.
# https://ldapwiki.com/wiki/DN%20Escape%20Values
ldap_dn_escape() {
escaped=$(echo "$1" | sed -r \
-e 's/[,\\#+<>;"=/?]/\\\0/g' \
-e 's/^ (.*)$/\\ \1/' \
-e 's/^(.*) $/\1\\ /' \
)
[ -z "$DEBUG" ] || log "Escaped '$1' to '$escaped'."
echo "$escaped"
}
# The different client implementations.
ldap_auth_curl() {
[ -z "$DEBUG" ] || verbose="-v"
attrs=$(echo "$ATTRS" | sed "s/ /,/g")
output=$(curl $verbose -s -m "$TIMEOUT" -u "$USERDN:$password" \
"$SERVER/$BASEDN?dn,$attrs?$SCOPE?$FILTER")
[ $? -ne 0 ] && return 1
return 0
}
ldap_auth_ldapsearch() {
common_opts="-o nettimeout=$TIMEOUT -H $SERVER -x"
[ -z "$DEBUG" ] || common_opts="-v $common_opts"
if [ -z "$BASEDN" ]; then
output=$(ldapwhoami $common_opts -D "$USERDN" -w "$password")
else
output=$(ldapsearch $common_opts -LLL \
-D "$USERDN" -w "$password" \
-s "$SCOPE" -b "$BASEDN" "$FILTER" dn $ATTRS)
fi
[ $? -ne 0 ] && return 1
return 0
}
# Source the config file.
if [ -z "$1" ]; then
log "Usage: ldap-auth.sh <config-file>"
exit 2
fi
CONFIG_FILE=$(realpath "$1")
if [ ! -e "$CONFIG_FILE" ]; then
log "'$CONFIG_FILE': not found"
exit 2
elif [ ! -f "$CONFIG_FILE" ]; then
log "'$CONFIG_FILE': not a file"
exit 2
elif [ ! -r "$CONFIG_FILE" ]; then
log "'$CONFIG_FILE': no read permission"
exit 2
fi
. "$CONFIG_FILE"
# Validate config.
err=0
if [ -z "$SERVER" ] || [ -z "$USERDN" ]; then
log "SERVER and USERDN need to be configured."
err=1
fi
if [ -z "$TIMEOUT" ]; then
log "TIMEOUT needs to be configured."
err=1
fi
if [ ! -z "$BASEDN" ]; then
if [ -z "$SCOPE" ] || [ -z "$FILTER" ]; then
log "BASEDN, SCOPE and FILTER may only be configured together."
err=1
fi
elif [ ! -z "$ATTRS" ]; then
log "Configuring ATTRS only makes sense when enabling searching."
err=1
fi
# Check username and password are present and not malformed.
if [ -z "$username" ] || [ -z "$password" ]; then
log "Need username and password environment variables."
err=1
elif [ ! -z "$USERNAME_PATTERN" ]; then
username_match=$(echo "$username" | sed -r "s/$USERNAME_PATTERN/x/")
if [ "$username_match" != "x" ]; then
log "Username '$username' has an invalid format."
err=1
fi
fi
[ $err -ne 0 ] && exit 2
# Do the authentication.
case "$CLIENT" in
"curl")
ldap_auth_curl
;;
"ldapsearch")
ldap_auth_ldapsearch
;;
*)
log "Unsupported client '$CLIENT', revise the configuration."
exit 2
;;
esac
result=$?
entries=0
if [ $result -eq 0 ] && [ ! -z "$BASEDN" ]; then
entries=$(echo "$output" | grep -cie '^dn\s*:')
[ "$entries" != "1" ] && result=1
fi
if [ ! -z "$DEBUG" ]; then
cat >&2 <<-EOF
Result: $result
Number of entries: $entries
Client output:
$output
EOF
fi
if [ $result -ne 0 ]; then
log "User '$username' failed to authenticate."
type on_auth_failure > /dev/null && on_auth_failure
exit 1
fi
log "User '$username' authenticated successfully."
type on_auth_success > /dev/null && on_auth_success
exit 0