sha512

Pure Haskell SHA-512, HMAC-SHA512 (docs.ppad.tech/sha512).
git clone git://git.ppad.tech/sha512.git
Log | Files | Refs | README | LICENSE

README.md (5296B)


      1 # ppad-sha512
      2 
      3 A pure Haskell implementation of SHA-512 and HMAC-SHA512 on strict and
      4 lazy ByteStrings, as specified by RFC's [6234][r6234] and [2104][r2104].
      5 
      6 ## Usage
      7 
      8 A sample GHCi session:
      9 
     10 ```
     11   > :set -XOverloadedStrings
     12   >
     13   > -- import qualified
     14   > import qualified Crypto.Hash.SHA512 as SHA512
     15   >
     16   > -- 'hash' and 'hmac' operate on strict bytestrings
     17   >
     18   > let hash_s = SHA512.hash "strict bytestring input"
     19   > let hmac_s = SHA512.hmac "strict secret" "strict bytestring input"
     20   >
     21   > -- 'hash_lazy' and 'hmac_lazy' operate on lazy bytestrings
     22   > -- but note that the key for HMAC is always strict
     23   >
     24   > let hash_l = SHA512.hash_lazy "lazy bytestring input"
     25   > let hmac_l = SHA512.hmac_lazy "strict secret" "lazy bytestring input"
     26   >
     27   > -- results are always unformatted 512-bit (64-byte) strict bytestrings
     28   >
     29   > import qualified Data.ByteString as BS
     30   >
     31   > BS.take 10 hash_s
     32   "\189D*\v\166\245N\216\&1\243"
     33   > BS.take 10 hmac_l
     34   "#}9\185\179\233[&\246\205"
     35   >
     36   > -- you can use third-party libraries for rendering if needed
     37   > -- e.g., using base64-bytestring:
     38   >
     39   > import qualified Data.ByteString.Base64 as B64
     40   >
     41   > B64.encode (BS.take 16 hash_s)
     42   "vUQqC6b1Ttgx8+ydx4MmtQ=="
     43   > B64.encode (BS.take 16 hmac_l)
     44   "I305ubPpWyb2zUi4pwDkrw=="
     45 ```
     46 
     47 ## Documentation
     48 
     49 Haddocks (API documentation, etc.) are hosted at
     50 [docs.ppad.tech/sha512][hadoc].
     51 
     52 ## Performance
     53 
     54 The aim is best-in-class performance for pure, highly-auditable Haskell
     55 code.
     56 
     57 Current benchmark figures on my mid-2020 MacBook Air look like (use
     58 `cabal bench` to run the benchmark suite):
     59 
     60 ```
     61   benchmarking ppad-sha512/SHA512 (32B input)/hash
     62   time                 1.820 μs   (1.798 μs .. 1.841 μs)
     63                        0.999 R²   (0.998 R² .. 0.999 R²)
     64   mean                 1.821 μs   (1.803 μs .. 1.846 μs)
     65   std dev              73.84 ns   (55.50 ns .. 103.6 ns)
     66   variance introduced by outliers: 55% (severely inflated)
     67 
     68   benchmarking ppad-sha512/SHA512 (32B input)/hash_lazy
     69   time                 1.760 μs   (1.737 μs .. 1.783 μs)
     70                        0.999 R²   (0.998 R² .. 0.999 R²)
     71   mean                 1.738 μs   (1.725 μs .. 1.757 μs)
     72   std dev              52.44 ns   (42.70 ns .. 74.57 ns)
     73   variance introduced by outliers: 40% (moderately inflated)
     74 
     75   benchmarking ppad-sha512/HMAC-SHA512 (32B input)/hmac
     76   time                 5.864 μs   (5.693 μs .. 6.024 μs)
     77                        0.997 R²   (0.995 R² .. 0.999 R²)
     78   mean                 5.779 μs   (5.719 μs .. 5.864 μs)
     79   std dev              241.8 ns   (184.5 ns .. 331.8 ns)
     80   variance introduced by outliers: 53% (severely inflated)
     81 
     82   benchmarking ppad-sha512/HMAC-SHA512 (32B input)/hmac_lazy
     83   time                 5.734 μs   (5.684 μs .. 5.791 μs)
     84                        0.999 R²   (0.999 R² .. 1.000 R²)
     85   mean                 5.741 μs   (5.688 μs .. 5.802 μs)
     86   std dev              189.8 ns   (153.6 ns .. 271.8 ns)
     87   variance introduced by outliers: 41% (moderately inflated)
     88 ```
     89 
     90 Compare this to Hackage's famous SHA package:
     91 
     92 ```
     93   benchmarking ppad-sha512/SHA512 (32B input)/SHA.sha512
     94   time                 3.198 μs   (3.133 μs .. 3.262 μs)
     95                        0.996 R²   (0.993 R² .. 0.998 R²)
     96   mean                 3.212 μs   (3.149 μs .. 3.300 μs)
     97   std dev              247.6 ns   (192.3 ns .. 347.0 ns)
     98   variance introduced by outliers: 81% (severely inflated)
     99 
    100   benchmarking ppad-sha512/HMAC-SHA512 (32B input)/SHA.hmacSha512
    101   time                 11.71 μs   (11.35 μs .. 12.08 μs)
    102                        0.992 R²   (0.988 R² .. 0.995 R²)
    103   mean                 11.68 μs   (11.39 μs .. 12.06 μs)
    104   std dev              1.036 μs   (871.1 ns .. 1.224 μs)
    105   variance introduced by outliers: 83% (severely inflated)
    106 ```
    107 
    108 Or the relevant SHA-512-based functions from a library with similar
    109 aims, [noble-hashes][noble] (though with no HMAC-SHA512 benchmark
    110 available):
    111 
    112 ```
    113 SHA512 32B x 217,296 ops/sec @ 4μs/op ± 2.00% (min: 3μs, max: 20ms)
    114 ```
    115 
    116 ## Security
    117 
    118 This library aims at the maximum security achievable in a
    119 garbage-collected language under an optimizing compiler such as GHC, in
    120 which strict constant-timeness can be challenging to achieve.
    121 
    122 The HMAC-SHA512 functions within pass all [Wycheproof vectors][wyche],
    123 as well as various other useful unit test vectors found around the
    124 internet.
    125 
    126 If you discover any vulnerabilities, please disclose them via
    127 security@ppad.tech.
    128 
    129 ## Development
    130 
    131 You'll require [Nix][nixos] with [flake][flake] support enabled. Enter a
    132 development shell with:
    133 
    134 ```
    135 $ nix develop
    136 ```
    137 
    138 Then do e.g.:
    139 
    140 ```
    141 $ cabal repl ppad-sha512
    142 ```
    143 
    144 to get a REPL for the main library.
    145 
    146 ## Attribution
    147 
    148 This implementation has benefitted immensely from the [SHA][hacka]
    149 package available on Hackage, which was used as a reference during
    150 development. Many parts wound up being direct translations.
    151 
    152 [nixos]: https://nixos.org/
    153 [flake]: https://nixos.org/manual/nix/unstable/command-ref/new-cli/nix3-flake.html
    154 [hadoc]: https://docs.ppad.tech/sha512
    155 [hacka]: https://hackage.haskell.org/package/SHA
    156 [r6234]: https://datatracker.ietf.org/doc/html/rfc6234
    157 [r2104]: https://datatracker.ietf.org/doc/html/rfc2104
    158 [noble]: https://github.com/paulmillr/noble-hashes
    159 [wyche]: https://github.com/C2SP/wycheproof