Implementation of the Digest HTTP headers according to RFC 3230.
Project description
Introduction
A small library to provide the server and client side methods to require, negotiation and generate Digest HTTP headers as per RFC 3230.
Clients can generate Digest headers of the form: Digest: SHA-256=xyz, MD5=abc. Server can require certain algorithms by sending Want-Digest headers of the form: Want-Digest: SHA-256, SHA;q=0.5, MD5;q=0.
Installation
Install using pip:
pip install rfc3230-digest-headers
Overview of the protocol
The protocol works as follows:
- Client and server agree on what the
instancebytes are for the endpoint in question. Usually the request body or the content of the resource before applying transformations. - Client sends request
- If the client did not directly send a valid
Digest, the server responds withWant-Digestheader to indicate which algorithms it supports.- Form of the
Want-Digestheader:Want-Digest: SHA-256, SHA;q=0.5, MD5;q=0 - The server can specify
qvaluesto indicate preference of algorithms. - No value equals
q=1.0. q=0means "do not use this algorithm".
- Form of the
- Client generates
Digestheader using one of the supported algorithms and sends it in the request.- Form of the
Digestheader:Digest: SHA-256=xyz, MD5=abc
- Form of the
- Server verifies the
Digestheader and processes the request.
Usage
Functional Interface (Recommended)
The recommended way to use this library is via the functional interface in rfc3230_digest_headers.functional. It provides simple functions for generating and verifying Digest headers:
create_digest(instance, ...): Generate a Digest header for client requests.verify_digest(request_headers, instance, ...): Verify Digest headers on the server.
Example: Basic Client-Server Flow
Simply import the two functions from the package (no need for .functional), and use them as follows:
from rfc3230_digest_headers import create_digest, verify_digest
# Client: prepare request
instance = b"Hello, World!"
digest_header = create_digest(instance)
request_headers = {digest_header.header_name: digest_header.header_value}
# Server: verify request
is_valid, want_digest = verify_digest(request_headers, instance)
if not is_valid and want_digest:
# Server responds with Want-Digest header
print(want_digest.header_name, want_digest.header_value)
# Client: handle Want-Digest and retry
if want_digest:
digest_header = create_digest(instance, want_digest_header=want_digest.header_value)
request_headers = {digest_header.header_name: digest_header.header_value}
# Server verifies again
is_valid, _ = verify_digest(request_headers, instance)
print("Accepted?", is_valid)
Configure which algorithms the server accepts
You can specify which algorithms the server accepts when verifying a Digest header by passing the qvalues parameter to verify_digest.
It should be a dictionary mapping DigestHeaderAlgorithm values to their respective q-values (float between 0.0 and 1.0, or None for default of 1.0).
from rfc3230_digest_headers import verify_digest, DigestHeaderAlgorithm
instance_bytes = b"Hello, World!"
request_headers = {"Digest": "SHA-256=..., MD5=..."}
is_valid, want_digest_header_should_be_added = verify_digest(
request_headers=request_headers,
instance=instance_bytes,
qvalues={
DigestHeaderAlgorithm.SHA256: 1.0,
DigestHeaderAlgorithm.SHA: 0.5,
DigestHeaderAlgorithm.MD5: 0.0 # If the client sends MD5, they will receive an error
},
)
print(is_valid) # True if the Digest header is valid
print(want_digest_header_should_be_added) # None if valid, otherwise contains the `Want-Digest` header to be sent to the client for negotiation
Configure which algorithms to use for generating the Digest header on the client
You can specify which algorithms to use when generating the Digest header by passing the algorithms parameter to create_digest.
It can be a list of DigestHeaderAlgorithm values, or the special values "auto" (to use the highest priority algorithm from a Want-Digest header) or "all" (to use all acceptable algorithms from a Want-Digest header).
from rfc3230_digest_headers import create_digest, DigestHeaderAlgorithm
instance = b"Hello, World!"
# Use only SHA-256 and MD5
digest_header = create_digest(
instance,
algorithms=[DigestHeaderAlgorithm.SHA256, DigestHeaderAlgorithm.MD5]
)
print(digest_header.header_name, digest_header.header_value)
The "auto" and "all" options are used when negotiating algorithms based on a Want-Digest header received from the server.
from rfc3230_digest_headers import create_digest, DigestHeaderAlgorithm
instance = b"Hello, World!"
want_digest_header_value = "SHA-256, SHA;q=0.5, MD5;q=0"
# Option 1: Use "auto" to select the highest priority algorithm
digest_header = create_digest(
instance,
algorithms="auto",
want_digest_header=want_digest_header_value
)
print(digest_header.header_name, digest_header.header_value) # Will use SHA-256
# Option 2: Use "all" to include all acceptable algorithms
digest_header = create_digest(
instance,
algorithms="all",
want_digest_header=want_digest_header_value
)
print(digest_header.header_name, digest_header.header_value) # Will use SHA-256 and SHA
Older enum-oriented Interface
These usage examples demonstrate the older enum-oriented interface directly on DigestHeaderAlgorithm.
Generate a Digest header
The client generates a Digest for their instance.
from rfc3230_digest_headers import DigestHeaderAlgorithm
instance_bytes = b"Hello, World!"
header = DigestHeaderAlgorithm.make_digest_header(
instance=instance_bytes,
algorithms=[DigestHeaderAlgorithm.SHA256, DigestHeaderAlgorithm.MD5]
)
print(header.header_name) # "Digest"
print(header.header_value) # "SHA-256=..., MD5=..."
Verify a Digest header
The server receives a request with a Digest header and verifies it.
from rfc3230_digest_headers import DigestHeaderAlgorithm
instance_bytes = b"Hello, World!"
request_headers = {"Digest": "SHA-256=..., MD5=..."}
is_valid, want_digest_header_should_be_added = DigestHeaderAlgorithm.verify_request(
request_headers=request_headers,
instance=instance_bytes,
qvalues={
DigestHeaderAlgorithm.SHA256: 1.0,
DigestHeaderAlgorithm.SHA: 0.5,
DigestHeaderAlgorithm.MD5: 0.0 # If the client sends MD5, they will receive an error
},
)
print(is_valid) # True if the Digest header is valid
print(want_digest_header_should_be_added) # None if valid, otherwise contains the `Want-Digest` header to be sent to the client for negotiation
Server-side negotiation of algorithms
The server can indicate which algorithms the endpoint requires by sending a Want-Digest header. The header is automatically generated when attempting to verify invalid request headers. In the following example, the client sends a Digest header with an unsupported algorithm (MD5 with a q-value of 0.0), so the server responds with a Want-Digest header indicating which algorithms are supported.
from rfc3230_digest_headers import DigestHeaderAlgorithm
# Fake request from client without an invalid Digest header
instance_bytes = b"Hello, World!"
request_headers = {"Digest": "SHA-256=..., MD5=..."}
is_valid, want_digest_header_should_be_added = DigestHeaderAlgorithm.verify_request(
request_headers=request_headers,
instance=instance_bytes,
qvalues={
DigestHeaderAlgorithm.SHA256: 1.0,
DigestHeaderAlgorithm.SHA: 0.5,
DigestHeaderAlgorithm.MD5: 0.0 # If the client sends MD5, they will receive an error
},
)
if want_digest_header_should_be_added:
print(want_digest_header_should_be_added.header_name) # "Want-Digest"
print(want_digest_header_should_be_added.header_value) # "SHA-256, SHA;q=0.5, MD5;q=0"
# Send the response with the generated Want-Digest header
...
Client-side negotiation of algorithms
When an endpoint responds with a Want-Digest header, the client can parse it and generate a valid Digest header. In the following example, imagine that we initially sent a request with b'Hello, World!' as body, and the server responded with an HTTP error code and a Want-Digest header. The client sees that its original request failed, and that the server wants a Digest header. The client then generates a valid Digest header using the highest priority algorithm from the Want-Digest header and re-sends the request.
from rfc3230_digest_headers import DigestHeaderAlgorithm
instance_bytes = b"Hello, World!"
# Exemplary Want-Digest header from server's response
want_digest_header_value = "SHA-256, SHA;q=0.5, MD5;q=0"
# Option 1: Use make_digest_header with the want_digest_header parameter
# This will automatically handle negotiation
header = DigestHeaderAlgorithm.make_digest_header(
instance=instance_bytes,
algorithms="auto", # Use the highest priority algorithm from Want-Digest
want_digest_header=want_digest_header_value
)
print(header.header_name) # "Digest"
print(header.header_value) # "sha-256=..."
# Option 2: Explicitly use handle_want_digest_header (legacy approach)
header = DigestHeaderAlgorithm.handle_want_digest_header(
instance=instance_bytes,
want_digest_header=want_digest_header_value,
algorithms="auto" # Use the highest priority algorithm from Want-Digest
)
print(header.header_name) # "Digest"
print(header.header_value) # "sha-256=..."
# re-send the request with the generated Digest header
...
You can also use algorithms="all" to include all acceptable algorithms from the Want-Digest header, or provide an explicit list like algorithms=[DigestHeaderAlgorithm.SHA256, DigestHeaderAlgorithm.MD5] to only use specific algorithms that you support.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file rfc3230_digest_headers-1.1.4.tar.gz.
File metadata
- Download URL: rfc3230_digest_headers-1.1.4.tar.gz
- Upload date:
- Size: 9.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
01dc4670dd2af3841fd3f108637e43929961758fd3f57e05f3ca0b3401f4720c
|
|
| MD5 |
1d4a4c02d2d0f1e1e7924c404092219e
|
|
| BLAKE2b-256 |
428e6045459a025dd177e3f428d542b325cb3cb0f2af0b518ad7390b6af357d8
|
Provenance
The following attestation bundles were made for rfc3230_digest_headers-1.1.4.tar.gz:
Publisher:
publish.yml on Mari6814/py-rfc3230-digest-headers
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
rfc3230_digest_headers-1.1.4.tar.gz -
Subject digest:
01dc4670dd2af3841fd3f108637e43929961758fd3f57e05f3ca0b3401f4720c - Sigstore transparency entry: 723871257
- Sigstore integration time:
-
Permalink:
Mari6814/py-rfc3230-digest-headers@93dc96ad32c57ba69ae57fb47c0ec60d8971d047 -
Branch / Tag:
refs/tags/v1.1.4 - Owner: https://github.com/Mari6814
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@93dc96ad32c57ba69ae57fb47c0ec60d8971d047 -
Trigger Event:
release
-
Statement type:
File details
Details for the file rfc3230_digest_headers-1.1.4-py3-none-any.whl.
File metadata
- Download URL: rfc3230_digest_headers-1.1.4-py3-none-any.whl
- Upload date:
- Size: 11.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7d309476bafa0e96223732e543049af2e905bcf0b44fc561bbffc8c26e23eed1
|
|
| MD5 |
45ac9ed4b63525ac83a6a27c8354d4e1
|
|
| BLAKE2b-256 |
2a9a61bcbbe8eb002fa9a02878b01246ed0582b66b71c9d88a97fe9cffafabfe
|
Provenance
The following attestation bundles were made for rfc3230_digest_headers-1.1.4-py3-none-any.whl:
Publisher:
publish.yml on Mari6814/py-rfc3230-digest-headers
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
rfc3230_digest_headers-1.1.4-py3-none-any.whl -
Subject digest:
7d309476bafa0e96223732e543049af2e905bcf0b44fc561bbffc8c26e23eed1 - Sigstore transparency entry: 723871260
- Sigstore integration time:
-
Permalink:
Mari6814/py-rfc3230-digest-headers@93dc96ad32c57ba69ae57fb47c0ec60d8971d047 -
Branch / Tag:
refs/tags/v1.1.4 - Owner: https://github.com/Mari6814
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@93dc96ad32c57ba69ae57fb47c0ec60d8971d047 -
Trigger Event:
release
-
Statement type: