#! /usr/bin/env bash set -e MIN_BASH_VERSION=4 MIN_GPG_VERSION=2.2 MIN_OPENSSL_VERSION=1.1 die() { echo "$@" >&2 exit 1 } check_version(){ [[ $2 == $3 ]] && return 0 local IFS=. local i ver1=($2) ver2=($3) for ((i=${#ver1[@]}; i<${#ver2[@]}; i++)); do ver1[i]=0; done for ((i=0; i<${#ver1[@]}; i++)); do [[ -z ${ver2[i]} ]] && ver2[i]=0 ((10#${ver1[i]} > 10#${ver2[i]})) && return 0 ((10#${ver1[i]} < 10#${ver2[i]})) && die \ "Error: ${1} ${3}+ not found" done } check_tools(){ if [ -z "${BASH_VERSINFO}" ] \ || [ -z "${BASH_VERSINFO[0]}" ] \ || [ ${BASH_VERSINFO[0]} -lt ${MIN_BASH_VERSION} ]; then die "Error: bash ${MIN_BASH_VERSION}+ not found"; fi for cmd in "$@"; do command -v "$1" >/dev/null || die "Error: $cmd not found" case $cmd in gpg) version=$(gpg --version | head -n1 | cut -d" " -f3) check_version "gpg" "${version}" "${MIN_GPG_VERSION}" ;; openssl) version=$(openssl version | cut -d" " -f2 | sed 's/[a-z]//g') check_version "openssl" "${version}" "${MIN_OPENSSL_VERSION}" ;; esac done } get_temp(){ echo "$( mktemp \ --quiet \ --directory \ -t "$(basename "$0").XXXXXX" 2>/dev/null || mktemp \ --quiet \ --directory )" } gpg_env(){ GNUPGHOME=$(get_temp); export GNUPGPHOME killall gpg-agent 2> /dev/null gpg-agent --daemon --extra-socket "$GNUPGHOME/S.gpg-agent" 2> /dev/null echo "export PATH=$GNUPGHOME:$PATH \ export GNUPGHOME=$GNUPGHOME; \ export GPG_AGENT_INFO=$GNUPGHOME/S.gpg-agent" } gpg_cleanup(){ gpgconf --kill gpg-agent rm -rf "$GNUPGHOME" } verify_file() { local filename="${1?}" local sig_count=0 local seen_fingerprints="" local fingerprint local signer for sig_filename in "${filename%.*}".*.asc; do gpg --verify "${sig_filename}" "${filename}" >/dev/null 2>&1 || { echo "Invalid signature: ${sig_filename}"; exit 1; } fingerprint=$( \ gpg --list-packets "${sig_filename}" \ | grep keyid \ | sed 's/.*keyid //g' ) signer=$( \ gpg \ --list-keys \ --with-colons "${fingerprint}" 2>&1 \ | awk -F: '$1 == "uid" {print $10}' \ | head -n1 \ ) [[ "${seen_fingerprints}" == *"${fingerprint}"* ]] && { echo "Duplicate signature: ${sig_filename}"; exit 1; } echo "Verified signature by \"${signer}\"" seen_fingerprints="${seen_fingerprints} ${fingerprint}" ((sig_count=sig_count+1)) done [[ "$sig_count" -ge "$threshold" ]] || { echo "Minimum number of signatures not met: ${sig_count}/${threshold}"; exit 1; } } verify_files() { [ $# -lt 3 ] || die \ "Usage: verify-files (, /dev/null for target_file in ${target_files}; do verify_file "${target_file}" done gpg_cleanup } get_files(){ if command -v git >/dev/null; then git ls-files | grep -v .siglog else find . \ -type f \ -not -path "./.git/*" \ -not -path "./.siglog/*" fi } cmd_manifest() { mkdir -p .${PROGRAM} printf "$(get_files | xargs openssl sha256 -r)" \ | sed -e 's/ \*/ /g' -e 's/ \.\// /g' \ | LC_ALL=C sort -k2 \ > .${PROGRAM}/manifest.txt } cmd_verify() { ( [ -d ".${PROGRAM}" ] && ls .${PROGRAM}/*.asc >/dev/null 2>&1 ) \ || die "Error: No signatures" cmd_manifest for file in .${PROGRAM}/*.asc; do gpg --verify "$file" .${PROGRAM}/manifest.txt done } cmd_sign(){ cmd_manifest gpg --armor --detach-sig .${PROGRAM}/manifest.txt local fingerprint=$( \ gpg --list-packets .${PROGRAM}/manifest.txt.asc \ | grep "issuer key ID" \ | sed 's/.*\([A-Z0-9]\{16\}\).*/\1/g' \ ) mv .${PROGRAM}/manifest.{"txt.asc","${fingerprint}.asc"} } cmd_version() { cat <<-_EOF ============================================ = siglog: simple multisig trust toolchain = = = = v0.0.1 = = = = https://gitlab.com/pchq/siglog = ============================================ _EOF } cmd_usage() { cmd_version cat <<-_EOF Usage: $PROGRAM verify Verify all signing policies for this directory are met $PROGRAM sign Add signature to manifest for this directory $PROGRAM manifest Generate hash manifest for this directory $PROGRAM help Show this text. $PROGRAM version Show version information. _EOF } PROGRAM="${0##*/}" COMMAND="$1" check_tools head cut find sort sed gpg openssl case "$1" in verify) shift; cmd_verify "$@" ;; sign) shift; cmd_sign "$@" ;; manifest) shift; cmd_manifest "$@" ;; version|--version) shift; cmd_version "$@" ;; help|--help) shift; cmd_usage "$@" ;; *) cmd_usage "$@" ;; esac exit 0