70 lines
		
	
	
	
		
			2.1 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
			
		
		
	
	
			70 lines
		
	
	
	
		
			2.1 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
{- git-annex "WORM" backend -- Write Once, Read Many
 | 
						|
 -
 | 
						|
 - Copyright 2010 Joey Hess <id@joeyh.name>
 | 
						|
 -
 | 
						|
 - Licensed under the GNU AGPL version 3 or higher.
 | 
						|
 -}
 | 
						|
 | 
						|
module Backend.WORM (backends) where
 | 
						|
 | 
						|
import Annex.Common
 | 
						|
import Types.Key
 | 
						|
import Types.Backend
 | 
						|
import Types.KeySource
 | 
						|
import Backend.Utilities
 | 
						|
import Git.FilePath
 | 
						|
import Utility.Metered
 | 
						|
import qualified Utility.RawFilePath as R
 | 
						|
 | 
						|
import qualified Data.ByteString.Char8 as S8
 | 
						|
import qualified Data.ByteString.Short as S (toShort, fromShort)
 | 
						|
import System.PosixCompat.Files (modificationTime)
 | 
						|
 | 
						|
backends :: [Backend]
 | 
						|
backends = [backend]
 | 
						|
 | 
						|
backend :: Backend
 | 
						|
backend = Backend
 | 
						|
	{ backendVariety = WORMKey
 | 
						|
	, genKey = Just keyValue
 | 
						|
	, verifyKeyContent = Nothing
 | 
						|
	, verifyKeyContentIncrementally = Nothing
 | 
						|
	, canUpgradeKey = Just needsUpgrade
 | 
						|
	, fastMigrate = Just removeProblemChars
 | 
						|
	, isStableKey = const True
 | 
						|
	, isCryptographicallySecure = False
 | 
						|
	, isCryptographicallySecureKey = const (pure False)
 | 
						|
	}
 | 
						|
 | 
						|
{- The key includes the file size, modification time, and the
 | 
						|
 - original filename relative to the top of the git repository.
 | 
						|
 -}
 | 
						|
keyValue :: KeySource -> MeterUpdate -> Annex Key
 | 
						|
keyValue source _ = do
 | 
						|
	let f = contentLocation source
 | 
						|
	stat <- liftIO $ R.getFileStatus (fromOsPath f)
 | 
						|
	sz <- liftIO $ getFileSize' f stat
 | 
						|
	relf <- fromOsPath . getTopFilePath
 | 
						|
		<$> inRepo (toTopFilePath $ keyFilename source)
 | 
						|
	return $ mkKey $ \k -> k
 | 
						|
		{ keyName = genKeyName relf
 | 
						|
		, keyVariety = WORMKey
 | 
						|
		, keySize = Just sz
 | 
						|
		, keyMtime = Just $ modificationTime stat
 | 
						|
		}
 | 
						|
 | 
						|
{- Old WORM keys could contain spaces and carriage returns, 
 | 
						|
 - and can be upgraded to remove them. -}
 | 
						|
needsUpgrade :: Key -> Bool
 | 
						|
needsUpgrade key =
 | 
						|
	any (`S8.elem` S.fromShort (fromKey keyName key)) [' ', '\r']
 | 
						|
 | 
						|
removeProblemChars :: Key -> Backend -> AssociatedFile -> Bool -> Annex (Maybe Key)
 | 
						|
removeProblemChars oldkey newbackend _ _
 | 
						|
	| migratable = return $ Just $ alterKey oldkey $ \d -> d
 | 
						|
		{ keyName = S.toShort $ encodeBS $ reSanitizeKeyName $ decodeBS $ S.fromShort $ keyName d }
 | 
						|
	| otherwise = return Nothing
 | 
						|
  where
 | 
						|
	migratable = oldvariety == newvariety
 | 
						|
	oldvariety = fromKey keyVariety oldkey
 | 
						|
	newvariety = backendVariety newbackend
 |