----------------------------------------------------------------------------- -- -- Module : Network.Google.Storage.Sync -- Copyright : (c) 2012-13 Brian W Bush -- License : MIT -- -- Maintainer : Brian W Bush -- Stability : Stable -- Portability : Portable -- -- | Synchronization of filesystem directories with Google Storage buckets. -- ----------------------------------------------------------------------------- {-# LANGUAGE BangPatterns #-} module Network.Google.Storage.Sync ( RegexExclusion , sync ) where import Control.Exception (SomeException, finally, handle) import Control.Monad (filterM, liftM, when) import Crypto.GnuPG (Recipient) import Crypto.MD5 (MD5Info, md5Base64, md5Empty) import qualified Data.ByteString.Lazy as LBS (ByteString, readFile) import qualified Data.Digest.Pure.MD5 as MD5 (md5) import Data.List ((\\), sort) import Data.Maybe (catMaybes, fromJust, fromMaybe, mapMaybe) import Data.Time.Clock (UTCTime, addUTCTime, getCurrentTime) import Data.Time.Clock.POSIX (posixSecondsToUTCTime) import Data.Time.Format (parseTime) import Network.Google (AccessToken, ProjectId, toAccessToken) import Network.Google.OAuth2 (OAuth2Client(..), OAuth2Tokens(..), refreshTokens, validateTokens) import Network.Google.Storage (BucketName, KeyName, MIMEType, StorageAcl, deleteObjectUsingManager, getBucketUsingManager, putObjectUsingManager) import Network.Google.Storage.Encrypted (putEncryptedObject, putEncryptedObjectUsingManager) import Network.HTTP.Conduit (closeManager, def, newManager) import System.Directory (doesDirectoryExist, getDirectoryContents) import System.FilePath (combine, splitDirectories) import System.FilePath.Posix (joinPath) import System.IO (hFlush, stdout) import System.Locale (defaultTimeLocale) import System.PosixCompat.Files (fileSize, getFileStatus, getSymbolicLinkStatus, isSymbolicLink, modificationTime) import Text.Regex.Posix ((=~)) import Text.XML.Light (Element, QName(qName), filterChildrenName, filterChildName, ppTopElement, strContent) -- | A regular expression used for excluding files from synchronization. type RegexExclusion = String -- | Synchronize a filesystem directory with a Google Storage bucket. sync :: ProjectId -- ^ The Google project ID. -> StorageAcl -- ^ The pre-defined access control. -> BucketName -- ^ The bucket name. -> OAuth2Client -- ^ The OAuth 2.0 client information. -> OAuth2Tokens -- ^ The OAuth 2.0 tokens. -> FilePath -- ^ The directory to be synchronized. -> [Recipient] -- ^ The recipients for GnuPG encryption of the uploaded files. -> [RegexExclusion] -- ^ The regular expressions used for excluding files from synchronization. -> Bool -- ^ Whether to write a file \".md5sum\" of MD5 sums of synchronized files into the root directory. -> Bool -- ^ Whether to delete keys from the bucket that do not correspond to files on the filesystem. -> IO () -- ^ The IO action for the synchronization. sync projectId acl bucket client tokens directory recipients exclusions md5sums purge = do putStrLn $ "DIRECTORY " ++ directory putStrLn $ "PROJECT " ++ projectId putStrLn $ "BUCKET " ++ bucket putStrLn $ "ACCESS " ++ show acl putStr "LOCAL " hFlush stdout let byETag = null recipients && md5sums local <- liftM sort $ walkDirectories byETag directory print $ length local putStr "EXCLUDED " hFlush stdout let local' = filter (makeExcluder exclusions) local print $ length local - length local' manager <- newManager def finally ( sync' (getBucketUsingManager manager projectId bucket) ((if null recipients then putObjectUsingManager manager else putEncryptedObjectUsingManager manager recipients) projectId acl bucket) (deleteObjectUsingManager manager projectId bucket) client tokens directory byETag local' md5sums purge )( closeManager manager ) -- | A function for listing a bucket. type Lister = AccessToken -- ^ The OAuth 2.0 access token. -> IO Element -- ^ The action returning the XML with the metadata for the objects. -- | A function for putting an object into a bucket. type Putter = KeyName -- ^ The object's key. -> Maybe MIMEType -- ^ The object's MIME type. -> LBS.ByteString -- ^ The object's data. -> Maybe MD5Info -- ^ The MD5 checksum. -> AccessToken -- ^ The OAuth 2.0 access token. -> IO [(String, String)] -- ^ The action to put the object and return the response header. -- | A function for deleting an object from a bucket. type Deleter = KeyName -- ^ The object's key. -> AccessToken -- ^ The OAuth 2.0 access token. -> IO [(String, String)] -- ^ The action to put the object and return the response header. -- | A function for determining whether to exclude an object from synchronization. type Excluder = ObjectMetadata -- ^ The object's metadata. -> Bool -- ^ Whether to exclude the object from synchronization. -- | An expiration time and the tokens which expire then. type TokenClock = (UTCTime, OAuth2Tokens) -- | Check whether a token has expired and refresh it if necessary. checkExpiration :: OAuth2Client -- ^ The OAuth 2.0 client information. -> TokenClock -- ^ The token and its expiration. -> IO TokenClock -- ^ The action to update the token and its expiration. checkExpiration client (expirationTime, tokens) = do now <- getCurrentTime if now > expirationTime then do tokens' <- refreshTokens client tokens start <- getCurrentTime let expirationTime' = addUTCTime (fromRational (expiresIn tokens') - 60) start putStrLn $ "REFRESH " ++ show expirationTime' return (expirationTime', tokens') else return (expirationTime, tokens) -- | Make a function to exclude objects based on regular expressions for filenames. makeExcluder :: [RegexExclusion] -- ^ The regular expressions. -> Excluder -- ^ The function for excluding objects. makeExcluder exclusions (ObjectMetadata candidate _ _ _) = let match :: String -> Bool match exclusion = candidate =~ exclusion in not $ any match exclusions -- | Synchronize a filesystem directory with a Google Storage bucket. sync' :: Lister -- ^ The bucket listing function. -> Putter -- ^ The object putting function. -> Deleter -- ^ The object deletion function. -> OAuth2Client -- ^ The OAuth 2.0 client information. -> OAuth2Tokens -- ^ The OAuth 2.0 tokens. -> FilePath -- ^ The directory to be synchronized. -> Bool -- ^ Whether to use ETags in comparing object metadata. -> [ObjectMetadata] -- ^ The local file system objects to be synchronized. -> Bool -- ^ Whether to write a file \".md5sum\" of MD5 sums of synchronized files into the root directory. -> Bool -- ^ Whether to delete keys from the bucket that do not correspond to files on the filesystem. -> IO () -- ^ The IO action for the synchronization. sync' lister putter deleter client tokens directory byETag local md5sums purge = do now <- getCurrentTime tokenClock@(_, tokens') <- checkExpiration client (addUTCTime (-60) now, tokens) putStr "REMOTE " hFlush stdout remote' <- lister $ toAccessToken $ accessToken tokens' let remote = sort $ parseMetadata remote' print $ length remote let tolerance = 300 sameKey :: ObjectMetadata -> ObjectMetadata -> Bool sameKey (ObjectMetadata key _ _ _) (ObjectMetadata key' _ _ _) = key == key' sameETag :: ObjectMetadata -> ObjectMetadata -> Bool sameETag (ObjectMetadata key eTag _ _) (ObjectMetadata key' eTag' _ _) = key == key' && fst eTag == fst eTag' earlierTime :: ObjectMetadata -> ObjectMetadata -> Bool earlierTime (ObjectMetadata key _ _ time) (ObjectMetadata key' _ _ time') = key == key' && addUTCTime tolerance time < time' putStr "PUTS " hFlush stdout let changedObjects = deleteFirstsBy' (if byETag then sameETag else earlierTime) local remote print $ length changedObjects putStr "DELETES " hFlush stdout let deletedObjects = deleteFirstsBy' sameKey remote local print $ length deletedObjects tokenClock' <- walkPutter client tokenClock directory putter changedObjects tokenClock'' <- if purge then walkDeleter client tokenClock' deleter deletedObjects else return tokenClock' when md5sums $ writeFile (combine directory ".md5sum") $ unlines $ map (\x -> (fst . eTag) x ++ " ./" ++ key x) local -- | Delete the first occurrence of items in the second list from the first list, assuming both lists are sorted. deleteFirstsBy' :: Ord a => (a -> a -> Bool) -- ^ The equality function. -> [a] -- ^ The first list, which must have been sorted. -> [a] -- ^ The second list, which must have been sorted. -> [a] -- ^ The items in the first list not appearing in the second list. deleteFirstsBy' _ [] _ = [] deleteFirstsBy' _ xx [] = xx deleteFirstsBy' equal xx@(x : xs) yy@(y : ys) | equal x y = deleteFirstsBy' equal xs ys | x > y = deleteFirstsBy' equal xx ys | otherwise = x : deleteFirstsBy' equal xs yy -- | Put a list of objects. walkPutter :: OAuth2Client -- ^ The OAuth 2.0 client information. -> TokenClock -- ^ The token and its expiration. -> FilePath -- ^ The directory to be synchronized. -> Putter -- ^ The object putting function. -> [ObjectMetadata] -- ^ Description of the objects to be put. -> IO TokenClock -- ^ The action to update the token and its expiration. walkPutter _ tokenClock _ _ [] = return tokenClock walkPutter client tokenClock directory putter (x : xs) = do let key' = key x tokenClock'@(_, tokens') <- checkExpiration client tokenClock putStrLn $ "PUT " ++ key' handle handler ( do let eTag' x = if eTag x == md5Empty then Nothing else Just $ eTag x bytes <- LBS.readFile $ combine directory key' putter key' Nothing bytes (eTag' x) (toAccessToken $ accessToken tokens') return () ) walkPutter client tokenClock' directory putter xs -- | Delete a list of objects. walkDeleter :: OAuth2Client -- ^ The OAuth 2.0 client information. -> TokenClock -- ^ The token and its expiration. -> Deleter -- ^ The object deletion function. -> [ObjectMetadata] -- ^ Description of the objects to be deleted. -> IO TokenClock -- ^ The action to update the token and its expiration. walkDeleter _ tokenClock _ [] = return tokenClock walkDeleter client tokenClock deleter (x : xs) = do let key' = key x tokenClock'@(_, tokens') <- checkExpiration client tokenClock putStrLn $ "DELETE " ++ key' handle handler ( do deleter key' (toAccessToken $ accessToken tokens') return () ) walkDeleter client tokenClock' deleter xs -- | Exception handler. handler :: SomeException -- ^ The exception. -> IO () -- ^ An empty action. handler exception = putStrLn $ " FAIL " ++ show exception -- | Object metadata. data ObjectMetadata = ObjectMetadata { key :: String -- ^ The object's key. , eTag :: MD5Info -- ^ The object's MD5 sum. , size :: Int -- ^ The object's size, in bytes. , lastModified :: UTCTime -- ^ The object's modification time. } deriving (Show) instance Ord ObjectMetadata where compare (ObjectMetadata key _ _ _) (ObjectMetadata key' _ _ _) = compare key key' instance Eq ObjectMetadata where (ObjectMetadata key _ _ _) == (ObjectMetadata key' _ _ _) = key == key' -- | Parse XML metadata into object descriptions. parseMetadata :: Element -- ^ The XML metadata. -> [ObjectMetadata] -- ^ The object descriptions. parseMetadata root = let makeMetadata :: Element -> Maybe ObjectMetadata makeMetadata element = do let finder :: String -> Element -> Maybe String finder name = liftM strContent . filterChildName ((name ==) . qName) key <- finder "Key" element eTag <- finder "ETag" element size <- finder "Size" element lastModified <- finder "LastModified" element return $ ObjectMetadata key (tail . init $ eTag, undefined) (read size) (fromJust $ parseTime defaultTimeLocale "%FT%T%QZ" lastModified) in mapMaybe makeMetadata $ filterChildrenName (("Contents" ==) . qName) root -- | Gather file metadata from the file system. walkDirectories :: Bool -- ^ Whether to compute MD5 sums. -> FilePath -- ^ The directory to be synchronized. -> IO [ObjectMetadata] -- ^ Action returning file descriptions. walkDirectories eTags directory = walkDirectories' eTags directory [""] -- | Gather file metadata from the file system. walkDirectories' :: Bool -- ^ Whether to compute MD5 sums. -> FilePath -- ^ The directory to be synchronized. -> [FilePath] -- ^ The subdirectories still remaining to be described. -> IO [ObjectMetadata] -- ^ Action returning file descriptions. walkDirectories' _ _ [] = return [] walkDirectories' eTags directory (y : ys) = do let isWalkable path = do directoryExists <- doesDirectoryExist path status <- getSymbolicLinkStatus path return $ directoryExists && not (isSymbolicLink status) isFileable path = do directoryExists <- doesDirectoryExist path status <- getSymbolicLinkStatus path return $ not directoryExists && not (isSymbolicLink status) handler :: a -> SomeException -> IO a handler def exception = do putStrLn $ " LIST " ++ y putStrLn $ " FAIL " ++ show exception return def makeMetadata :: FilePath -> IO (Maybe ObjectMetadata) makeMetadata file = handle (handler Nothing) ( do let key = (joinPath . splitDirectories) file path = combine directory key bytes <- LBS.readFile path status <- getFileStatus path let !lastTime = posixSecondsToUTCTime $ realToFrac $ modificationTime status !size = fromIntegral $ fileSize status !eTag = if eTags then md5Base64 bytes else md5Empty return $ Just $ ObjectMetadata key eTag size lastTime ) files <- handle (handler []) ( liftM (map (combine y) . (\\ [".", ".."])) $ getDirectoryContents (combine directory y) ) y' <- filterM (isWalkable . combine directory) files y'' <- filterM (isFileable . combine directory) files x' <- mapM makeMetadata y'' liftM (catMaybes x' ++) $ walkDirectories' eTags directory (y' ++ ys)