2014-07-27 00:14:09 +00:00
|
|
|
{- Legacy chunksize support for directory special remote.
|
|
|
|
-
|
|
|
|
- Can be removed eventually.
|
|
|
|
-
|
2015-01-21 16:50:09 +00:00
|
|
|
- Copyright 2011-2012 Joey Hess <id@joeyh.name>
|
2014-07-27 00:14:09 +00:00
|
|
|
-
|
2019-03-13 19:48:14 +00:00
|
|
|
- Licensed under the GNU AGPL version 3 or higher.
|
2014-07-27 00:14:09 +00:00
|
|
|
-}
|
|
|
|
|
|
|
|
module Remote.Directory.LegacyChunked where
|
|
|
|
|
|
|
|
import qualified Data.ByteString.Lazy as L
|
|
|
|
import qualified Data.ByteString as S
|
|
|
|
|
2016-01-20 20:36:33 +00:00
|
|
|
import Annex.Common
|
2014-07-27 00:14:09 +00:00
|
|
|
import Utility.FileMode
|
2014-08-03 19:35:23 +00:00
|
|
|
import Remote.Helper.Special
|
2014-07-27 00:14:09 +00:00
|
|
|
import qualified Remote.Helper.Chunked.Legacy as Legacy
|
2019-01-17 19:40:44 +00:00
|
|
|
import Annex.Tmp
|
2014-07-27 00:14:09 +00:00
|
|
|
import Utility.Metered
|
|
|
|
|
|
|
|
withCheckedFiles :: (FilePath -> IO Bool) -> FilePath -> (FilePath -> Key -> [FilePath]) -> Key -> ([FilePath] -> IO Bool) -> IO Bool
|
|
|
|
withCheckedFiles _ [] _locations _ _ = return False
|
|
|
|
withCheckedFiles check d locations k a = go $ locations d k
|
|
|
|
where
|
|
|
|
go [] = return False
|
|
|
|
go (f:fs) = do
|
|
|
|
let chunkcount = f ++ Legacy.chunkCount
|
|
|
|
ifM (check chunkcount)
|
|
|
|
( do
|
|
|
|
chunks <- Legacy.listChunks f <$> readFile chunkcount
|
|
|
|
ifM (allM check chunks)
|
|
|
|
( a chunks , return False )
|
|
|
|
, do
|
|
|
|
chunks <- Legacy.probeChunks f check
|
|
|
|
if null chunks
|
|
|
|
then go fs
|
|
|
|
else a chunks
|
|
|
|
)
|
|
|
|
withStoredFiles :: FilePath -> (FilePath -> Key -> [FilePath]) -> Key -> ([FilePath] -> IO Bool) -> IO Bool
|
|
|
|
withStoredFiles = withCheckedFiles doesFileExist
|
|
|
|
|
|
|
|
{- Splits a ByteString into chunks and writes to dests, obeying configured
|
|
|
|
- chunk size (not to be confused with the L.ByteString chunk size). -}
|
|
|
|
storeLegacyChunked :: MeterUpdate -> ChunkSize -> [FilePath] -> L.ByteString -> IO [FilePath]
|
|
|
|
storeLegacyChunked _ _ [] _ = error "bad storeLegacyChunked call"
|
|
|
|
storeLegacyChunked meterupdate chunksize alldests@(firstdest:_) b
|
|
|
|
| L.null b = do
|
|
|
|
-- always write at least one file, even for empty
|
|
|
|
L.writeFile firstdest b
|
|
|
|
return [firstdest]
|
|
|
|
| otherwise = storeLegacyChunked' meterupdate chunksize alldests (L.toChunks b) []
|
|
|
|
storeLegacyChunked' :: MeterUpdate -> ChunkSize -> [FilePath] -> [S.ByteString] -> [FilePath] -> IO [FilePath]
|
|
|
|
storeLegacyChunked' _ _ [] _ _ = error "ran out of dests"
|
|
|
|
storeLegacyChunked' _ _ _ [] c = return $ reverse c
|
|
|
|
storeLegacyChunked' meterupdate chunksize (d:dests) bs c = do
|
|
|
|
bs' <- withFile d WriteMode $
|
|
|
|
feed zeroBytesProcessed chunksize bs
|
|
|
|
storeLegacyChunked' meterupdate chunksize dests bs' (d:c)
|
|
|
|
where
|
|
|
|
feed _ _ [] _ = return []
|
|
|
|
feed bytes sz (l:ls) h = do
|
|
|
|
let len = S.length l
|
|
|
|
let s = fromIntegral len
|
|
|
|
if s <= sz || sz == chunksize
|
|
|
|
then do
|
|
|
|
S.hPut h l
|
|
|
|
let bytes' = addBytesProcessed bytes len
|
|
|
|
meterupdate bytes'
|
|
|
|
feed bytes' (sz - s) ls h
|
|
|
|
else return (l:ls)
|
|
|
|
|
|
|
|
storeHelper :: (FilePath -> FilePath -> IO ()) -> Key -> ([FilePath] -> IO [FilePath]) -> FilePath -> FilePath -> IO Bool
|
|
|
|
storeHelper finalizer key storer tmpdir destdir = do
|
|
|
|
void $ liftIO $ tryIO $ createDirectoryIfMissing True tmpdir
|
|
|
|
Legacy.storeChunks key tmpdir destdir storer recorder finalizer
|
|
|
|
where
|
|
|
|
recorder f s = do
|
|
|
|
void $ tryIO $ allowWrite f
|
|
|
|
writeFile f s
|
|
|
|
void $ tryIO $ preventWrite f
|
|
|
|
|
|
|
|
store :: ChunkSize -> (FilePath -> FilePath -> IO ()) -> Key -> L.ByteString -> MeterUpdate -> FilePath -> FilePath -> IO Bool
|
|
|
|
store chunksize finalizer k b p = storeHelper finalizer k $ \dests ->
|
|
|
|
storeLegacyChunked p chunksize dests b
|
|
|
|
|
|
|
|
{- Need to get a single ByteString containing every chunk.
|
|
|
|
- Done very innefficiently, by writing to a temp file.
|
|
|
|
- :/ This is legacy code..
|
|
|
|
-}
|
2014-07-27 04:30:04 +00:00
|
|
|
retrieve :: (FilePath -> Key -> [FilePath]) -> FilePath -> Preparer Retriever
|
2019-01-17 19:40:44 +00:00
|
|
|
retrieve locations d basek a = withOtherTmp $ \tmpdir -> do
|
2014-07-27 00:14:09 +00:00
|
|
|
showLongNote "This remote uses the deprecated chunksize setting. So this will be quite slow."
|
|
|
|
let tmp = tmpdir </> keyFile basek ++ ".directorylegacy.tmp"
|
2014-08-03 05:12:24 +00:00
|
|
|
a $ Just $ byteRetriever $ \k sink -> do
|
|
|
|
liftIO $ void $ withStoredFiles d locations k $ \fs -> do
|
2014-07-27 00:14:09 +00:00
|
|
|
forM_ fs $
|
|
|
|
S.appendFile tmp <=< S.readFile
|
|
|
|
return True
|
2014-08-03 05:12:24 +00:00
|
|
|
b <- liftIO $ L.readFile tmp
|
|
|
|
liftIO $ nukeFile tmp
|
|
|
|
sink b
|
2014-07-27 00:14:09 +00:00
|
|
|
|
2014-08-06 17:45:19 +00:00
|
|
|
checkKey :: FilePath -> (FilePath -> Key -> [FilePath]) -> Key -> Annex Bool
|
|
|
|
checkKey d locations k = liftIO $ withStoredFiles d locations k $
|
|
|
|
-- withStoredFiles checked that it exists
|
|
|
|
const $ return True
|