CryptoSignatures.jl

codecov

CryptoSignatures.jl aims to be a versatile cryptographic signature library in Julia. Currently supports digital signature algorithm for all available elliptic curves in X9.62 specification. Implementation for modular prime groups is coming shortly.

ECDSA

The first step is to select a curve to make a cryptographic signature with an elliptic curve digital signature algorithm (ECDSA). Curves from X9.62 specification are already available in CryptoGroups.Specs module. For instance, an elliptic prime group with 192-bit length prime modulus, also known as secp192r1, can be instantiated as:

using CryptoSignatures
import CryptoGroups

curve = CryptoGroups.spec(:secp192r1)
ctx = DSAContext(curve, "sha1")

where ctx stores all relevant parameters on how to make and verify signatures. The second argument specifies a hash function name, which is forwarded to Nettle. In case hashing is done externally to avoid hashing twice, nothing can be passed as an argument like DSAContext(Curve_P_192, nothing).

To make a signature, first, we need to pick a key and calculate a corresponding public key:

private_key = CryptoSignatures.generate_key(ctx)
public_key = CryptoSignatures.public_key(ctx, private_key; mode = :uncompressed)

where public_key is stored as an octet in uncompressed notation, available are uncompressed, :compressed and :hybrid modes. Note that compressed mode for binary curves is limited as decompression is not implemented.

Let's say our message is M = "abc". That we can sign with a private key:

signature = CryptoSignatures.sign(ctx, Vector{UInt8}(M), private_key)

Note that the signature is issued with a k value derived deterministically with a pseudorandom number generator where a seed contains a message, private key and a global seed CryptoSignatures.SEED computed when module is loaded. A signature on a relative generator which can be done by passing it as an argument behind the message sign(ctx, message, generator, private_key).

The message can be verified with verify method using the public key and the issued signature:

CryptoSignatures.verify(ctx, Vector{UInt8}(M), public_key, signature) == true

returning true if the message had been issued by the owner of a public_key. In case the signature had been issued with a relative generator, it is verified as verify(ctx, message, generator, public_key).

DSA

To use an ordinary DSA with modular arithmetics, we need to instantiate the DSAContext. To do so, we need to select a prime modulus p for which we know group order q and generator g. With CryptoGroups we can generate those parameters and then use them for creating DSAContext:

using CryptoSignatures
import CryptoGroups 

group = CryptoGroups.spec(:RFC5114_2048_224)
ctx = DSAContext(group, "sha1")

As for ECDSA context, we generate a private key and a public key:

private_key = CryptoSignatures.generate_key(ctx)
public_key = CryptoSignatures.public_key(ctx, private_key)

Which can be used to sign and verify messages as before:

M = "abc"

signature = CryptoSignatures.sign(ctx, Vector{UInt8}(M), private_key)

verify(ctx, Vector{UInt8}(M), public_key, signature) == true

Security Considerations

It's important to state that the underlying implementation does not use constant time operations, thus making it vulnerable to side-channel attacks where the adversary can measure the time that it takes to make different signatures.

Another concern is that the implementation is slow, around 10...100 times more than state-of-the-art implementations in C. This can quickly become a bottleneck and attractive avenue for adversaries performing DDOS attacks.

It is also essential to state that only two tests are available for the signature algorithm. In practice, there are many attack vectors on how to fool improperly implemented verify function, which needs to be tested in detail.

In a nutshell, use it for small projects, but when you become big, don't shy away from the responsibility of including this library in your security audit to make it better.

Further Work

The performance could be addressed by wrapping the OpenSSL libcrypto library for doing operations on elliptic curves. RSA signatures could be something to add, as well as a blind signature algorithm.