module Development.Shake.Storage(
withStorage
) where
import Development.Shake.Binary
import Development.Shake.Locks
import Development.Shake.Types
import Development.Shake.Timing
import Control.Arrow
import Control.Exception as E
import Control.Monad
import Control.Concurrent
import Data.Binary.Get
import Data.Binary.Put
import Data.Time
import Data.Char
import Development.Shake.Classes
import qualified Data.HashMap.Strict as Map
import Data.List
import System.Directory
import System.Exit
import System.FilePath
import System.IO
import qualified Data.ByteString.Lazy.Char8 as LBS
type Map = Map.HashMap
databaseVersion :: String -> String
databaseVersion x = "SHAKE-DATABASE-8-" ++ tail (init $ show x) ++ "\r\n"
withStorage
:: (Eq w, Eq k, Hashable k
,Binary w, BinaryWith w k, BinaryWith w v)
=> ShakeOptions
-> (String -> IO ())
-> w
-> (Map k v -> (k -> v -> IO ()) -> IO a)
-> IO a
withStorage ShakeOptions{shakeVerbosity,shakeOutput,shakeVersion,shakeFlush,shakeFiles,shakeStorageLog} diagnostic witness act = do
let dbfile = shakeFiles <.> "database"
bupfile = shakeFiles <.> "bup"
createDirectoryIfMissing True $ takeDirectory shakeFiles
b <- doesFileExist bupfile
when b $ do
unexpected "Backup file exists, restoring over the previous file\n"
diagnostic $ "Backup file move to original"
E.catch (removeFile dbfile) (\(e :: SomeException) -> return ())
renameFile bupfile dbfile
addTiming "Database read"
withBinaryFile dbfile ReadWriteMode $ \h -> do
n <- hFileSize h
diagnostic $ "Reading file of size " ++ show n
src <- LBS.hGet h $ fromInteger n
if not $ ver `LBS.isPrefixOf` src then do
unless (LBS.null src) $ do
let good x = isAlphaNum x || x `elem` "-_ "
let bad = LBS.takeWhile good $ LBS.take 50 src
outputErr $ unlines
["Error when reading Shake database " ++ dbfile
," Invalid version stamp detected"
," Expected: " ++ takeWhile good (LBS.unpack ver)
," Found : " ++ LBS.unpack bad
,"All rules will be rebuilt"]
continue h Map.empty
else
join $ handleJust (\e -> if asyncException e then Nothing else Just e) (\err -> do
msg <- showException err
outputErr $ unlines $
("Error when reading Shake database " ++ dbfile) :
map (" "++) (lines msg) ++
["All files will be rebuilt"]
when shakeStorageLog $ do
hSeek h AbsoluteSeek 0
i <- hFileSize h
bs <- LBS.hGet h $ fromInteger i
let cor = shakeFiles <.> "corrupt"
LBS.writeFile cor bs
unexpected $ "Backup of corrupted file stored at " ++ cor ++ ", " ++ show i ++ " bytes\n"
return $ continue h Map.empty) $
case readChunks $ LBS.drop (LBS.length ver) src of
(slop, []) -> do
when (slop > 0) $ unexpected $ "Last " ++ show slop ++ " bytes do not form a whole record\n"
diagnostic $ "Read 0 chunks, plus " ++ show slop ++ " slop"
return $ continue h Map.empty
(slop, w:xs) -> do
when (slop > 0) $ unexpected $ "Last " ++ show slop ++ " bytes do not form a whole record\n"
diagnostic $ "Read " ++ show (length xs + 1) ++ " chunks, plus " ++ show slop ++ " slop"
diagnostic $ "Chunk sizes " ++ show (map LBS.length (w:xs))
let ws = decode w
f mp (k, v) = Map.insert k v mp
mp = foldl' f Map.empty $ map (runGet $ getWith ws) xs
if Map.null mp || (ws == witness && Map.size mp * 2 > length xs 2) then do
when (not (Map.null mp) && slop /= 0) $ do
diagnostic $ "Dropping last " ++ show slop ++ " bytes of database (incomplete)"
now <- hFileSize h
hSetFileSize h $ now slop
hSeek h AbsoluteSeek $ now slop
hFlush h
diagnostic $ "Drop complete"
return $ continue h mp
else do
addTiming "Database compression"
unexpected "Compressing database\n"
diagnostic "Compressing database"
hClose h
return $ do
renameFile dbfile bupfile
withBinaryFile dbfile ReadWriteMode $ \h -> do
reset h mp
removeFile bupfile
diagnostic "Compression complete"
continue h mp
where
unexpected x = when shakeStorageLog $ do
t <- getCurrentTime
appendFile (shakeFiles <.> "storage") $ "\n[" ++ show t ++ "]: " ++ x
outputErr x = do
when (shakeVerbosity >= Quiet) $ shakeOutput Quiet x
unexpected x
ver = LBS.pack $ databaseVersion shakeVersion
writeChunk h s = do
diagnostic $ "Writing chunk " ++ show (LBS.length s)
LBS.hPut h $ toChunk s
reset h mp = do
diagnostic $ "Resetting database to " ++ show (Map.size mp) ++ " elements"
hSetFileSize h 0
hSeek h AbsoluteSeek 0
LBS.hPut h ver
writeChunk h $ encode witness
mapM_ (writeChunk h . runPut . putWith witness) $ Map.toList mp
hFlush h
diagnostic "Flush"
continue h mp = do
when (Map.null mp) $
reset h mp
flushThread outputErr shakeFlush h $ \out -> do
addTiming "With database"
act mp $ \k v -> out $ toChunk $ runPut $ putWith witness (k, v)
flushThread :: (String -> IO ()) -> Maybe Double -> Handle -> ((LBS.ByteString -> IO ()) -> IO a) -> IO a
flushThread outputErr flush h act = do
alive <- newVar True
kick <- newEmptyMVar
lock <- newLock
case flush of
Nothing -> return ()
Just flush -> do
let delay = ceiling $ flush * 1000000
let loop = do
takeMVar kick
threadDelay delay
b <- withVar alive $ \b -> do
when b $ do
tryTakeMVar kick
withLock lock $ hFlush h
return b
when b loop
forkIO $ do
let msg = "Warning: Flushing Shake journal failed, on abnormal termination you may lose some data, "
(loop >> return ()) `E.catch` \(e :: SomeException) -> outputErr $ msg ++ show e ++ "\n"
return ()
(act $ \s -> do
withLock lock $ LBS.hPut h s
tryPutMVar kick ()
return ())
`finally` do
modifyVar_ alive $ const $ return False
tryPutMVar kick ()
readChunks :: LBS.ByteString -> (Integer, [LBS.ByteString])
readChunks x
| Just (n, x) <- grab 4 x
, Just (y, x) <- grab (fromIntegral (decode n :: Word32)) x
= second (y :) $ readChunks x
| otherwise = (toInteger $ LBS.length x, [])
where
grab i x | LBS.length a == i = Just (a, b)
| otherwise = Nothing
where (a,b) = LBS.splitAt i x
toChunk :: LBS.ByteString -> LBS.ByteString
toChunk x = n `LBS.append` x
where n = encode (fromIntegral $ LBS.length x :: Word32)
showException :: SomeException -> IO String
showException err = do
let msg = show err
E.catch (evaluate $ rnf msg `seq` msg) (\(_ :: SomeException) -> return "Unknown exception (error while showing error message)")
asyncException :: SomeException -> Bool
asyncException e
| Just (_ :: AsyncException) <- fromException e = True
| Just (_ :: ExitCode) <- fromException e = True
| otherwise = False