aboutsummaryrefslogtreecommitdiffstats
path: root/roles/web/files/acme/acme-client.sh
blob: 1f4b15d1f637aedb74982c7a8c44d7252e6bff81 (plain)
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
#!/bin/sh
#
# This script can be both used to request/obtain new certificate(s) from
# Let's Encrypt through ACME challenges:
#     $ ./acme-client.sh -n -N
# to expand the domains listed in the certificate:
#     $ ./acme-client.sh -e
# and be used to renew the obtained certificate(s) (default action):
#     $ ./acme-client.sh
# which can be called by periodic(8).
#
# This script will be weekly executed in order to renew the certificate(s).
# See "/etc/periodic.conf".
#
# Output files:
#   * .../etc/acme/privkey.pem : account private key
#   * .../etc/ssl/acme/private/<domain>.pem : domain private key
#
# XXX/TODO:
#   * How to remove/revoke a SAN from the certificate?
#
#
# Aaron LI
# 2017-04-19
#

umask 027

BASEDIR="/usr/local/etc/acme"
SSLDIR="/usr/local/etc/ssl/acme"
DOMAINSFILE="${BASEDIR}/domains.txt"
CHALLENGEDIR="/usr/local/www/acme/.well-known/acme-challenge"
# Default to show verbose information
VERBOSE="true"
# Additional arguments for "acme-client"
ARGS=""


usage() {
    cat << _EOF_
usage:
`basename $0` [-h] [-efLnNv] [-d domains.txt]

    -e : allow expanding the domains listed in the certificate
    -f : force updating the certificate signature even if its too soon
    -n : create a new 4096-bit RSA account key if one does not already exist
    -N : create a new 4096-bit RSA domain key if one does not already exist
    -q : be quiet (default to show verbose information)

    -d domains.txt : text file with one domain and its sub-domains per line
                     (default: ${DOMAINSFILE})
_EOF_
}


while getopts "efhnNqd:" opt; do
    case "$opt" in
        h)
            usage
            exit 1
            ;;
        e)
            ARGS="${ARGS} -e"
            ;;
        f)
            ARGS="${ARGS} -F"
            ;;
        n)
            ARGS="${ARGS} -n"
            ;;
        N)
            ARGS="${ARGS} -N"
            ;;
        q)
            VERBOSE="false"
            ;;
        d)
            DOMAINSFILE="${OPTARG}"
            ;;
        [?])
            usage
            exit 2
            ;;
    esac
done

if [ "${VERBOSE}" = "true" ]; then
    ARGS="${ARGS} -v"
fi

# HACK???
[ ! -f "/etc/ssl/cert.pem" ] && \
    ln -sv /usr/local/etc/ssl/cert.pem /etc/ssl/cert.pem

[ ! -d "${CHALLENGEDIR}" ] && mkdir -pv ${CHALLENGEDIR}
[ ! -d "${SSLDIR}/private" ] && mkdir -pvm700 "${SSLDIR}/private"

printf "\n=== $(date) ===\n=== CMD: $0 $* ===\n"

grep -v '^\s*#' "${DOMAINSFILE}" | while read domain line; do
    printf "-------------------------------------------------------------\n"
    printf "[${domain}] ${line}\n"
    printf "-------------------------------------------------------------\n"
    CERTSDIR="${SSLDIR}/${domain}"
    [ ! -d "${CERTSDIR}" ] && mkdir -pm755 "${CERTSDIR}"
    set +e  # RC=2 when time to expire > 30 days
    acme-client -b -C "${CHALLENGEDIR}" \
                -k "${SSLDIR}/private/${domain}.pem" \
                -c "${CERTSDIR}" \
		${ARGS} \
                ${domain} ${line}
    RC=$?
    set -e
    [ $RC -ne 0 -a $RC -ne 2 ] && exit $RC
done

exit 0