LIB: wrap canonicalizePath so it doesn't follow the last symlink component

canonicalizePath tries to remove as many symlink components as possible,
but this is not what we want for a lot of operations like copying
files. So instead we preserve the last path component if it's a symlink
and canonicalize it's parent filepath.
This commit is contained in:
Julian Ospald 2015-12-21 17:15:31 +01:00
parent 89157bc0d9
commit 988350967a
No known key found for this signature in database
GPG Key ID: 220CD1C5BDEED020
2 changed files with 48 additions and 21 deletions

View File

@ -35,6 +35,7 @@ import Data.Default
import Data.List import Data.List
( (
delete delete
, foldl'
, isPrefixOf , isPrefixOf
, sort , sort
, sortBy , sortBy
@ -64,6 +65,7 @@ import Data.Word
import Safe import Safe
( (
atDef atDef
, initDef
) )
import System.Directory import System.Directory
( (
@ -79,6 +81,7 @@ import System.Directory
import System.FilePath import System.FilePath
( (
combine combine
, normalise
, equalFilePath , equalFilePath
, joinPath , joinPath
, splitDirectories , splitDirectories
@ -248,7 +251,7 @@ readFileWith :: (FilePath -> IO a)
-> FilePath -> FilePath
-> IO (AnchoredFile a b) -> IO (AnchoredFile a b)
readFileWith fd ff fp = do readFileWith fd ff fp = do
cfp <- canonicalizePath fp cfp <- canonicalizePath' fp
let fn = topDir cfp let fn = topDir cfp
bd = baseDir cfp bd = baseDir cfp
file <- handleDT fn $ do file <- handleDT fn $ do
@ -260,14 +263,14 @@ readFileWith fd ff fp = do
readFile :: FilePath -> IO (AnchoredFile FileInfo FileInfo) readFile :: FilePath -> IO (AnchoredFile FileInfo FileInfo)
readFile fp = readFileWith getFileInfo getFileInfo =<< canonicalizePath fp readFile fp = readFileWith getFileInfo getFileInfo =<< canonicalizePath' fp
-- |Build a list of AnchoredFile, given the path to a directory, filling -- |Build a list of AnchoredFile, given the path to a directory, filling
-- the free variables via `getFileInfo`. -- the free variables via `getFileInfo`.
readDirectory :: FilePath -> IO [AnchoredFile FileInfo FileInfo] readDirectory :: FilePath -> IO [AnchoredFile FileInfo FileInfo]
readDirectory fp = readDirectoryWith getFileInfo getFileInfo readDirectory fp = readDirectoryWith getFileInfo getFileInfo
=<< canonicalizePath fp =<< canonicalizePath' fp
-- | same as readDirectory but allows us to, for example, use -- | same as readDirectory but allows us to, for example, use
@ -277,7 +280,7 @@ readDirectoryWith :: (FilePath -> IO a)
-> FilePath -> FilePath
-> IO [AnchoredFile a b] -> IO [AnchoredFile a b]
readDirectoryWith fd ff p = buildWith' buildAtOnce' fd ff readDirectoryWith fd ff p = buildWith' buildAtOnce' fd ff
=<< canonicalizePath p =<< canonicalizePath' p
@ -304,7 +307,7 @@ buildWith' :: Builder a b
-> IO [AnchoredFile a b] -> IO [AnchoredFile a b]
buildWith' bf' fd ff p = buildWith' bf' fd ff p =
do do
cfp <- canonicalizePath p cfp <- canonicalizePath' p
tree <- bf' fd ff cfp tree <- bf' fd ff cfp
return $ fmap (cfp :/) (removeNonexistent tree) return $ fmap (cfp :/) (removeNonexistent tree)
@ -313,7 +316,7 @@ buildWith' bf' fd ff p =
-- IO function passed to our builder and finally executed here: -- IO function passed to our builder and finally executed here:
buildAtOnce' :: Builder a b buildAtOnce' :: Builder a b
buildAtOnce' fd ff p = do buildAtOnce' fd ff p = do
cfp <- canonicalizePath p cfp <- canonicalizePath' p
contents <- getAllDirsFiles cfp contents <- getAllDirsFiles cfp
for contents $ \n -> handleDT n $ do for contents $ \n -> handleDT n $ do
let subf = cfp </> n let subf = cfp </> n
@ -425,6 +428,30 @@ hiddenFile str
| otherwise = False | otherwise = False
-- |Like `normalise` from System.FilePath but removes occurences of '..'.
-- Note that this sort of misbehaves if the path contains symlink
-- components.
normalize :: FilePath -> FilePath
normalize fp =
joinPath $ foldl' ff [] (splitDirectories . normalise $ fp)
where
ff ["/"] ".." = ["/"]
ff x ".." = initDef [] x
ff x y = x ++ [y]
-- |Like `canonicalizePath` from System.Directory, but preserves the last
-- component if it's a symlink.
canonicalizePath' :: FilePath -> IO FilePath
canonicalizePath' fp = do
isSymlink <- PF.isSymbolicLink <$> PF.getSymbolicLinkStatus fp
if isSymlink
then do
cbase <- canonicalizePath (baseDir fp)
return $ cbase </> topDir fp
else canonicalizePath fp
---- IO HELPERS: ---- ---- IO HELPERS: ----
@ -436,7 +463,7 @@ goUp (bp :/ _) = Data.DirTree.readFile bp
goUp' :: FilePath -> IO (AnchoredFile FileInfo FileInfo) goUp' :: FilePath -> IO (AnchoredFile FileInfo FileInfo)
goUp' fp = do goUp' fp = do
cfp <- canonicalizePath fp cfp <- canonicalizePath' fp
Data.DirTree.readFile $ baseDir cfp Data.DirTree.readFile $ baseDir cfp

View File

@ -143,8 +143,8 @@ copyDir :: DirCopyMode
-> FilePath -- ^ destination dir -> FilePath -- ^ destination dir
-> IO () -> IO ()
copyDir cm from' to' = do copyDir cm from' to' = do
from <- canonicalizePath from' from <- canonicalizePath' from'
to <- canonicalizePath to' to <- canonicalizePath' to'
go from to go from to
where where
go from to = do go from to = do
@ -204,13 +204,13 @@ copyFile :: FilePath -- ^ source file
-> FilePath -- ^ destination file -> FilePath -- ^ destination file
-> IO () -> IO ()
copyFile from' to' = do copyFile from' to' = do
from <- canonicalizePath from' from <- canonicalizePath' from'
tod <- canonicalizePath (baseDir to') tod <- canonicalizePath' (baseDir to')
let to = tod </> takeFileName to' let to = tod </> takeFileName to'
fileSanityThrow from fileSanityThrow from
throwNotAbsolute to throwNotAbsolute to
throwDirDoesExist to throwDirDoesExist to
toC <- canonicalizePath (takeDirectory to) toC <- canonicalizePath' (takeDirectory to)
let to' = toC </> takeFileName to let to' = toC </> takeFileName to
throwSameFile from to' throwSameFile from to'
SD.copyFile from to' SD.copyFile from to'
@ -226,8 +226,8 @@ copyFile from' to' = do
-- * anything that `copyFile` throws -- * anything that `copyFile` throws
copyFileToDir :: FilePath -> FilePath -> IO () copyFileToDir :: FilePath -> FilePath -> IO ()
copyFileToDir from' to' = do copyFileToDir from' to' = do
from <- canonicalizePath from' from <- canonicalizePath' from'
to <- canonicalizePath to' to <- canonicalizePath' to'
let name = takeFileName from let name = takeFileName from
dirSanityThrow to dirSanityThrow to
copyFile from (to </> name) copyFile from (to </> name)
@ -254,7 +254,7 @@ easyCopy cm from to = doFileOrDir from (copyDir cm from to)
-- * anything that `removeFile` throws -- * anything that `removeFile` throws
deleteFile :: FilePath -> IO () deleteFile :: FilePath -> IO ()
deleteFile fp' = do deleteFile fp' = do
fp <- canonicalizePath fp' fp <- canonicalizePath' fp'
fileSanityThrow fp fileSanityThrow fp
removeFile fp removeFile fp
@ -269,7 +269,7 @@ deleteFile fp' = do
-- * anything that `removeDirectory` throws -- * anything that `removeDirectory` throws
deleteDir :: FilePath -> IO () deleteDir :: FilePath -> IO ()
deleteDir fp' = do deleteDir fp' = do
fp <- canonicalizePath fp' fp <- canonicalizePath' fp'
dirSanityThrow fp dirSanityThrow fp
removeDirectory fp removeDirectory fp
@ -283,7 +283,7 @@ deleteDir fp' = do
-- * anything that `removeDirectoryRecursive` throws -- * anything that `removeDirectoryRecursive` throws
deleteDirRecursive :: FilePath -> IO () deleteDirRecursive :: FilePath -> IO ()
deleteDirRecursive fp' = do deleteDirRecursive fp' = do
fp <- canonicalizePath fp' fp <- canonicalizePath' fp'
dirSanityThrow fp dirSanityThrow fp
removeDirectoryRecursive fp removeDirectoryRecursive fp
@ -297,7 +297,7 @@ deleteDirRecursive fp' = do
-- * anything that `deleteDir`/`deleteFile` throws -- * anything that `deleteDir`/`deleteFile` throws
easyDelete :: FilePath -> IO () easyDelete :: FilePath -> IO ()
easyDelete fp' = do easyDelete fp' = do
fp <- canonicalizePath fp' fp <- canonicalizePath' fp'
doFileOrDir fp (deleteDir fp) (deleteFile fp) doFileOrDir fp (deleteDir fp) (deleteFile fp)
@ -316,7 +316,7 @@ easyDelete fp' = do
openFile :: FilePath openFile :: FilePath
-> IO ProcessHandle -> IO ProcessHandle
openFile fp' = do openFile fp' = do
fp <- canonicalizePath fp' fp <- canonicalizePath' fp'
fileSanityThrow fp fileSanityThrow fp
spawnProcess "xdg-open" [fp] spawnProcess "xdg-open" [fp]
@ -332,7 +332,7 @@ executeFile :: FilePath -- ^ program
-> [String] -- ^ arguments -> [String] -- ^ arguments
-> IO ProcessHandle -> IO ProcessHandle
executeFile prog' args = do executeFile prog' args = do
prog <- canonicalizePath prog' prog <- canonicalizePath' prog'
fileSanityThrow prog fileSanityThrow prog
unlessM (fileAccess prog False False True) (throw $ FileNotExecutable prog) unlessM (fileAccess prog False False True) (throw $ FileNotExecutable prog)
spawnProcess prog args spawnProcess prog args
@ -353,7 +353,7 @@ executeFile prog' args = do
-- * `throwFileDoesNotExist` if the filepath is neither a file or directory -- * `throwFileDoesNotExist` if the filepath is neither a file or directory
doFileOrDir :: FilePath -> IO () -> IO () -> IO () doFileOrDir :: FilePath -> IO () -> IO () -> IO ()
doFileOrDir fp' iod iof = do doFileOrDir fp' iod iof = do
fp <- canonicalizePath fp' fp <- canonicalizePath' fp'
isD <- doesDirectoryExist fp isD <- doesDirectoryExist fp
isF <- doesFileExist fp isF <- doesFileExist fp
case (isD, isF) of case (isD, isF) of