Skip to main content

A Python keyring backend unlocked by ssh-agent signatures

Project description

ssh-agent-keyring

ssh-agent-keyring is a Python keyring backend that keeps secrets encrypted at rest and unlocks them transparently through ssh-agent.

It is designed for SSH-first environments such as headless Linux servers, remote development machines, and terminal-centric workflows where desktop keyring services are not available or not desirable.

Why this exists

On a plain SSH server, the usual choices are often unsatisfying:

  • desktop-oriented keyring backends are unavailable
  • file-based backends usually require a separate keyring password
  • plaintext storage is convenient, but unacceptable for real secrets

ssh-agent-keyring takes a different approach:

  • secrets are stored encrypted at rest
  • unlock happens through a configured SSH key already available in ssh-agent
  • keyring set/get/del work in ordinary SSH sessions without a second keyring password

Features

  • keyring backend implementation: ssh_agent_keyring.backend.SSHAgentKeyring
  • encrypted local store
  • no plaintext secret storage
  • no plaintext service / username metadata in the store
  • no root or system service dependency
  • works well in SSH sessions when SSH_AUTH_SOCK is available
  • simple user-space configuration

How it works

The backend stores a random data-encryption key (DEK) in wrapped form. To unwrap it, the backend:

  1. uses ssh-keygen -Y sign with a configured SSH public key
  2. relies on ssh-agent to produce a deterministic signature over a backend-specific challenge
  3. derives a key-encryption key from that signature with HKDF-SHA256
  4. unwraps the DEK
  5. uses the DEK to encrypt and decrypt secret records with ChaCha20Poly1305

Record identifiers are derived with HMAC, so the on-disk store does not expose service and username pairs in plaintext.

Requirements

  • Python 3.10+
  • keyring
  • cryptography
  • OpenSSH tools providing:
    • ssh-add
    • ssh-keygen
  • a running ssh-agent
  • SSH_AUTH_SOCK set in the current environment
  • the configured SSH key loaded in the agent

Installation

From PyPI:

pip install ssh-agent-keyring

If you are installing in a user environment on a remote machine, a typical setup is:

python3 -m pip install --user ssh-agent-keyring

Configuration

The backend reads configuration from:

~/.config/ssh-agent-keyring/config.json

Example:

{
  "public_key": "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIExampleBase64KeyMaterial",
  "store_path": "~/.local/share/ssh-agent-keyring/store.json"
}

Then select the backend in:

~/.config/python_keyring/keyringrc.cfg
[backend]
default-keyring=ssh_agent_keyring.backend.SSHAgentKeyring

Configuration fields

  • public_key: SSH public key to use for unlocking the store
  • public_key_path: path to a public key file; useful instead of embedding the key in JSON
  • store_path: path to the encrypted JSON store

Environment variable overrides

These override file-based configuration:

  • SSH_AGENT_KEYRING_PUBLIC_KEY
  • SSH_AGENT_KEYRING_PUBLIC_KEY_PATH
  • SSH_AGENT_KEYRING_STORE_PATH

Usage

With the Python API:

import keyring

keyring.set_password("example-service", "alice", "s3cr3t")
print(keyring.get_password("example-service", "alice"))
keyring.delete_password("example-service", "alice")

With the CLI:

keyring set example-service alice
keyring get example-service alice
keyring del example-service alice

With this backend, keyring set prompts only for the secret being stored. It does not prompt for a separate keyring password.

Operational model

Transparent unlock works when all of the following are true:

  • SSH_AUTH_SOCK is available in the current session
  • ssh-agent is reachable
  • the configured key is loaded in that agent
  • the current session is allowed to use that agent

If the agent is unavailable, the wrong key is configured, or the key is not loaded, the encrypted store cannot be unlocked.

Security notes

This package improves on plaintext file storage, but it intentionally shifts trust to the SSH agent session.

Important implications:

  • anyone able to use your SSH_AUTH_SOCK can potentially unlock the keyring during that session
  • agent forwarding policy matters
  • the security of the backend depends on the security of the selected SSH key and the agent socket
  • this is not a hardware-backed secret manager or multi-user secret vault

The store is designed to avoid:

  • plaintext secrets on disk
  • plaintext service/user metadata on disk
  • a second unlock password unrelated to your SSH workflow

Limitations

  • this backend depends on OpenSSH tooling and current ssh-agent behavior
  • it is aimed at Unix-like environments
  • it does not provide secret sharing, rotation workflows, audit logs, or remote synchronization
  • it assumes the configured SSH key can be used for agent-backed signing

Testing

Run the test suite with:

python3 -m unittest discover -s tests -v

The project includes tests for roundtrips, tampering, key mismatch, signer failures, and store handling edge cases.

Development

Project layout:

  • ssh_agent_keyring/agent.py - agent and signing integration
  • ssh_agent_keyring/crypto.py - key derivation and encryption helpers
  • ssh_agent_keyring/backend.py - keyring backend implementation
  • tests/test_backend.py - automated test suite

License

License information has not been added yet.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

ssh_agent_keyring-0.2.0-py3-none-any.whl (8.3 kB view details)

Uploaded Python 3

File details

Details for the file ssh_agent_keyring-0.2.0-py3-none-any.whl.

File metadata

File hashes

Hashes for ssh_agent_keyring-0.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 fc7ed9ac94ce0f8d36a5eeb1989de05fd754a3840f3c6bf8e9ded1acbfce8ba2
MD5 3ce0ed8f21517b7b1d99e494a2178012
BLAKE2b-256 38e84ccf843e1ea530b4e575c3c0f4c2b39a022bf6b308255e357043428b6b28

See more details on using hashes here.

Provenance

The following attestation bundles were made for ssh_agent_keyring-0.2.0-py3-none-any.whl:

Publisher: pypi-publish.yml on monperrus/ssh_agent_keyring

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page