mirror of
https://github.com/mail-in-a-box/mailinabox.git
synced 2025-04-04 00:17:06 +00:00
139 lines
4.6 KiB
Bash
139 lines
4.6 KiB
Bash
#####
|
|
##### This file is part of Mail-in-a-Box-LDAP which is released under the
|
|
##### terms of the GNU Affero General Public License as published by the
|
|
##### Free Software Foundation, either version 3 of the License, or (at
|
|
##### your option) any later version. See file LICENSE or go to
|
|
##### https://github.com/downtownallday/mailinabox-ldap for full license
|
|
##### details.
|
|
#####
|
|
|
|
#
|
|
# REST helper functions
|
|
#
|
|
# requirements:
|
|
# system packages: [ curl ]
|
|
# lib scripts: [ system.sh, color-output.sh ]
|
|
#
|
|
|
|
rest_urlencoded() {
|
|
# Issue a REST call having data urlencoded
|
|
#
|
|
# eg: rest_urlencoded POST /admin/mail/users/add "email=alice@abc.com" "password=secret"
|
|
#
|
|
# When providing a URI (/mail/users/add) and not a URL
|
|
# (https://host/mail/users/add), PRIMARY_HOSTNAME must be set!
|
|
#
|
|
# The function will set the following global variables regardless
|
|
# of exit code:
|
|
# REST_HTTP_CODE
|
|
# REST_OUTPUT
|
|
# REST_ERROR
|
|
#
|
|
# Return values:
|
|
# 0 indicates success (curl returned 0 or a code deemed to be
|
|
# successful and HTTP status is >=200 but <300)
|
|
# 1 curl returned with non-zero code that indicates an error
|
|
# 2 the response status was <200 or >= 300
|
|
#
|
|
# Debug messages are sent to stderr
|
|
#
|
|
local verb="$1" # eg "POST"
|
|
local uri="$2" # eg "/mail/users/add"
|
|
local auth_user="$3"
|
|
local auth_pass="$4"
|
|
shift; shift; shift; shift # remaining arguments are data or curl args
|
|
|
|
local url
|
|
local is_local="false"
|
|
case "$uri" in
|
|
http:* | https:* )
|
|
url="$uri"
|
|
;;
|
|
* )
|
|
url="https://$PRIMARY_HOSTNAME${uri}"
|
|
is_local="true"
|
|
;;
|
|
esac
|
|
|
|
local data=()
|
|
local item output onlydata="false"
|
|
|
|
if [ ! -z "$auth_user" ]; then
|
|
data+=("--user" "${auth_user}:${auth_pass}")
|
|
fi
|
|
|
|
for item; do
|
|
case "$item" in
|
|
-- )
|
|
onlydata="true"
|
|
;;
|
|
--* )
|
|
# curl argument
|
|
if $onlydata; then
|
|
data+=("--data-urlencode" "$item");
|
|
else
|
|
# if argument is like "--header=<val>", then change to
|
|
# "--header <val>" because curl wants the latter
|
|
local arg="$(awk -F= '{print $1}' <<<"$item")"
|
|
local val="$(awk -F= '{print substr($0,length($1)+2)}' <<<"$item")"
|
|
if [ -z "$val" ]; then
|
|
data+=("$item")
|
|
else
|
|
data+=("$arg" "$val")
|
|
fi
|
|
fi
|
|
;;
|
|
* )
|
|
onlydata="true"
|
|
data+=("--data-urlencode" "$item");
|
|
;;
|
|
esac
|
|
done
|
|
|
|
echo "spawn: curl -w \"%{http_code}\" -X $verb ${data[@]} $url" 1>&2
|
|
# pipe through 'tr' to avoid bash "warning: command substitution: ignored null byte in input" where curl places a \0 between output and http_code
|
|
output=$(curl -s -S -w "%{http_code}" -X $verb "${data[@]}" $url | tr -d '\0')
|
|
local code=$?
|
|
|
|
# http status is last 3 characters of output, extract it
|
|
REST_HTTP_CODE=$(awk '{S=substr($0,length($0)-2)} END {print S}' <<<"$output")
|
|
REST_OUTPUT=$(awk 'BEGIN{L=""}{ if(L!="") print L; L=$0 } END { print substr(L,1,length(L)-3) }' <<<"$output")
|
|
REST_ERROR=""
|
|
[ -z "$REST_HTTP_CODE" ] && REST_HTTP_CODE="000"
|
|
|
|
if [ $code -ne 0 ]; then
|
|
if [ $code -eq 56 -a $REST_HTTP_CODE -eq 200 ]; then
|
|
# this is okay, I guess. happens sometimes during
|
|
# POST /admin/mail/aliases/remove
|
|
# 56=Unexpected EOF
|
|
echo "Ignoring curl return code 56 due to 200 status" 1>&2
|
|
|
|
elif [ $code -ne 16 -o $REST_HTTP_CODE -ne 200 ]; then
|
|
# any error code will fail the rest call except for a 16
|
|
# with a 200 HTTP status.
|
|
# 16="a problem was detected in the HTTP2 framing layer. This is somewhat generic and can be one out of several problems"
|
|
REST_ERROR="CURL failed with code $code"
|
|
echo "${F_DANGER}$REST_ERROR${F_RESET}" 1>&2
|
|
echo "$output" 1>&2
|
|
return 1
|
|
fi
|
|
fi
|
|
if [ $REST_HTTP_CODE -lt 200 -o $REST_HTTP_CODE -ge 300 ]; then
|
|
if [ -z "$REST_OUTPUT" ]; then
|
|
REST_ERROR="Server returned status $REST_HTTP_CODE"
|
|
else
|
|
REST_ERROR="Server returned status $REST_HTTP_CODE: $REST_OUTPUT"
|
|
fi
|
|
echo "${F_DANGER}$REST_ERROR${F_RESET}" 1>&2
|
|
if $is_local && [ $REST_HTTP_CODE -ge 500 ]; then
|
|
echo -n "$F_WARN"
|
|
tail -100 /var/log/syslog | grep -i "(traceback|err|warn|fail|fatal|uncaught)" 1>&2
|
|
echo -n "$F_RESET"
|
|
fi
|
|
return 2
|
|
fi
|
|
echo "CURL succeded, HTTP status $REST_HTTP_CODE" 1>&2
|
|
echo "$output" 1>&2
|
|
return 0
|
|
}
|