module Crypto.Secp256k1
(
Msg
, msg
, getMsg
, SecKey
, secKey
, getSecKey
, derivePubKey
, PubKey
, importPubKey
, exportPubKey
, Sig
, signMsg
, verifySig
, normalizeSig
, importSig
, laxImportSig
, exportSig
, CompactSig(..)
, exportCompactSig
, importCompactSig
, Tweak
, tweak
, getTweak
, tweakAddSecKey
, tweakMulSecKey
, tweakAddPubKey
, tweakMulPubKey
, combinePubKeys
) where
import Control.Monad
import Crypto.Secp256k1.Internal
import Data.ByteString (ByteString)
import qualified Data.ByteString as BS
import qualified Data.ByteString.Base16 as B16
import Data.Maybe
import Data.String
import Data.String.Conversions
import Foreign
import System.IO.Unsafe
import Test.QuickCheck
import Text.Read
newtype PubKey = PubKey (ForeignPtr PubKey64)
newtype Msg = Msg (ForeignPtr Msg32)
newtype Sig = Sig (ForeignPtr Sig64)
newtype SecKey = SecKey (ForeignPtr SecKey32)
newtype Tweak = Tweak (ForeignPtr Tweak32)
decodeHex :: ConvertibleStrings a ByteString => a -> Maybe ByteString
decodeHex str = if BS.null r then Just bs else Nothing where
(bs, r) = B16.decode $ cs str
instance Read PubKey where
readPrec = parens $ do
Ident "PubKey" <- lexP
String str <- lexP
maybe pfail return $ importPubKey =<< decodeHex str
instance IsString PubKey where
fromString = fromMaybe e . (importPubKey <=< decodeHex) where
e = error "Could not decode public key from hex string"
instance Show PubKey where
showsPrec d k = showParen (d > 10) $
showString "PubKey " . shows (B16.encode $ exportPubKey True k)
instance Read Msg where
readPrec = parens $ do
Ident "Msg" <- lexP
String str <- lexP
maybe pfail return $ msg =<< decodeHex str
instance IsString Msg where
fromString = fromMaybe e . (msg <=< decodeHex) where
e = error "Could not decode message from hex string"
instance Show Msg where
showsPrec d m = showParen (d > 10) $
showString "Msg " . shows (B16.encode $ getMsg m)
instance Read Sig where
readPrec = parens $ do
Ident "Sig" <- lexP
String str <- lexP
maybe pfail return $ importSig =<< decodeHex str
instance IsString Sig where
fromString = fromMaybe e . (importSig <=< decodeHex) where
e = error "Could not decode signature from hex string"
instance Show Sig where
showsPrec d s = showParen (d > 10) $
showString "Sig " . shows (B16.encode $ exportSig s)
instance Read SecKey where
readPrec = parens $ do
Ident "SecKey" <- lexP
String str <- lexP
maybe pfail return $ secKey =<< decodeHex str
instance IsString SecKey where
fromString = fromMaybe e . (secKey <=< decodeHex) where
e = error "Colud not decode secret key from hex string"
instance Show SecKey where
showsPrec d k = showParen (d > 10) $
showString "SecKey " . shows (B16.encode $ getSecKey k)
instance Read Tweak where
readPrec = parens $ do
Ident "Tweak" <- lexP
String str <- lexP
maybe pfail return $ tweak =<< decodeHex str
instance IsString Tweak where
fromString = fromMaybe e . (tweak <=< decodeHex) where
e = error "Could not decode tweak from hex string"
instance Show Tweak where
showsPrec d k = showParen (d > 10) $
showString "Tweak " . shows (B16.encode $ getTweak k)
instance Eq PubKey where
fp1 == fp2 = getPubKey fp1 == getPubKey fp2
instance Eq Msg where
fm1 == fm2 = getMsg fm1 == getMsg fm2
instance Eq Sig where
fg1 == fg2 = exportCompactSig fg1 == exportCompactSig fg2
instance Eq SecKey where
fk1 == fk2 = getSecKey fk1 == getSecKey fk2
instance Eq Tweak where
ft1 == ft2 = getTweak ft1 == getTweak ft2
msg :: ByteString -> Maybe Msg
msg bs
| BS.length bs == 32 = unsafePerformIO $ do
fp <- mallocForeignPtr
withForeignPtr fp $ flip poke (Msg32 bs)
return $ Just $ Msg fp
| otherwise = Nothing
secKey :: ByteString -> Maybe SecKey
secKey bs
| BS.length bs == 32 = unsafePerformIO $ do
fp <- mallocForeignPtr
ret <- withForeignPtr fp $ \p -> do
poke p (SecKey32 bs)
ecSecKeyVerify ctx p
if isSuccess ret
then return $ Just $ SecKey fp
else return Nothing
| otherwise = Nothing
normalizeSig :: Sig -> (Sig, Bool)
normalizeSig (Sig fg) = unsafePerformIO $ do
fg' <- mallocForeignPtr
ret <- withForeignPtr fg $ \pg -> withForeignPtr fg' $ \pg' ->
ecdsaSignatureNormalize ctx pg' pg
return (Sig fg', isSuccess ret)
tweak :: ByteString -> Maybe Tweak
tweak bs
| BS.length bs == 32 = unsafePerformIO $ do
fp <- mallocForeignPtr
withForeignPtr fp $ flip poke (Tweak32 bs)
return $ Just $ Tweak fp
| otherwise = Nothing
getSecKey :: SecKey -> ByteString
getSecKey (SecKey fk) = getSecKey32 $ unsafePerformIO $ withForeignPtr fk peek
getPubKey :: PubKey -> ByteString
getPubKey (PubKey fp) = getPubKey64 $ unsafePerformIO $ withForeignPtr fp peek
getMsg :: Msg -> ByteString
getMsg (Msg fm) = getMsg32 $ unsafePerformIO $ withForeignPtr fm peek
getTweak :: Tweak -> ByteString
getTweak (Tweak ft) = getTweak32 $ unsafePerformIO $ withForeignPtr ft peek
importPubKey :: ByteString -> Maybe PubKey
importPubKey bs = unsafePerformIO $ useByteString bs $ \(b, l) -> do
fp <- mallocForeignPtr
ret <- withForeignPtr fp $ \p -> ecPubKeyParse ctx p b l
if isSuccess ret then return $ Just $ PubKey fp else return Nothing
exportPubKey :: Bool -> PubKey -> ByteString
exportPubKey compress (PubKey pub) = unsafePerformIO $
withForeignPtr pub $ \p -> alloca $ \l -> allocaBytes z $ \o -> do
poke l (fromIntegral z)
ret <- ecPubKeySerialize ctx o l p c
unless (isSuccess ret) $ error "could not serialize public key"
n <- peek l
packByteString (o, n)
where
c = if compress then compressed else uncompressed
z = if compress then 33 else 65
exportCompactSig :: Sig -> CompactSig
exportCompactSig (Sig fg) = unsafePerformIO $
withForeignPtr fg $ \pg -> alloca $ \pc -> do
ret <- ecdsaSignatureSerializeCompact ctx pc pg
unless (isSuccess ret) $ error "Could not obtain compact signature"
peek pc
importCompactSig :: CompactSig -> Maybe Sig
importCompactSig c = unsafePerformIO $ alloca $ \pc -> do
poke pc c
fg <- mallocForeignPtr
ret <- withForeignPtr fg $ \pg -> ecdsaSignatureParseCompact ctx pg pc
if isSuccess ret then return $ Just $ Sig fg else return Nothing
importSig :: ByteString -> Maybe Sig
importSig bs = unsafePerformIO $
useByteString bs $ \(b, l) -> do
fg <- mallocForeignPtr
ret <- withForeignPtr fg $ \g -> ecdsaSignatureParseDer ctx g b l
if isSuccess ret then return $ Just $ Sig fg else return Nothing
laxImportSig :: ByteString -> Maybe Sig
laxImportSig bs = unsafePerformIO $
useByteString bs $ \(b, l) -> do
fg <- mallocForeignPtr
ret <- withForeignPtr fg $ \g -> laxDerParse ctx g b l
if isSuccess ret then return $ Just $ Sig fg else return Nothing
exportSig :: Sig -> ByteString
exportSig (Sig fg) = unsafePerformIO $
withForeignPtr fg $ \g -> alloca $ \l -> allocaBytes 72 $ \o -> do
poke l 72
ret <- ecdsaSignatureSerializeDer ctx o l g
unless (isSuccess ret) $ error "could not serialize signature"
n <- peek l
packByteString (o, n)
verifySig :: PubKey -> Sig -> Msg -> Bool
verifySig (PubKey fp) (Sig fg) (Msg fm) = unsafePerformIO $
withForeignPtr fp $ \p -> withForeignPtr fg $ \g ->
withForeignPtr fm $ \m -> isSuccess <$> ecdsaVerify ctx g m p
signMsg :: SecKey -> Msg -> Sig
signMsg (SecKey fk) (Msg fm) = unsafePerformIO $
withForeignPtr fk $ \k -> withForeignPtr fm $ \m -> do
fg <- mallocForeignPtr
ret <- withForeignPtr fg $ \g -> ecdsaSign ctx g m k nullFunPtr nullPtr
unless (isSuccess ret) $ error "could not sign message"
return $ Sig fg
derivePubKey :: SecKey -> PubKey
derivePubKey (SecKey fk) = unsafePerformIO $ withForeignPtr fk $ \k -> do
fp <- mallocForeignPtr
ret <- withForeignPtr fp $ \p -> ecPubKeyCreate ctx p k
unless (isSuccess ret) $ error "could not compute public key"
return $ PubKey fp
tweakAddSecKey :: SecKey -> Tweak -> Maybe SecKey
tweakAddSecKey (SecKey fk) (Tweak ft) = unsafePerformIO $
withForeignPtr fk $ \k -> withForeignPtr ft $ \t -> do
fk' <- mallocForeignPtr
ret <- withForeignPtr fk' $ \k' -> do
key <- peek k
poke k' key
ecSecKeyTweakAdd ctx k' t
if isSuccess ret then return $ Just $ SecKey fk' else return Nothing
tweakMulSecKey :: SecKey -> Tweak -> Maybe SecKey
tweakMulSecKey (SecKey fk) (Tweak ft) = unsafePerformIO $
withForeignPtr fk $ \k -> withForeignPtr ft $ \t -> do
fk' <- mallocForeignPtr
ret <- withForeignPtr fk' $ \k' -> do
key <- peek k
poke k' key
ecSecKeyTweakMul ctx k' t
if isSuccess ret then return $ Just $ SecKey fk' else return Nothing
tweakAddPubKey :: PubKey -> Tweak -> Maybe PubKey
tweakAddPubKey (PubKey fp) (Tweak ft) = unsafePerformIO $
withForeignPtr fp $ \p -> withForeignPtr ft $ \t -> do
fp' <- mallocForeignPtr
ret <- withForeignPtr fp' $ \p' -> do
pub <- peek p
poke p' pub
ecPubKeyTweakAdd ctx p' t
if isSuccess ret then return $ Just $ PubKey fp' else return Nothing
tweakMulPubKey :: PubKey -> Tweak -> Maybe PubKey
tweakMulPubKey (PubKey fp) (Tweak ft) = unsafePerformIO $
withForeignPtr fp $ \p -> withForeignPtr ft $ \t -> do
fp' <- mallocForeignPtr
ret <- withForeignPtr fp' $ \p' -> do
pub <- peek p
poke p' pub
ecPubKeyTweakMul ctx p' t
if isSuccess ret then return $ Just $ PubKey fp' else return Nothing
combinePubKeys :: [PubKey] -> Maybe PubKey
combinePubKeys pubs = unsafePerformIO $ pointers [] pubs $ \ps ->
allocaArray (length ps) $ \a -> do
pokeArray a ps
fp <- mallocForeignPtr
ret <- withForeignPtr fp $ \p ->
ecPubKeyCombine ctx p a (fromIntegral $ length ps)
if isSuccess ret
then return $ Just $ PubKey fp
else return Nothing
where
pointers ps [] f = f ps
pointers ps (PubKey fp : pubs') f =
withForeignPtr fp $ \p -> pointers (p:ps) pubs' f
instance Arbitrary Msg where
arbitrary = gen_msg
where
valid_bs = bs_gen `suchThat` isJust
bs_gen = (msg . BS.pack) <$> replicateM 32 arbitrary
gen_msg = fromJust <$> valid_bs
instance Arbitrary SecKey where
arbitrary = gen_key where
valid_bs = bs_gen `suchThat` isJust
bs_gen = (secKey . BS.pack) <$> replicateM 32 arbitrary
gen_key = fromJust <$> valid_bs
instance Arbitrary PubKey where
arbitrary = do
key <- arbitrary
return $ derivePubKey key