2011-04-17 04:40:23 +00:00
|
|
|
{- common functions for encryptable remotes
|
2011-04-16 17:25:27 +00:00
|
|
|
-
|
|
|
|
- Copyright 2011 Joey Hess <joey@kitenet.net>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
2011-08-17 00:49:54 +00:00
|
|
|
module Remote.Helper.Encryptable where
|
2011-04-16 17:25:27 +00:00
|
|
|
|
|
|
|
import qualified Data.Map as M
|
|
|
|
|
2011-10-05 20:02:51 +00:00
|
|
|
import Common.Annex
|
2011-06-02 01:56:04 +00:00
|
|
|
import Types.Remote
|
2011-04-16 17:25:27 +00:00
|
|
|
import Crypto
|
2011-04-16 22:22:52 +00:00
|
|
|
import qualified Annex
|
2011-04-17 05:13:21 +00:00
|
|
|
import Config
|
2011-04-16 17:25:27 +00:00
|
|
|
|
|
|
|
{- Encryption setup for a remote. The user must specify whether to use
|
|
|
|
- an encryption key, or not encrypt. An encrypted cipher is created, or is
|
|
|
|
- updated to be accessible to an additional encryption key. -}
|
|
|
|
encryptionSetup :: RemoteConfig -> Annex RemoteConfig
|
|
|
|
encryptionSetup c =
|
|
|
|
case (M.lookup "encryption" c, extractCipher c) of
|
|
|
|
(Nothing, Nothing) -> error "Specify encryption=key or encryption=none"
|
2011-04-16 23:45:59 +00:00
|
|
|
(Just "none", Nothing) -> return c
|
|
|
|
(Just "none", Just _) -> error "Cannot change encryption type of existing remote."
|
2011-04-16 17:25:27 +00:00
|
|
|
(Nothing, Just _) -> return c
|
2011-04-17 22:18:27 +00:00
|
|
|
(Just _, Nothing) -> use "encryption setup" $ genCipher c
|
|
|
|
(Just _, Just v) -> use "encryption updated" $ updateCipher c v
|
2011-04-16 17:25:27 +00:00
|
|
|
where
|
2011-04-17 22:18:27 +00:00
|
|
|
use m a = do
|
2011-04-16 17:25:27 +00:00
|
|
|
cipher <- liftIO a
|
2011-04-17 22:18:27 +00:00
|
|
|
showNote $ m ++ " " ++ describeCipher cipher
|
2011-04-16 17:25:27 +00:00
|
|
|
return $ M.delete "encryption" $ storeCipher c cipher
|
2011-04-16 22:22:52 +00:00
|
|
|
|
2011-04-17 01:41:14 +00:00
|
|
|
{- Modifies a Remote to support encryption.
|
|
|
|
-
|
|
|
|
- Two additional functions must be provided by the remote,
|
|
|
|
- to support storing and retrieving encrypted content. -}
|
2011-04-17 04:40:23 +00:00
|
|
|
encryptableRemote
|
2011-04-17 01:41:14 +00:00
|
|
|
:: Maybe RemoteConfig
|
|
|
|
-> ((Cipher, Key) -> Key -> Annex Bool)
|
2012-03-04 07:36:39 +00:00
|
|
|
-> ((Cipher, Key) -> Key -> FilePath -> Annex Bool)
|
2011-12-31 08:11:39 +00:00
|
|
|
-> Remote
|
|
|
|
-> Remote
|
2011-04-17 04:40:23 +00:00
|
|
|
encryptableRemote c storeKeyEncrypted retrieveKeyFileEncrypted r =
|
2011-04-17 01:41:14 +00:00
|
|
|
r {
|
|
|
|
storeKey = store,
|
|
|
|
retrieveKeyFile = retrieve,
|
2012-01-20 17:23:11 +00:00
|
|
|
retrieveKeyFileCheap = retrieveCheap,
|
2011-04-17 01:41:14 +00:00
|
|
|
removeKey = withkey $ removeKey r,
|
2011-04-17 05:13:21 +00:00
|
|
|
hasKey = withkey $ hasKey r,
|
|
|
|
cost = cost r + encryptedRemoteCostAdj
|
2011-04-17 01:41:14 +00:00
|
|
|
}
|
|
|
|
where
|
2011-05-15 16:25:58 +00:00
|
|
|
store k = cip k >>= maybe
|
|
|
|
(storeKey r k)
|
2011-07-15 16:47:14 +00:00
|
|
|
(`storeKeyEncrypted` k)
|
2012-01-20 17:23:11 +00:00
|
|
|
retrieve k f = cip k >>= maybe
|
|
|
|
(retrieveKeyFile r k f)
|
2012-03-04 07:36:39 +00:00
|
|
|
(\enck -> retrieveKeyFileEncrypted enck k f)
|
2012-01-20 17:23:11 +00:00
|
|
|
retrieveCheap k f = cip k >>= maybe
|
|
|
|
(retrieveKeyFileCheap r k f)
|
|
|
|
(\_ -> return False)
|
2011-05-15 16:25:58 +00:00
|
|
|
withkey a k = cip k >>= maybe (a k) (a . snd)
|
|
|
|
cip = cipherKey c
|
2011-04-16 22:22:52 +00:00
|
|
|
|
2011-12-08 20:01:46 +00:00
|
|
|
{- Gets encryption Cipher. The decrypted Ciphers are cached in the Annex
|
2011-05-01 18:05:10 +00:00
|
|
|
- state. -}
|
|
|
|
remoteCipher :: RemoteConfig -> Annex (Maybe Cipher)
|
2011-12-08 20:01:46 +00:00
|
|
|
remoteCipher c = go $ extractCipher c
|
2011-05-15 06:49:43 +00:00
|
|
|
where
|
2011-12-08 20:01:46 +00:00
|
|
|
go Nothing = return Nothing
|
|
|
|
go (Just encipher) = do
|
|
|
|
cache <- Annex.getState Annex.ciphers
|
|
|
|
case M.lookup encipher cache of
|
|
|
|
Just cipher -> return $ Just cipher
|
|
|
|
Nothing -> decrypt encipher cache
|
|
|
|
decrypt encipher cache = do
|
|
|
|
showNote "gpg"
|
|
|
|
cipher <- liftIO $ decryptCipher c encipher
|
|
|
|
Annex.changeState (\s -> s { Annex.ciphers = M.insert encipher cipher cache })
|
|
|
|
return $ Just cipher
|
2011-05-01 18:05:10 +00:00
|
|
|
|
|
|
|
{- Gets encryption Cipher, and encrypted version of Key. -}
|
|
|
|
cipherKey :: Maybe RemoteConfig -> Key -> Annex (Maybe (Cipher, Key))
|
|
|
|
cipherKey Nothing _ = return Nothing
|
2011-10-11 18:43:45 +00:00
|
|
|
cipherKey (Just c) k = maybe Nothing encrypt <$> remoteCipher c
|
2011-05-15 16:25:58 +00:00
|
|
|
where
|
2011-10-11 18:43:45 +00:00
|
|
|
encrypt ciphertext = Just (ciphertext, encryptKey ciphertext k)
|