git-annex/Remote/Helper/Encryptable.hs

87 lines
2.8 KiB
Haskell
Raw Normal View History

2011-04-17 04:40:23 +00:00
{- common functions for encryptable remotes
-
- 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
import qualified Data.Map as M
2011-10-05 20:02:51 +00:00
import Common.Annex
import Types.Remote
import Crypto
import qualified Annex
import Config
{- 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"
(Just "none", Nothing) -> return c
(Just "none", Just _) -> error "Cannot change encryption type of existing remote."
(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
where
2011-04-17 22:18:27 +00:00
use m a = do
cipher <- liftIO a
2011-04-17 22:18:27 +00:00
showNote $ m ++ " " ++ describeCipher cipher
return $ M.delete "encryption" $ storeCipher c cipher
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)
-> ((Cipher, 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,
removeKey = withkey $ removeKey r,
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)
2011-05-15 16:25:58 +00:00
retrieve k f = cip k >>= maybe
(retrieveKeyFile r k f)
2011-07-15 16:47:14 +00:00
(`retrieveKeyFileEncrypted` f)
2011-05-15 16:25:58 +00:00
withkey a k = cip k >>= maybe (a k) (a . snd)
cip = cipherKey c
{- Gets encryption Cipher. The decrypted Ciphers are cached in the Annex
- state. -}
remoteCipher :: RemoteConfig -> Annex (Maybe Cipher)
remoteCipher c = go $ extractCipher c
2011-05-15 06:49:43 +00:00
where
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
{- 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)