module SSH.Crypto where import Codec.Utils (fromOctets, i2osp) import Control.Monad (replicateM) import Control.Monad.Trans.State import Data.Digest.Pure.SHA (bytestringDigest, sha1) import Data.Int import qualified Codec.Crypto.RSA as RSA import qualified Data.ByteString.Lazy as LBS import qualified OpenSSL.DSA as DSA import SSH.Packet import SSH.NetReader import SSH.Util (strictLBS) data Cipher = Cipher { cType :: CipherType , cMode :: CipherMode , cBlockSize :: Int , cKeySize :: Int } data CipherType = AES data CipherMode = CBC data HMAC = HMAC { hDigestSize :: Int , hFunction :: LBS.ByteString -> LBS.ByteString } data PublicKey = RSAPublicKey { rpubE :: Integer , rpubN :: Integer } | DSAPublicKey { dpubP :: Integer , dpubQ :: Integer , dpubG :: Integer , dpubY :: Integer } deriving (Eq, Show) data KeyPair = RSAKeyPair { rprivPub :: PublicKey , rprivD :: Integer } | DSAKeyPair { dprivPub :: PublicKey , dprivX :: Integer } deriving (Eq, Show) generator :: Integer generator = 2 safePrime :: Integer safePrime = 179769313486231590770839156793787453197860296048756011706444423684197180216158519368947833795864925541502180565485980503646440548199239100050792877003355816639229553136239076508735759914822574862575007425302077447712589550957937778424442426617334727629299387668709205606050270810842907692932019128194467627007 toBlocks :: (Integral a, Integral b) => a -> LBS.ByteString -> [b] toBlocks _ m | m == LBS.empty = [] toBlocks bs m = b : rest where b = fromOctets (256 :: Integer) (LBS.unpack (LBS.take (fromIntegral bs) m)) rest = toBlocks bs (LBS.drop (fromIntegral bs) m) fromBlocks :: Integral a => Int -> [a] -> LBS.ByteString fromBlocks bs = LBS.concat . map (LBS.pack . i2osp bs) blob :: PublicKey -> LBS.ByteString blob (RSAPublicKey e n) = doPacket $ do string "ssh-rsa" integer e integer n blob (DSAPublicKey p q g y) = doPacket $ do string "ssh-dss" integer p integer q integer g integer y blobToKey :: LBS.ByteString -> PublicKey blobToKey s = flip evalState s $ do t <- readString case t of "ssh-rsa" -> do e <- readInteger n <- readInteger return $ RSAPublicKey e n "ssh-dss" -> do [p, q, g, y] <- replicateM 4 readInteger return $ DSAPublicKey p q g y u -> error $ "unknown public key format: " ++ u sign :: KeyPair -> LBS.ByteString -> IO LBS.ByteString sign (RSAKeyPair (RSAPublicKey _ n) d) m = return $ LBS.concat [ netString "ssh-rsa" , netLBS (RSA.rsassa_pkcs1_v1_5_sign RSA.ha_SHA1 (RSA.PrivateKey 256 n d) m) ] sign (DSAKeyPair (DSAPublicKey p q g y) x) m = do (r, s) <- DSA.signDigestedDataWithDSA (DSA.tupleToDSAKeyPair (p, q, g, y, x)) digest return $ LBS.concat [ netString "ssh-dss" , netLBS $ LBS.concat [ LBS.pack $ i2osp 20 r , LBS.pack $ i2osp 20 s ] ] where digest = strictLBS . bytestringDigest . sha1 $ m