Skip to content

Instantly share code, notes, and snippets.

@michaelrice
Created September 21, 2018 23:11
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save michaelrice/3c052dce7b1e2e1bf46e224c4e9bb692 to your computer and use it in GitHub Desktop.
Save michaelrice/3c052dce7b1e2e1bf46e224c4e9bb692 to your computer and use it in GitHub Desktop.
#!/usr/bin/env bash
# Copyright 2018 Michael Rice <michael@michaelrice.org>
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
set -e
print_help() {
cat << EOF
Usage: ${0##*/} [-h] [-k KEYFILE] [-c CERTFILE] [-m METADATAFILE] [-e ENTITYID] [-b BASEURL]
Example:
${0##*/} -k keystone_fed.key -c keystone_fed.cert -m mellon_metadata.xml \\
-e https://aio.mycloud.com:5000/v3/mellon/metadata -b https://aio.mycloud.com:5000/v3/mellon
Output:
3 Files will be created in the current working directory
1. keystone_fed.cert
2. keystone_fed.key
3. mellon_metadata.xml
Example:
${0##*/} -h
Output:
This help message
EOF
}
while getopts ":k:c:m:e:b:" opt; do
case $opt in
k)
KEYFILE=$OPTARG
;;
c)
CERTFILE=$OPTARG
;;
m)
METADATAFILE=$OPTARG
;;
e)
ENTITYID=$OPTARG
;;
b)
BASEURL=$OPTARG
;;
h)
print_help
exit 0
;;
*)
print_help >&2
exit 1
;;
esac
done
PROG="$(basename "$0")"
if [ "$#" -lt 5 ]; then
print_help
exit 1
fi
if [ -z "$ENTITYID" ]; then
echo "$PROG: An entity ID is required." >&2
exit 1
fi
if [ -z "$BASEURL" ]; then
echo "$PROG: The URL to the MellonEndpointPath is required." >&2
exit 1
fi
if ! echo "$BASEURL" | grep -q '^https\?://'; then
echo "$PROG: The URL must start with \"http://\" or \"https://\"." >&2
exit 1
fi
HOST="$(echo "$BASEURL" | sed 's#^[a-z]*://\([^/]*\).*#\1#')"
BASEURL="$(echo "$BASEURL" | sed 's#/$##')"
echo "Output files:"
echo "Private key: ${KEYFILE}"
echo "Certificate: ${CERTFILE}"
echo "Metadata: ${METADATAFILE}"
echo "Host: ${HOST}"
echo
echo "Endpoints:"
echo "SingleLogoutService (SOAP): ${BASEURL}/logout"
echo "SingleLogoutService (HTTP-Redirect): ${BASEURL}/logout"
echo "AssertionConsumerService (HTTP-POST): ${BASEURL}/postResponse"
echo "AssertionConsumerService (HTTP-Artifact): ${BASEURL}/artifactResponse"
echo "AssertionConsumerService (PAOS): ${BASEURL}/paosResponse"
echo
# No files should not be readable by the rest of the world.
umask 0077
TEMPLATEFILE="$(mktemp -t mellon_create_sp.XXXXXXXXXX)"
cat >"$TEMPLATEFILE" <<EOF
RANDFILE = /dev/urandom
[req]
default_bits = 2048
default_keyfile = privkey.pem
distinguished_name = req_distinguished_name
prompt = no
policy = policy_anything
[req_distinguished_name]
commonName = $HOST
EOF
openssl req -utf8 -batch -config "$TEMPLATEFILE" -new -x509 -days 3652 -nodes -out "${CERTFILE}" -keyout "${KEYFILE}" 2>/dev/null
rm -f "$TEMPLATEFILE"
CERT="$(grep -v '^-----' "${CERTFILE}")"
cat >"${METADATAFILE}" <<EOF
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<EntityDescriptor
entityID="${ENTITYID}"
xmlns="urn:oasis:names:tc:SAML:2.0:metadata">
<SPSSODescriptor
AuthnRequestsSigned="true"
WantAssertionsSigned="true"
protocolSupportEnumeration="urn:oasis:names:tc:SAML:2.0:protocol">
<KeyDescriptor use="signing">
<ds:KeyInfo xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
<ds:X509Data>
<ds:X509Certificate>$CERT</ds:X509Certificate>
</ds:X509Data>
</ds:KeyInfo>
</KeyDescriptor>
<KeyDescriptor use="encryption">
<ds:KeyInfo xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
<ds:X509Data>
<ds:X509Certificate>$CERT</ds:X509Certificate>
</ds:X509Data>
</ds:KeyInfo>
</KeyDescriptor>
<SingleLogoutService
Binding="urn:oasis:names:tc:SAML:2.0:bindings:SOAP"
Location="$BASEURL/logout" />
<SingleLogoutService
Binding="urn:oasis:names:tc:SAML:2.0:bindings:HTTP-Redirect"
Location="$BASEURL/logout" />
<NameIDFormat>urn:oasis:names:tc:SAML:2.0:nameid-format:transient</NameIDFormat>
<AssertionConsumerService
index="0"
isDefault="true"
Binding="urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST"
Location="$BASEURL/postResponse" />
<AssertionConsumerService
index="1"
Binding="urn:oasis:names:tc:SAML:2.0:bindings:HTTP-Artifact"
Location="$BASEURL/artifactResponse" />
<AssertionConsumerService
index="2"
Binding="urn:oasis:names:tc:SAML:2.0:bindings:PAOS"
Location="$BASEURL/paosResponse" />
</SPSSODescriptor>
</EntityDescriptor>
EOF
umask 0777
chmod go+r "${METADATAFILE}"
chmod go+r "${CERTFILE}"
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment