Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions CHANGELOG.rst
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,8 @@ Changelog
method for computing hashes.
* Added :doc:`/hazmat/primitives/hpke` support implementing :rfc:`9180` for
hybrid authenticated encryption.
* Added new :doc:`/hazmat/primitives/asymmetric/mldsa` module with
support for ML-DSA-65 signing and verification with the AWS-LC backend.

.. _v46-0-6:

Expand Down
4 changes: 4 additions & 0 deletions docs/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,10 @@
"sphinx_inline_tabs",
]

doctest_global_setup = """
from cryptography.hazmat.backends.openssl.backend import backend as _backend
"""

if spelling is not None:
extensions.append("sphinxcontrib.spelling")

Expand Down
1 change: 1 addition & 0 deletions docs/hazmat/primitives/asymmetric/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ private key is able to decrypt it.
.. toctree::
:maxdepth: 1

mldsa
ed25519
x25519
ed448
Expand Down
231 changes: 231 additions & 0 deletions docs/hazmat/primitives/asymmetric/mldsa.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,231 @@
.. hazmat::

ML-DSA signing
=================

.. currentmodule:: cryptography.hazmat.primitives.asymmetric.mldsa

ML-DSA is a post-quantum digital signature algorithm based on module
lattices, standardized in `FIPS 204`_.

Signing & Verification
~~~~~~~~~~~~~~~~~~~~~~~

.. doctest::
:skipif: not _backend.mldsa_supported()

>>> from cryptography.hazmat.primitives.asymmetric.mldsa import MlDsa65PrivateKey
>>> private_key = MlDsa65PrivateKey.generate()
>>> signature = private_key.sign(b"my authenticated message")
>>> public_key = private_key.public_key()
>>> public_key.verify(signature, b"my authenticated message")

Context-based Signing & Verification
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

ML-DSA supports context strings to bind additional information to signatures.
The context can be up to 255 bytes and is used to differentiate signatures in
different contexts or protocols.

.. doctest::
:skipif: not _backend.mldsa_supported()

>>> from cryptography.hazmat.primitives.asymmetric.mldsa import MlDsa65PrivateKey
>>> private_key = MlDsa65PrivateKey.generate()
>>> context = b"email-signature-v1"
>>> signature = private_key.sign(b"my authenticated message", context)
>>> public_key = private_key.public_key()
>>> # Verification requires the same context
>>> public_key.verify(signature, b"my authenticated message", context)

Key interfaces
~~~~~~~~~~~~~~

.. class:: MlDsa65PrivateKey

.. versionadded:: 47.0

.. classmethod:: generate()

Generate an ML-DSA-65 private key.

:returns: :class:`MlDsa65PrivateKey`

:raises cryptography.exceptions.UnsupportedAlgorithm: If ML-DSA-65 is
not supported by the backend ``cryptography`` is using.

.. classmethod:: from_seed_bytes(data)

Load an ML-DSA-65 private key from seed bytes.

:param data: 32 byte seed.
:type data: :term:`bytes-like`

:returns: :class:`MlDsa65PrivateKey`

:raises ValueError: If the seed is not 32 bytes.

:raises cryptography.exceptions.UnsupportedAlgorithm: If ML-DSA-65 is
not supported by the backend ``cryptography`` is using.

.. doctest::
:skipif: not _backend.mldsa_supported()

>>> from cryptography.hazmat.primitives.asymmetric import mldsa
>>> private_key = mldsa.MlDsa65PrivateKey.generate()
>>> seed = private_key.private_bytes_raw()
>>> same_key = mldsa.MlDsa65PrivateKey.from_seed_bytes(seed)

.. method:: public_key()

:returns: :class:`MlDsa65PublicKey`

.. method:: sign(data, context=None)

Sign the data using ML-DSA-65. An optional context string can be
provided.

:param data: The data to sign.
:type data: :term:`bytes-like`

:param context: An optional context string (up to 255 bytes).
:type context: :term:`bytes-like` or ``None``

:returns bytes: The signature (3309 bytes).

:raises ValueError: If the context is longer than 255 bytes.

.. method:: private_bytes(encoding, format, encryption_algorithm)

Allows serialization of the key to bytes. Encoding (
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.PEM`,
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.DER`, or
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.Raw`) and
format (
:attr:`~cryptography.hazmat.primitives.serialization.PrivateFormat.PKCS8`
or
:attr:`~cryptography.hazmat.primitives.serialization.PrivateFormat.Raw`
) are chosen to define the exact serialization.

This method only returns the serialization of the seed form of the
private key, never the expanded one.

:param encoding: A value from the
:class:`~cryptography.hazmat.primitives.serialization.Encoding` enum.

:param format: A value from the
:class:`~cryptography.hazmat.primitives.serialization.PrivateFormat`
enum. If the ``encoding`` is
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.Raw`
then ``format`` must be
:attr:`~cryptography.hazmat.primitives.serialization.PrivateFormat.Raw`
, otherwise it must be
:attr:`~cryptography.hazmat.primitives.serialization.PrivateFormat.PKCS8`.

:param encryption_algorithm: An instance of an object conforming to the
:class:`~cryptography.hazmat.primitives.serialization.KeySerializationEncryption`
interface.

:return bytes: Serialized key.

.. method:: private_bytes_raw()

Allows serialization of the key to raw bytes. This method is a
convenience shortcut for calling :meth:`private_bytes` with
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.Raw`
encoding,
:attr:`~cryptography.hazmat.primitives.serialization.PrivateFormat.Raw`
format, and
:class:`~cryptography.hazmat.primitives.serialization.NoEncryption`.

This method only returns the seed form of the private key (32 bytes).

:return bytes: Raw key (32-byte seed).

.. class:: MlDsa65PublicKey

.. versionadded:: 47.0

.. classmethod:: from_public_bytes(data)

:param bytes data: 1952 byte public key.

:returns: :class:`MlDsa65PublicKey`

:raises ValueError: If the public key is not 1952 bytes.

:raises cryptography.exceptions.UnsupportedAlgorithm: If ML-DSA-65 is
not supported by the backend ``cryptography`` is using.

.. doctest::
:skipif: not _backend.mldsa_supported()

>>> from cryptography.hazmat.primitives import serialization
>>> from cryptography.hazmat.primitives.asymmetric import mldsa
>>> private_key = mldsa.MlDsa65PrivateKey.generate()
>>> public_key = private_key.public_key()
>>> public_bytes = public_key.public_bytes(
... encoding=serialization.Encoding.Raw,
... format=serialization.PublicFormat.Raw
... )
>>> loaded_public_key = mldsa.MlDsa65PublicKey.from_public_bytes(public_bytes)

.. method:: public_bytes(encoding, format)

Allows serialization of the key to bytes. Encoding (
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.PEM`,
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.DER`, or
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.Raw`) and
format (
:attr:`~cryptography.hazmat.primitives.serialization.PublicFormat.SubjectPublicKeyInfo`
or
:attr:`~cryptography.hazmat.primitives.serialization.PublicFormat.Raw`
) are chosen to define the exact serialization.

:param encoding: A value from the
:class:`~cryptography.hazmat.primitives.serialization.Encoding` enum.

:param format: A value from the
:class:`~cryptography.hazmat.primitives.serialization.PublicFormat`
enum. If the ``encoding`` is
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.Raw`
then ``format`` must be
:attr:`~cryptography.hazmat.primitives.serialization.PublicFormat.Raw`
, otherwise it must be
:attr:`~cryptography.hazmat.primitives.serialization.PublicFormat.SubjectPublicKeyInfo`.

:returns bytes: The public key bytes.

.. method:: public_bytes_raw()

Allows serialization of the key to raw bytes. This method is a
convenience shortcut for calling :meth:`public_bytes` with
:attr:`~cryptography.hazmat.primitives.serialization.Encoding.Raw`
encoding and
:attr:`~cryptography.hazmat.primitives.serialization.PublicFormat.Raw`
format.

:return bytes: 1952-byte raw public key.

.. method:: verify(signature, data, context=None)

Verify a signature using ML-DSA-65. If a context string was used during
signing, the same context must be provided for verification to succeed.

:param signature: The signature to verify.
:type signature: :term:`bytes-like`

:param data: The data to verify.
:type data: :term:`bytes-like`

:param context: An optional context string (up to 255 bytes) that was
used during signing.
:type context: :term:`bytes-like` or ``None``

:returns: None
:raises cryptography.exceptions.InvalidSignature: Raised when the
signature cannot be verified.
:raises ValueError: If the context is longer than 255 bytes.


.. _`FIPS 204`: https://csrc.nist.gov/pubs/fips/204/final