S3: Added progress display for uploading and downloading.

This commit is contained in:
Joey Hess 2012-11-18 22:49:07 -04:00
parent 7ade03bd5f
commit 7df1e71fe3
7 changed files with 46 additions and 41 deletions

View file

@ -12,6 +12,7 @@ import Types.Meters
import Utility.Observed import Utility.Observed
import qualified Data.ByteString.Lazy as L import qualified Data.ByteString.Lazy as L
import qualified Data.ByteString as S
{- Sends the content of a file to an action, updating the meter as it's {- Sends the content of a file to an action, updating the meter as it's
- consumed. -} - consumed. -}
@ -21,5 +22,19 @@ withMeteredFile f meterupdate a = withBinaryFile f ReadMode $ \h ->
{- Sends the content of a file to a Handle, updating the meter as it's {- Sends the content of a file to a Handle, updating the meter as it's
- written. -} - written. -}
sendMeteredFile :: FilePath -> MeterUpdate -> Handle -> IO () streamMeteredFile :: FilePath -> MeterUpdate -> Handle -> IO ()
sendMeteredFile f meterupdate h = withMeteredFile f meterupdate $ L.hPut h streamMeteredFile f meterupdate h = withMeteredFile f meterupdate $ L.hPut h
{- Writes a ByteString to a Handle, updating a meter as it's written. -}
meteredWrite :: MeterUpdate -> Handle -> L.ByteString -> IO ()
meteredWrite meterupdate h = go . L.toChunks
where
go [] = return ()
go (c:cs) = do
S.hPut h c
meterupdate $ toInteger $ S.length c
go cs
meteredWriteFile :: MeterUpdate -> FilePath -> L.ByteString -> IO ()
meteredWriteFile meterupdate f b = withBinaryFile f WriteMode $ \h ->
meteredWrite meterupdate h b

View file

@ -11,6 +11,7 @@ import qualified Data.ByteString.Lazy as L
import qualified Data.ByteString as S import qualified Data.ByteString as S
import qualified Data.Map as M import qualified Data.Map as M
import qualified Control.Exception as E import qualified Control.Exception as E
import Data.Int
import Common.Annex import Common.Annex
import Types.Remote import Types.Remote
@ -21,8 +22,8 @@ import Remote.Helper.Special
import Remote.Helper.Encryptable import Remote.Helper.Encryptable
import Remote.Helper.Chunked import Remote.Helper.Chunked
import Crypto import Crypto
import Data.Int
import Annex.Content import Annex.Content
import Meters
remote :: RemoteType remote :: RemoteType
remote = RemoteType { remote = RemoteType {

View file

@ -10,10 +10,10 @@ module Remote.Helper.Chunked where
import Common.Annex import Common.Annex
import Utility.DataUnits import Utility.DataUnits
import Types.Remote import Types.Remote
import Meters
import qualified Data.Map as M import qualified Data.Map as M
import qualified Data.ByteString.Lazy as L import qualified Data.ByteString.Lazy as L
import qualified Data.ByteString as S
import Data.Int import Data.Int
import qualified Control.Exception as E import qualified Control.Exception as E
@ -121,25 +121,10 @@ storeChunked chunksize dests storer content =
storer d chunk storer d chunk
storechunks sz (d:useddests) ds b' storechunks sz (d:useddests) ds b'
{- Write a L.ByteString to a file, updating a progress meter {- Writes a series of chunks to a file. The feeder is called to get
- after each chunk of the L.ByteString, typically every 64 kb or so. -} - each chunk. -}
meteredWriteFile :: MeterUpdate -> FilePath -> L.ByteString -> IO ()
meteredWriteFile meterupdate dest b =
meteredWriteFileChunks meterupdate dest [b] return
{- Writes a series of major chunks to a file. The feeder is called to get
- each major chunk. Then each chunk of the L.ByteString is written,
- with the meter updated after each chunk. -}
meteredWriteFileChunks :: MeterUpdate -> FilePath -> [v] -> (v -> IO L.ByteString) -> IO () meteredWriteFileChunks :: MeterUpdate -> FilePath -> [v] -> (v -> IO L.ByteString) -> IO ()
meteredWriteFileChunks meterupdate dest chunks feeder = meteredWriteFileChunks meterupdate dest chunks feeder =
E.bracket (openFile dest WriteMode) hClose (feed chunks []) withBinaryFile dest WriteMode $ \h ->
where forM_ chunks $ \c ->
feed [] [] _ = noop meteredWrite meterupdate h =<< feeder c
feed (c:cs) [] h = do
bs <- L.toChunks <$> feeder c
unless (null bs) $
feed cs bs h
feed cs (b:bs) h = do
S.hPut h b
meterupdate $ toInteger $ S.length b
feed cs bs h

View file

@ -152,26 +152,29 @@ storeHelper (conn, bucket) r k p file = do
isxheader (h, _) = "x-amz-" `isPrefixOf` h isxheader (h, _) = "x-amz-" `isPrefixOf` h
retrieve :: Remote -> Key -> AssociatedFile -> FilePath -> Annex Bool retrieve :: Remote -> Key -> AssociatedFile -> FilePath -> Annex Bool
retrieve r k _f d = s3Action r False $ \(conn, bucket) -> do retrieve r k _f d = s3Action r False $ \(conn, bucket) ->
res <- liftIO $ getObject conn $ bucketKey r bucket k metered Nothing k $ \meterupdate -> do
case res of res <- liftIO $ getObject conn $ bucketKey r bucket k
Right o -> do case res of
liftIO $ L.writeFile d $ obj_data o Right o -> do
return True liftIO $ meteredWriteFile meterupdate d $
Left e -> s3Warning e obj_data o
return True
Left e -> s3Warning e
retrieveCheap :: Remote -> Key -> FilePath -> Annex Bool retrieveCheap :: Remote -> Key -> FilePath -> Annex Bool
retrieveCheap _ _ _ = return False retrieveCheap _ _ _ = return False
retrieveEncrypted :: Remote -> (Cipher, Key) -> Key -> FilePath -> Annex Bool retrieveEncrypted :: Remote -> (Cipher, Key) -> Key -> FilePath -> Annex Bool
retrieveEncrypted r (cipher, enck) _ f = s3Action r False $ \(conn, bucket) -> do retrieveEncrypted r (cipher, enck) k d = s3Action r False $ \(conn, bucket) ->
res <- liftIO $ getObject conn $ bucketKey r bucket enck metered Nothing k $ \meterupdate -> do
case res of res <- liftIO $ getObject conn $ bucketKey r bucket enck
Right o -> liftIO $ decrypt cipher (feedBytes $ obj_data o) $ case res of
readBytes $ \content -> do Right o -> liftIO $ decrypt cipher (\h -> meteredWrite meterupdate h $ obj_data o) $
L.writeFile f content readBytes $ \content -> do
return True L.writeFile d content
Left e -> s3Warning e return True
Left e -> s3Warning e
remove :: Remote -> Key -> Annex Bool remove :: Remote -> Key -> Annex Bool
remove r k = s3Action r False $ \(conn, bucket) -> do remove r k = s3Action r False $ \(conn, bucket) -> do

View file

@ -97,7 +97,7 @@ storeEncrypted r (cipher, enck) k p = metered (Just p) k $ \meterupdate ->
davAction r False $ \(baseurl, user, pass) -> do davAction r False $ \(baseurl, user, pass) -> do
let url = davLocation baseurl enck let url = davLocation baseurl enck
f <- inRepo $ gitAnnexLocation k f <- inRepo $ gitAnnexLocation k
liftIO $ encrypt cipher (sendMeteredFile f meterupdate) $ liftIO $ encrypt cipher (streamMeteredFile f meterupdate) $
readBytes $ storeHelper r url user pass readBytes $ storeHelper r url user pass
storeHelper :: Remote -> DavUrl -> DavUser -> DavPass -> L.ByteString -> IO Bool storeHelper :: Remote -> DavUrl -> DavUser -> DavPass -> L.ByteString -> IO Bool

1
debian/changelog vendored
View file

@ -9,6 +9,7 @@ git-annex (3.20121113) UNRELEASED; urgency=low
* OSX: Fix RunAtLoad value in plist file. * OSX: Fix RunAtLoad value in plist file.
* Getting a file from chunked directory special remotes no longer buffers * Getting a file from chunked directory special remotes no longer buffers
it all in memory. it all in memory.
* S3: Added progress display for uploading and downloading.
-- Joey Hess <joeyh@debian.org> Tue, 13 Nov 2012 13:17:07 -0400 -- Joey Hess <joeyh@debian.org> Tue, 13 Nov 2012 13:17:07 -0400

View file

@ -23,7 +23,7 @@ the MeterUpdate callback as the upload progresses.
* directory: **done** * directory: **done**
* web: Not applicable; does not upload * web: Not applicable; does not upload
* webdav: **done** * webdav: **done**
* S3: TODO * S3: **done**
* bup: TODO * bup: TODO
* hook: Would require the hook interface to somehow do this, which seems * hook: Would require the hook interface to somehow do this, which seems
too complicated. So skipping. too complicated. So skipping.