Skip to main content

A Python implementation of Shamir's Secret Sharing

Project description

codecov PyPI version License Sanity tests

Horcrux

A Python implementation of Shamir's Secret Sharing based on HashiCorp Vault's approach.

Horcrux lets you split secrets into multiple shares where any threshold number can reconstruct the original secret, but fewer shares reveal nothing. Think of it as splitting a key to a vault into pieces—you need enough pieces to open it, but each piece alone is useless.

Why Use Secret Sharing?

Single Point of Failure: Storing a password, encryption key, or wallet seed in one place means losing it loses everything.

Single Point of Compromise: Storing it in one place also means finding it steals everything.

Secret Sharing Solves Both: Split your secret into N shares where any K shares can reconstruct it. Now you can:

  • Lose up to (N-K) shares and still recover your secret
  • Require an attacker to compromise K locations instead of one
  • Distribute trust across multiple parties without any single party having access

Features

  • Zero runtime dependencies - Minimal attack surface
  • Information-theoretic security - Shares below threshold reveal nothing
  • Constant-time operations - Resistant to timing attacks
  • 100% test coverage - Extensively tested with property-based tests
  • Type-safe - Full type hints with strict mypy checking
  • Share versioning - Backward compatible format with version detection
  • Python 3.11+ - Modern Python with performance optimizations
  • Battle-tested - Based on HashiCorp Vault's proven implementation

Installation

pip install horcrux

Or with uv:

uv add horcrux

Quick Start

from shamir import split, combine

# Your secret (password, key, seed phrase, etc.)
secret = b"correct-horse-battery-staple"

# Split into 5 shares, any 3 can reconstruct
shares = split(secret, parts=5, threshold=3)

# Distribute shares to different locations:
# - Share 1: Home safe
# - Share 2: Bank deposit box
# - Share 3: Trusted family member
# - Share 4: Cloud storage (encrypted)
# - Share 5: Office safe

# Later, recover from any 3 shares
recovered = combine([shares[0], shares[2], shares[4]])
assert recovered == secret

Real-World Examples

🔐 Cryptocurrency Wallet Backup

Protect your crypto wallet seed phrase across multiple secure locations. Lose your house in a fire? Your backup shares in other cities can still recover your wallet. See examples/crypto_wallet.py for complete implementation with multiple backup strategies.

from examples.crypto_wallet import WalletBackup, BackupStrategy

# Your BIP39 seed phrase
seed = "witch collapse practice feed shame open despair creek road again ice least"

# Create backup with balanced strategy (5 shares, need 3)
backup = WalletBackup(seed, strategy=BackupStrategy.BALANCED)
shares = backup.create_shares()

# Store shares in geographically distributed locations
# Later, recover from any 3 shares
recovered_seed = WalletBackup.recover_seed([shares[0], shares[2], shares[4]])

Strategies Available:

  • Convenient: 3 shares, need 2 (easier management)
  • Balanced: 5 shares, need 3 (recommended)
  • Paranoid: 7 shares, need 4 (high security)
  • Maximum: 9 shares, need 5 (maximum security)

💼 Digital Inheritance

Ensure your family can access your digital assets after you pass away, but not before. Split passwords and account recovery information across family members and your estate lawyer. See examples/digital_inheritance.py for complete estate planning solution.

from examples.digital_inheritance import setup_inheritance_package

# Package your digital assets
assets = {
    'password_manager_master': 'your-master-password',
    'bitcoin_wallet_seed': '24 word seed phrase...',
    'recovery_email': 'backup@example.com',
    'bank_account_info': 'Account details...'
}

# Create 5 shares distributed to: Spouse, Children, Executor, Lawyer
shares = setup_inheritance_package(assets, num_shares=5, threshold=3)

# After legal declaration of death, any 3 beneficiaries can reconstruct

More Examples

API Reference

split(secret, parts, threshold, rng=None, version=None)

Split a secret into cryptographic shares.

Parameters:

  • secret (bytes): The secret data to split
  • parts (int): Total number of shares to create (2-255)
  • threshold (int): Minimum shares needed to reconstruct (2-255)
  • rng (Random, optional): Random number generator. Defaults to SystemRandom() (cryptographically secure)
  • version (int, optional): Share format version (0=legacy, 1=versioned). Defaults to 1

Returns: list[bytearray] - The generated shares

Raises: ValueError - If parameters are invalid (empty secret, parts < threshold, values > 255, etc.)

Example:

from shamir import split

secret = b"my-secret-key"
shares = split(secret, parts=5, threshold=3)
# Returns 5 shares, any 3 can reconstruct the secret

combine(parts)

Reconstruct a secret from shares.

Parameters:

  • parts (list[bytearray]): List of shares to combine (at least threshold required)

Returns: bytearray - The reconstructed secret

Raises:

  • ValueError - If fewer than 2 parts provided, parts have mismatched lengths, duplicate parts detected, or mixing different share versions

Important: This function does not validate the threshold. Providing fewer than the required threshold shares will produce an incorrect result without error. Always ensure you provide at least the threshold number of shares used during split().

Example:

from shamir import combine

# Combine any threshold number of shares
recovered = combine([shares[0], shares[2], shares[4]])
print(recovered.decode('utf-8'))

How It Works

Horcrux implements Shamir's Secret Sharing over the Galois Field GF(256):

  1. Polynomial Construction: For each byte of the secret, create a random polynomial of degree (threshold-1) with the secret byte as the constant term
  2. Share Generation: Evaluate the polynomial at N different x-coordinates to produce N shares
  3. Secret Recovery: Use Lagrange interpolation to reconstruct the polynomial from any K shares, then evaluate at x=0 to recover the secret
  4. Security: Information-theoretic security means K-1 shares reveal mathematically zero information about the secret

Each share consists of:

  • Version 1 format (default): [version_byte, y_values..., x_coordinate]
  • Legacy format: [y_values..., x_coordinate]

The library automatically detects and handles both formats for backward compatibility.

Security Properties

Information-Theoretic Security

Below threshold shares reveal nothing: This isn't just "hard to break" cryptography. With K-1 shares, for ANY possible secret value, there exists a valid Kth share that would reconstruct to that secret. This means K-1 shares provide zero information about which secret is the real one.

Security Guarantees

  • Constant-time operations: GF(256) operations use constant-time implementations to prevent timing attacks
  • No secret branching: Code paths don't branch based on secret values
  • Cryptographic RNG: Defaults to SystemRandom() which uses OS entropy
  • No dependencies: Zero runtime dependencies means minimal supply chain risk
  • Tested security: Property-based tests verify security invariants

Threat Model

Protected Against:

  • Compromise of up to (threshold-1) share locations
  • Loss of up to (parts-threshold) shares
  • Timing attacks on reconstruction
  • Partial information leakage from shares

Not Protected Against:

  • Compromise of threshold or more shares
  • Side-channel attacks on the system running the code (OS-level)
  • Social engineering to gather shares
  • Compromised random number generator

Best Practices

  1. Use high thresholds for valuable secrets: 3-of-5 minimum, 4-of-7 for high value
  2. Geographic distribution: Store shares in different physical locations
  3. Multiple custodians: Don't give one person multiple shares
  4. Secure storage: Encrypt shares before cloud storage (defense in depth)
  5. Regular audits: Periodically verify shares are still accessible
  6. Rotation after recovery: Generate new shares after reconstruction (old shares may be compromised)

Performance

  • Speed: Splits/combines 1MB secrets in <500ms on modern hardware
  • Scalability: Supports up to 255 shares with any threshold
  • Complexity: O(n×m×k) where n=secret length, m=number of shares, k=threshold
  • Memory: O(secret_size × parts) during split, O(secret_size × threshold) during combine

Tested with:

  • Secrets up to 10MB
  • Up to 255 shares (maximum possible)
  • Various threshold configurations

FAQ

Q: How many shares should I create? A: 5 shares with threshold of 3 is a good balance for most use cases. High-value secrets may want 7 shares with threshold of 4.

Q: What happens if I lose some shares? A: You can lose up to (parts - threshold) shares and still recover. With 5 shares and threshold of 3, you can lose 2 shares.

Q: Can an attacker learn anything from one share? A: No. This has information-theoretic security—below threshold shares reveal mathematically zero information about the secret.

Q: Should I encrypt shares before storing? A: Shares are already cryptographically secure, but encrypting before cloud storage adds defense in depth. Local storage (safe, deposit box) doesn't need encryption.

Q: Can I split already-encrypted data? A: Yes. Secret sharing works on any binary data, including encrypted data, password hashes, or random keys.

Q: What's the difference between version 0 and version 1 shares? A: Version 1 (default) includes a version byte for future compatibility. Version 0 is legacy format. Both are supported, and the library auto-detects which format is used. Use version 1 for new shares.

Q: Is this the same as multi-sig cryptocurrency wallets? A: Similar concept but different implementation. Multi-sig requires blockchain support. Secret sharing works for any secret (keys, passwords, files) and doesn't require blockchain.

Q: Can I use this for commercial projects? A: Yes. Licensed under MPL-2.0, which permits commercial use. See LICENSE for details.

Development

Setup

# Clone repository
git clone https://github.com/reidhoch/horcrux.git
cd horcrux

# Install dependencies
uv sync

# Install pre-commit hooks
uv run pre-commit install

Running Tests

# Run full test suite (142 tests)
uv run pytest

# With coverage report
uv run pytest --cov=shamir --cov-report=html

# Run specific test file
uv run pytest tests/test_shamir.py -v

# Run with parallelization
uv run pytest -n auto

Code Quality

# Run all pre-commit checks
uv run pre-commit run --all-files

# Individual checks
uv run ruff check shamir          # Linting
uv run ruff format shamir         # Formatting
uv run mypy shamir/               # Type checking

Contributing

See AGENTS.md for comprehensive development guidelines including:

  • Code conventions and style requirements
  • Testing best practices
  • Security considerations
  • API design principles
  • Code review guidelines

All contributions must:

  • Pass all tests (100% coverage required)
  • Pass ruff linting with all rules enabled
  • Pass mypy strict type checking
  • Include tests for new functionality
  • Follow existing code patterns

Comparison

Feature Horcrux secretsharing pyshamir
Runtime Dependencies 0 6+ 2+
Type Hints Full Partial None
Test Coverage 100% ~60% ~40%
Constant-time Ops
Share Versioning
Property Tests
Python 3.11+
Active Maintenance ⚠️ ⚠️

Changelog

See Releases for detailed version history.

Recent changes:

  • v1.1.0: Added share format versioning with backward compatibility
  • v1.0.7: Comprehensive security and property-based testing
  • v1.0.0: Initial stable release

License

This project is licensed under the Mozilla Public License 2.0 (MPL-2.0).

This means you can:

  • ✅ Use commercially
  • ✅ Modify and distribute
  • ✅ Use privately
  • ✅ Include in larger proprietary works

You must:

  • 📄 Include license and copyright notice
  • 📄 Disclose source for MPL-licensed files
  • 📄 State changes made to MPL-licensed files

See LICENSE for full details.

Acknowledgments

  • Based on HashiCorp Vault's implementation of Shamir's Secret Sharing
  • Implements the algorithm described in Shamir's 1979 paper: "How to share a secret"
  • GF(256) arithmetic inspired by various SSS implementations and academic papers
  • Testing approach inspired by property-based testing methodology

Citation

If you use Horcrux in academic work, please cite:

@software{horcrux,
  author = {Hochstedler, Reid},
  title = {Horcrux: A Python implementation of Shamir's Secret Sharing},
  year = {2025},
  url = {https://github.com/reidhoch/horcrux}
}

Questions? Open an issue or start a discussion.

Security Issues? See SECURITY.md for responsible disclosure guidelines.

Project details


Download files

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

Source Distribution

horcrux-1.1.1.tar.gz (573.1 kB view details)

Uploaded Source

Built Distribution

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

horcrux-1.1.1-py3-none-any.whl (18.7 kB view details)

Uploaded Python 3

File details

Details for the file horcrux-1.1.1.tar.gz.

File metadata

  • Download URL: horcrux-1.1.1.tar.gz
  • Upload date:
  • Size: 573.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.5

File hashes

Hashes for horcrux-1.1.1.tar.gz
Algorithm Hash digest
SHA256 6dc24ba4ca76bd34b33062171c945b237b60c125883e0150ef150cd328bbef0c
MD5 7e1352980cd35de6560d136007efcbd9
BLAKE2b-256 b086c98173998ccfa125e15217c1b74971743afc988e6e4df7bf242d368e0e77

See more details on using hashes here.

File details

Details for the file horcrux-1.1.1-py3-none-any.whl.

File metadata

  • Download URL: horcrux-1.1.1-py3-none-any.whl
  • Upload date:
  • Size: 18.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.5

File hashes

Hashes for horcrux-1.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 4f78b502b3df016f48c882e72d71458aaac83dd8bb01bc4cfbe6b8e600eb4bf7
MD5 3a0786a12c83daa7df4497aa56e49441
BLAKE2b-256 ab5fcec3d0071ef4a178412eeff3c01e4881f605d14bddea1b7d16be035d0ec5

See more details on using hashes here.

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