summaryrefslogtreecommitdiff
path: root/src/subcommands/m/subkey-to-ssh-agent
blob: 9bedb5e3eb1667673166b89724a9a0b2d4c19c85 (plain)
  1. # -*-shell-script-*-
  2. # This should be sourced by bash (though we welcome changes to make it POSIX sh compliant)
  3. # Monkeysphere subkey-to-ssh-agent subcommand
  4. #
  5. # The monkeysphere scripts are written by:
  6. # Jameson Rollins <jrollins@finestructure.net>
  7. # Jamie McClelland <jm@mayfirst.org>
  8. # Daniel Kahn Gillmor <dkg@fifthhorseman.net>
  9. #
  10. # They are Copyright 2008-2009, and are all released under the GPL,
  11. # version 3 or later.
  12. # try to add all authentication subkeys to the agent
  13. subkey_to_ssh_agent() {
  14. local sshaddresponse
  15. local secretkeys
  16. local authsubkeys
  17. local workingdir
  18. local keysuccess
  19. local subkey
  20. local publine
  21. local kname
  22. if ! test_gnu_dummy_s2k_extension ; then
  23. failure "Your version of GnuTLS does not seem capable of using with gpg's exported subkeys.
  24. You may want to consider patching or upgrading to GnuTLS 2.6 or later.
  25. For more details, see:
  26. http://lists.gnu.org/archive/html/gnutls-devel/2008-08/msg00005.html"
  27. fi
  28. # if there's no agent running, don't bother:
  29. if [ -z "$SSH_AUTH_SOCK" ] || ! which ssh-add >/dev/null ; then
  30. failure "No ssh-agent available."
  31. fi
  32. # and if it looks like it's running, but we can't actually talk to
  33. # it, bail out:
  34. ssh-add -l >/dev/null
  35. sshaddresponse="$?"
  36. if [ "$sshaddresponse" = "2" ]; then
  37. failure "Could not connect to ssh-agent"
  38. fi
  39. # get list of secret keys (to work around https://bugs.g10code.com/gnupg/issue945):
  40. secretkeys=$(gpg --list-secret-keys --with-colons --fixed-list-mode --fingerprint | \
  41. grep '^fpr:' | cut -f10 -d: | awk '{ print "0x" $1 "!" }')
  42. if [ -z "$secretkeys" ]; then
  43. failure "You have no secret keys in your keyring!
  44. You might want to run 'gpg --gen-key'."
  45. fi
  46. authsubkeys=$(gpg --list-secret-keys --with-colons --fixed-list-mode \
  47. --fingerprint --fingerprint $secretkeys | \
  48. cut -f1,5,10,12 -d: | grep -A1 '^ssb:[^:]*::[^:]*a[^:]*$' | \
  49. grep '^fpr::' | cut -f3 -d: | sort -u)
  50. if [ -z "$authsubkeys" ]; then
  51. failure "no authentication-capable subkeys available.
  52. You might want to 'monkeysphere gen-subkey'"
  53. fi
  54. workingdir=$(mktemp -d ${TMPDIR:-/tmp}/tmp.XXXXXXXXXX)
  55. umask 077
  56. mkfifo "$workingdir/passphrase"
  57. keysuccess=1
  58. # FIXME: we're currently allowing any other options to get passed
  59. # through to ssh-add. should we limit it to known ones? For
  60. # example: -d or -c and/or -t <lifetime>
  61. for subkey in $authsubkeys; do
  62. # choose a label by which this key will be known in the agent:
  63. # we are labelling the key by User ID instead of by
  64. # fingerprint, but filtering out all / characters to make sure
  65. # the filename is legit.
  66. primaryuid=$(gpg --with-colons --list-key "0x${subkey}!" | grep '^pub:' | cut -f10 -d: | tr -d /)
  67. #kname="[monkeysphere] $primaryuid"
  68. kname="$primaryuid"
  69. if [ "$1" = '-d' ]; then
  70. # we're removing the subkey:
  71. gpg --export "0x${subkey}!" | openpgp2ssh "$subkey" > "$workingdir/$kname"
  72. (cd "$workingdir" && ssh-add -d "$kname")
  73. else
  74. # we're adding the subkey:
  75. mkfifo "$workingdir/$kname"
  76. gpg --quiet --passphrase-fd 3 3<"$workingdir/passphrase" \
  77. --export-options export-reset-subkey-passwd,export-minimal,no-export-attributes \
  78. --export-secret-subkeys "0x${subkey}!" | openpgp2ssh "$subkey" > "$workingdir/$kname" &
  79. (cd "$workingdir" && DISPLAY=nosuchdisplay SSH_ASKPASS=/bin/false ssh-add "$@" "$kname" </dev/null )&
  80. passphrase_prompt "Enter passphrase for key $kname: " "$workingdir/passphrase"
  81. wait %2
  82. fi
  83. keysuccess="$?"
  84. rm -f "$workingdir/$kname"
  85. done
  86. rm -rf "$workingdir"
  87. # FIXME: sort out the return values: we're just returning the
  88. # success or failure of the final authentication subkey in this
  89. # case. What if earlier ones failed?
  90. exit "$keysuccess"
  91. }