LIB: implement copyDir and minor refactor

We also had to lock the FileOperation type to DTInfoZipper so
we can examine symlinks and other stuff, without re-reading
all the information.
This commit is contained in:
Julian Ospald 2015-12-18 04:22:13 +01:00
parent 0781cbf9d5
commit d4a5460128
No known key found for this signature in database
GPG Key ID: 220CD1C5BDEED020
3 changed files with 81 additions and 17 deletions

View File

@ -29,7 +29,8 @@ library
process, process,
stm, stm,
text, text,
time >= 1.4.2 time >= 1.4.2,
unix
hs-source-dirs: src hs-source-dirs: src
default-language: Haskell2010 default-language: Haskell2010
Default-Extensions: RecordWildCards Default-Extensions: RecordWildCards
@ -56,7 +57,8 @@ executable hsfm-gtk
stm, stm,
text, text,
time >= 1.4.2, time >= 1.4.2,
transformers >= 0.4 transformers >= 0.4,
unix
hs-source-dirs: src hs-source-dirs: src
default-language: Haskell2010 default-language: Haskell2010
Default-Extensions: RecordWildCards Default-Extensions: RecordWildCards

View File

@ -11,6 +11,10 @@ import Control.Monad
, void , void
, when , when
) )
import Data.List
(
isPrefixOf
)
import Data.Typeable import Data.Typeable
import System.Directory import System.Directory
( (
@ -31,6 +35,7 @@ data FmIOException = FileDoesNotExist String
| SameFile String String | SameFile String String
| NotAFile String | NotAFile String
| NotADir String | NotADir String
| DestinationInSource String String
deriving (Show, Typeable) deriving (Show, Typeable)
@ -79,3 +84,10 @@ throwSameFile :: FilePath -- ^ should be canonicalized
-> FilePath -- ^ should be canonicalized -> FilePath -- ^ should be canonicalized
-> IO () -> IO ()
throwSameFile fp1 fp2 = when (equalFilePath fp1 fp2) (throw $ SameFile fp1 fp2) throwSameFile fp1 fp2 = when (equalFilePath fp1 fp2) (throw $ SameFile fp1 fp2)
throwDestinationInSource :: FilePath -- ^ should be canonicalized
-> FilePath -- ^ should be canonicalized
-> IO ()
throwDestinationInSource source dest =
when (source `isPrefixOf` dest) (throw $ DestinationInSource dest source)

View File

@ -15,14 +15,18 @@ import Control.Monad
) )
import Data.DirTree import Data.DirTree
import Data.DirTree.Zipper import Data.DirTree.Zipper
import Data.Foldable
(
for_
)
import IO.Error import IO.Error
import System.Directory import System.Directory
( (
canonicalizePath canonicalizePath
, createDirectoryIfMissing
, doesDirectoryExist , doesDirectoryExist
, doesFileExist , doesFileExist
, executable , executable
, getPermissions
, removeDirectory , removeDirectory
, removeFile , removeFile
) )
@ -34,6 +38,11 @@ import System.FilePath
, takeDirectory , takeDirectory
, (</>) , (</>)
) )
import System.Posix.Files
(
createSymbolicLink
, readSymbolicLink
)
import System.Process import System.Process
( (
spawnProcess spawnProcess
@ -50,22 +59,58 @@ import qualified System.Directory as SD
-- |Data type describing an actual file operation that can be -- |Data type describing an actual file operation that can be
-- carried out via `doFile`. Useful to build up a list of operations -- carried out via `doFile`. Useful to build up a list of operations
-- or delay operations. -- or delay operations.
data FileOperation a b = FCopy (DTZipper a b) (DTZipper a b) data FileOperation = FCopy DTInfoZipper DTInfoZipper
| FMove FilePath FilePath | FMove FilePath FilePath
| FDelete (DTZipper a b) | FDelete DTInfoZipper
| FOpen (DTZipper a b) | FOpen DTInfoZipper
| FExecute (DTZipper a b) [String] | FExecute DTInfoZipper [String]
| None | None
runFileOp :: FileOperation a b -> IO () runFileOp :: FileOperation -> IO ()
runFileOp (FCopy from to) = copyFileToDir from to runFileOp (FCopy from@(File {}, _) to) = copyFileToDir from to
runFileOp (FCopy from@(Dir {}, _) to) = copyDir from to
runFileOp (FDelete fp) = easyDelete fp runFileOp (FDelete fp) = easyDelete fp
runFileOp (FOpen fp) = void $ openFile fp runFileOp (FOpen fp) = void $ openFile fp
runFileOp (FExecute fp args) = void $ executeFile fp args runFileOp (FExecute fp args) = void $ executeFile fp args
runFileOp _ = return () runFileOp _ = return ()
-- TODO: copy modes
copyDir :: DTInfoZipper -- ^ source dir
-> DTInfoZipper -- ^ destination dir
-> IO ()
copyDir from@(Dir fn _ _, _) to@(Dir {}, _) = do
let fromp = getFullPath from
top = getFullPath to
destdir = getFullPath to </> fn
dirSanityThrow fromp
dirSanityThrow top
throwDestinationInSource fromp top
createDirectoryIfMissing False destdir
for_ (goAllDown from) $ \f -> do
newDest <- zipLazy mkDirInfo mkFileInfo destdir
case f of
-- recreate symlink
sz@(Dir { name = n, dir = (DirInfo { sym = True }) }, _) -> do
let sympoint = getFullPath newDest </> n
-- delete old file/dir to be able to create symlink
for_ (goDown n newDest) $ \odtz ->
easyDelete odtz
symname <- readSymbolicLink (getFullPath sz)
createSymbolicLink symname sympoint
sz@(Dir {}, _) ->
copyDir sz newDest
sz@(File {}, _) ->
copyFileToDir sz newDest
copyDir from@(File _ _, _) _ = throw $ NotADir (getFullPath from)
copyDir _ to@(File _ _, _) = throw $ NotADir (getFullPath to)
-- |Copies the given file. -- |Copies the given file.
-- --
-- This will throw an exception if any of the filepaths are not absolute -- This will throw an exception if any of the filepaths are not absolute
@ -84,6 +129,7 @@ copyFile from@(File name _, _) to = do
let to' = toC </> takeFileName to let to' = toC </> takeFileName to
throwSameFile fp to' throwSameFile fp to'
SD.copyFile fp to' SD.copyFile fp to'
copyFile from _ = throw $ NotAFile (getFullPath from)
-- |Copies the given file to the given dir with the same filename. -- |Copies the given file to the given dir with the same filename.
@ -108,27 +154,31 @@ easyCopyFile from (Left to) = copyFile from to
easyCopyFile from (Right to) = copyFileToDir from to easyCopyFile from (Right to) = copyFileToDir from to
-- |Deletes the given file. -- |Deletes the given file or symlink.
-- --
-- This will throw an exception if the filepath is not absolute -- This will throw an exception if the filepath is not absolute
-- or the file does not exist. -- or the file does not exist.
-- --
-- It also throws exceptions from `removeFile`. -- It also throws exceptions from `removeFile`.
deleteFile :: DTZipper a b -> IO () deleteFile :: DTInfoZipper -> IO ()
deleteFile dtz@(File {}, _) = do deleteFile dtz@(File {}, _) = do
let fp = getFullPath dtz let fp = getFullPath dtz
fileSanityThrow fp fileSanityThrow fp
removeFile fp removeFile fp
deleteFile dtz@(Dir { dir = (DirInfo { sym = True }) }, _) = do
let fp = getFullPath dtz
throwNotAbsolute fp
removeFile fp
deleteFile dtz = throw $ NotAFile (getFullPath dtz) deleteFile dtz = throw $ NotAFile (getFullPath dtz)
-- |Deletes the given directory. -- |Deletes the given directory. Does not work on symlinks.
-- --
-- This will throw an exception if the filepath is not absolute -- This will throw an exception if the filepath is not absolute
-- or the directory does not exist. -- or the directory does not exist.
-- --
-- It also throws exceptions from `removeDirectory`. -- It also throws exceptions from `removeDirectory`.
deleteDir :: DTZipper a b -> IO () deleteDir :: DTInfoZipper -> IO ()
deleteDir dtz@(Dir {}, _) = do deleteDir dtz@(Dir {}, _) = do
let fp = getFullPath dtz let fp = getFullPath dtz
dirSanityThrow fp dirSanityThrow fp
@ -137,8 +187,9 @@ deleteDir dtz = throw $ NotADir (getFullPath dtz)
-- |Deletes a file or directory, whatever it may be. -- |Deletes a file or directory, whatever it may be.
easyDelete :: DTZipper a b -> IO () easyDelete :: DTInfoZipper -> IO ()
easyDelete dtz@(File {}, _) = deleteFile dtz easyDelete dtz@(File {}, _) = deleteFile dtz
easyDelete dtz@(Dir { dir = (DirInfo { sym = True }) }, _) = deleteFile dtz
easyDelete dtz@(Dir {}, _) = deleteDir dtz easyDelete dtz@(Dir {}, _) = deleteDir dtz
@ -160,13 +211,12 @@ openFile dtz = throw $ NotAFile (getFullPath dtz)
-- This will throw an exception if the filepath is not absolute -- This will throw an exception if the filepath is not absolute
-- or the file does not exist. It will also throw an exception -- or the file does not exist. It will also throw an exception
-- if the file is not executable. -- if the file is not executable.
executeFile :: DTZipper a b -- ^ program executeFile :: DTInfoZipper -- ^ program
-> [String] -- ^ arguments -> [String] -- ^ arguments
-> IO ProcessHandle -> IO ProcessHandle
executeFile dtz@(File {}, _) args = do executeFile dtz@(File { file = (FileInfo { permissions = p }) }, _) args = do
let fp = getFullPath dtz let fp = getFullPath dtz
fileSanityThrow fp fileSanityThrow fp
p <- getPermissions fp
unless (executable p) (throw $ FileNotExecutable fp) unless (executable p) (throw $ FileNotExecutable fp)
spawnProcess fp args spawnProcess fp args
executeFile dtz _ = throw $ NotAFile (getFullPath dtz) executeFile dtz _ = throw $ NotAFile (getFullPath dtz)