2013-06-10 13:10:30 -04:00
|
|
|
{- file copying
|
2010-11-18 13:48:28 -04:00
|
|
|
-
|
2015-01-21 12:50:09 -04:00
|
|
|
- Copyright 2010-2014 Joey Hess <id@joeyh.name>
|
2010-11-18 13:48:28 -04:00
|
|
|
-
|
2014-05-10 11:01:27 -03:00
|
|
|
- License: BSD-2-clause
|
2010-11-18 13:48:28 -04:00
|
|
|
-}
|
|
|
|
|
2013-06-10 13:10:30 -04:00
|
|
|
{-# LANGUAGE CPP #-}
|
|
|
|
|
|
|
|
module Utility.CopyFile (
|
|
|
|
copyFileExternal,
|
2014-08-26 17:06:43 -07:00
|
|
|
createLinkOrCopy,
|
|
|
|
CopyMetaData(..)
|
2013-06-10 13:10:30 -04:00
|
|
|
) where
|
2010-11-18 13:48:28 -04:00
|
|
|
|
2012-04-14 12:33:32 -04:00
|
|
|
import Common
|
2017-12-14 12:46:57 -04:00
|
|
|
import qualified BuildInfo
|
2010-11-18 13:48:28 -04:00
|
|
|
|
2015-04-18 14:13:07 -04:00
|
|
|
data CopyMetaData
|
|
|
|
-- Copy timestamps when possible, but no other metadata, and
|
|
|
|
-- when copying a symlink, makes a copy of its content.
|
|
|
|
= CopyTimeStamps
|
|
|
|
-- Copy all metadata when possible.
|
|
|
|
| CopyAllMetaData
|
2014-08-26 17:06:43 -07:00
|
|
|
deriving (Eq)
|
|
|
|
|
2010-11-18 13:48:28 -04:00
|
|
|
{- The cp command is used, because I hate reinventing the wheel,
|
|
|
|
- and because this allows easy access to features like cp --reflink. -}
|
2014-08-26 17:06:43 -07:00
|
|
|
copyFileExternal :: CopyMetaData -> FilePath -> FilePath -> IO Bool
|
|
|
|
copyFileExternal meta src dest = do
|
2011-05-17 03:10:13 -04:00
|
|
|
whenM (doesFileExist dest) $
|
2011-01-04 22:17:18 -04:00
|
|
|
removeFile dest
|
2012-05-15 14:18:51 -04:00
|
|
|
boolSystem "cp" $ params ++ [File src, File dest]
|
2012-12-13 00:24:19 -04:00
|
|
|
where
|
2013-06-14 11:54:44 -04:00
|
|
|
#ifndef __ANDROID__
|
2012-12-13 00:24:19 -04:00
|
|
|
params = map snd $ filter fst
|
2017-12-14 12:46:57 -04:00
|
|
|
[ (BuildInfo.cp_reflink_auto, Param "--reflink=auto")
|
|
|
|
, (allmeta && BuildInfo.cp_a, Param "-a")
|
|
|
|
, (allmeta && BuildInfo.cp_p && not BuildInfo.cp_a
|
2014-08-26 17:06:43 -07:00
|
|
|
, Param "-p")
|
2017-12-14 12:46:57 -04:00
|
|
|
, (not allmeta && BuildInfo.cp_preserve_timestamps
|
2014-08-26 17:06:43 -07:00
|
|
|
, Param "--preserve=timestamps")
|
2012-12-13 00:24:19 -04:00
|
|
|
]
|
2013-06-14 11:54:44 -04:00
|
|
|
#else
|
2015-07-02 15:24:33 -04:00
|
|
|
params = if allmeta then [] else []
|
2013-06-14 11:54:44 -04:00
|
|
|
#endif
|
2014-08-26 17:06:43 -07:00
|
|
|
allmeta = meta == CopyAllMetaData
|
2013-06-10 13:10:30 -04:00
|
|
|
|
|
|
|
{- Create a hard link if the filesystem allows it, and fall back to copying
|
|
|
|
- the file. -}
|
|
|
|
createLinkOrCopy :: FilePath -> FilePath -> IO Bool
|
|
|
|
createLinkOrCopy src dest = go `catchIO` const fallback
|
|
|
|
where
|
2014-10-09 14:53:13 -04:00
|
|
|
go = do
|
2013-06-10 13:10:30 -04:00
|
|
|
createLink src dest
|
|
|
|
return True
|
2014-10-09 14:53:13 -04:00
|
|
|
fallback = copyFileExternal CopyAllMetaData src dest
|