Compare commits
44 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| b603f72407 | |||
| 98ca6c5d86 | |||
| 8d948366f9 | |||
| 86e7496917 | |||
| 1b9b8cc886 | |||
| 395621b27a | |||
| 51da8bf5c2 | |||
| bebc96fa6d | |||
| 08fa277b31 | |||
| 51609781b2 | |||
| 3cb3a822d7 | |||
| 7fa4c041a9 | |||
| e66074af1c | |||
| 4032629407 | |||
| f2fe5a3419 | |||
| 5ac7450495 | |||
| b55cf6d9f3 | |||
| ae9a806c2e | |||
| 9c199c6da2 | |||
| eb72fce33f | |||
| 65bb09d133 | |||
| 908513da2b | |||
| 47dd729e8a | |||
| 620550dab4 | |||
| ebab5355bc | |||
| 8fdf1bf956 | |||
| 39913faed6 | |||
| 5ed249f5d6 | |||
| a8ccfc2587 | |||
| 8fec862304 | |||
| 646fe7cfea | |||
| 1bf27258c1 | |||
| 797dcaf725 | |||
| 0fa66cd581 | |||
| ee3ace362b | |||
| 05fcad14f1 | |||
| 456af3b1ab | |||
| f841a53985 | |||
| eb27c368e6 | |||
| c76df7f159 | |||
| 613754c58f | |||
| d8b0b99edf | |||
| 794c3a2fc4 | |||
| 8a28a5dd0f |
20
CHANGELOG
20
CHANGELOG
@@ -1,3 +1,23 @@
|
|||||||
|
0.7.2:
|
||||||
|
* fix tests, so they work with the sdist tarball too
|
||||||
|
* added the following function to HPath.IO: createSymlink
|
||||||
|
0.7.1:
|
||||||
|
* various cleanups and documentation improvements
|
||||||
|
* added the following functions to System.Posix.FilePath: splitSearchPath, getSearchPath, stripExtension, makeRelative, makeValid
|
||||||
|
0.7.0:
|
||||||
|
* use 'sendfile' from 'simple-sendfile' in _copyFile and do read/write as a fallback only
|
||||||
|
* add isFileName, hasParentDir, hiddenFile to System.Posix.FilePath
|
||||||
|
* add our own openFd version for more control
|
||||||
|
* small documentation improvements
|
||||||
|
* add a getDirectoryContents' version that works on Fd
|
||||||
|
* lift version constraints in benchmark
|
||||||
|
* remove fpToString and userStringToFP, use Data.ByteString.UTF8 directly instead
|
||||||
|
0.6.0:
|
||||||
|
* fixes 'throwDestinationInSource' to be more reliable.
|
||||||
|
* removes some unused HPathIOException constructors
|
||||||
|
* consistently provide exception constructor identifiers
|
||||||
|
* be less harsh when non-supported file types get passed to our functions, possibly ignoring them
|
||||||
|
* minor cleanups
|
||||||
0.5.9:
|
0.5.9:
|
||||||
* Adds our posix-paths fork and a lot of IO operations.
|
* Adds our posix-paths fork and a lot of IO operations.
|
||||||
0.5.8:
|
0.5.8:
|
||||||
|
|||||||
23
README.md
23
README.md
@@ -1,6 +1,6 @@
|
|||||||
# HPath
|
# HPath
|
||||||
|
|
||||||
[](http://travis-ci.org/hasufell/hpath)
|
[](https://gitter.im/hasufell/hpath?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) [](https://hackage.haskell.org/package/hpath) [](http://travis-ci.org/hasufell/hpath)
|
||||||
|
|
||||||
Support for well-typed paths in Haskell. Also provides ByteString based filepath
|
Support for well-typed paths in Haskell. Also provides ByteString based filepath
|
||||||
manipulation.
|
manipulation.
|
||||||
@@ -28,6 +28,8 @@ so it is forked as well and merged into this library.
|
|||||||
* safe filepath manipulation, never using String as filepath, but ByteString
|
* safe filepath manipulation, never using String as filepath, but ByteString
|
||||||
* still allowing sufficient control to interact with the underlying low-level calls
|
* still allowing sufficient control to interact with the underlying low-level calls
|
||||||
|
|
||||||
|
Note: this library was written for __posix__ systems and it will probably not support other systems.
|
||||||
|
|
||||||
## Differences to 'path'
|
## Differences to 'path'
|
||||||
|
|
||||||
* doesn't attempt to fake IO-related information into the path, so whether a path points to a file or directory is up to your IO-code to decide...
|
* doesn't attempt to fake IO-related information into the path, so whether a path points to a file or directory is up to your IO-code to decide...
|
||||||
@@ -43,9 +45,20 @@ so it is forked as well and merged into this library.
|
|||||||
|
|
||||||
## Differences to 'posix-paths'
|
## Differences to 'posix-paths'
|
||||||
|
|
||||||
* `hasTrailingPathSeparator` behaves in the same way as `System.FilePath`
|
|
||||||
* `dropTrailingPathSeparator` behaves in the same way as `System.FilePath`
|
|
||||||
* added various functions like `isValid`, `normalise` and `equalFilePath`
|
|
||||||
* uses the `word8` package for save word8 literals instead of `OverloadedStrings`
|
* uses the `word8` package for save word8 literals instead of `OverloadedStrings`
|
||||||
* has custom versions of `openFd` and `getDirectoryContents`
|
* `hasTrailingPathSeparator` and `dropTrailingPathSeparator` behave in the same way as their `System.FilePath` counterpart
|
||||||
|
* added various functions:
|
||||||
|
* `equalFilePath`
|
||||||
|
* `getSearchPath`
|
||||||
|
* `hasParentDir`
|
||||||
|
* `hiddenFile`
|
||||||
|
* `isFileName`
|
||||||
|
* `isValid`
|
||||||
|
* `makeRelative`
|
||||||
|
* `makeValid`
|
||||||
|
* `normalise`
|
||||||
|
* `splitSearchPath`
|
||||||
|
* `stripExtension`
|
||||||
|
* has a custom versions of `openFd` which allows more control over the flags than its unix package counterpart
|
||||||
|
* adds a `getDirectoryContents'` version that works on Fd
|
||||||
|
|
||||||
|
|||||||
19
hpath.cabal
19
hpath.cabal
@@ -1,5 +1,5 @@
|
|||||||
name: hpath
|
name: hpath
|
||||||
version: 0.5.9
|
version: 0.7.2
|
||||||
synopsis: Support for well-typed paths
|
synopsis: Support for well-typed paths
|
||||||
description: Support for well-typed paths, utilizing ByteString under the hood.
|
description: Support for well-typed paths, utilizing ByteString under the hood.
|
||||||
license: GPL-2
|
license: GPL-2
|
||||||
@@ -29,12 +29,14 @@ library
|
|||||||
HPath.Internal,
|
HPath.Internal,
|
||||||
System.Posix.Directory.Foreign,
|
System.Posix.Directory.Foreign,
|
||||||
System.Posix.Directory.Traversals,
|
System.Posix.Directory.Traversals,
|
||||||
|
System.Posix.FD,
|
||||||
System.Posix.FilePath
|
System.Posix.FilePath
|
||||||
build-depends: base >= 4.2 && <5
|
build-depends: base >= 4.2 && <5
|
||||||
, bytestring >= 0.9.2.0
|
, bytestring >= 0.9.2.0
|
||||||
, deepseq
|
, deepseq
|
||||||
, exceptions
|
, exceptions
|
||||||
, hspec
|
, hspec
|
||||||
|
, simple-sendfile >= 0.2.24
|
||||||
, unix >= 2.5
|
, unix >= 2.5
|
||||||
, unix-bytestring
|
, unix-bytestring
|
||||||
, utf8-string
|
, utf8-string
|
||||||
@@ -71,22 +73,24 @@ test-suite spec
|
|||||||
Hs-Source-Dirs: test
|
Hs-Source-Dirs: test
|
||||||
Main-Is: Main.hs
|
Main-Is: Main.hs
|
||||||
other-modules:
|
other-modules:
|
||||||
Spec
|
HPath.IO.CanonicalizePathSpec
|
||||||
HPath.IO.CopyDirRecursiveSpec
|
|
||||||
HPath.IO.CopyDirRecursiveOverwriteSpec
|
HPath.IO.CopyDirRecursiveOverwriteSpec
|
||||||
HPath.IO.CopyFileSpec
|
HPath.IO.CopyDirRecursiveSpec
|
||||||
HPath.IO.CopyFileOverwriteSpec
|
HPath.IO.CopyFileOverwriteSpec
|
||||||
|
HPath.IO.CopyFileSpec
|
||||||
HPath.IO.CreateDirSpec
|
HPath.IO.CreateDirSpec
|
||||||
HPath.IO.CreateRegularFileSpec
|
HPath.IO.CreateRegularFileSpec
|
||||||
|
HPath.IO.CreateSymlinkSpec
|
||||||
HPath.IO.DeleteDirRecursiveSpec
|
HPath.IO.DeleteDirRecursiveSpec
|
||||||
HPath.IO.DeleteDirSpec
|
HPath.IO.DeleteDirSpec
|
||||||
HPath.IO.DeleteFileSpec
|
HPath.IO.DeleteFileSpec
|
||||||
HPath.IO.GetDirsFilesSpec
|
HPath.IO.GetDirsFilesSpec
|
||||||
HPath.IO.GetFileTypeSpec
|
HPath.IO.GetFileTypeSpec
|
||||||
HPath.IO.MoveFileSpec
|
|
||||||
HPath.IO.MoveFileOverwriteSpec
|
HPath.IO.MoveFileOverwriteSpec
|
||||||
|
HPath.IO.MoveFileSpec
|
||||||
HPath.IO.RecreateSymlinkSpec
|
HPath.IO.RecreateSymlinkSpec
|
||||||
HPath.IO.RenameFileSpec
|
HPath.IO.RenameFileSpec
|
||||||
|
Spec
|
||||||
Utils
|
Utils
|
||||||
GHC-Options: -Wall
|
GHC-Options: -Wall
|
||||||
Build-Depends: base
|
Build-Depends: base
|
||||||
@@ -96,6 +100,7 @@ test-suite spec
|
|||||||
, hspec >= 1.3
|
, hspec >= 1.3
|
||||||
, process
|
, process
|
||||||
, unix
|
, unix
|
||||||
|
, unix-bytestring
|
||||||
, utf8-string
|
, utf8-string
|
||||||
|
|
||||||
benchmark bench.hs
|
benchmark bench.hs
|
||||||
@@ -110,9 +115,9 @@ benchmark bench.hs
|
|||||||
bytestring,
|
bytestring,
|
||||||
unix,
|
unix,
|
||||||
directory >= 1.1 && < 1.3,
|
directory >= 1.1 && < 1.3,
|
||||||
filepath >= 1.2 && < 1.4,
|
filepath >= 1.2 && < 1.5,
|
||||||
process >= 1.0 && < 1.3,
|
process >= 1.0 && < 1.3,
|
||||||
criterion >= 0.6 && < 0.9
|
criterion >= 0.6 && < 1.2
|
||||||
ghc-options: -O2
|
ghc-options: -O2
|
||||||
|
|
||||||
source-repository head
|
source-repository head
|
||||||
|
|||||||
79
src/HPath.hs
79
src/HPath.hs
@@ -45,9 +45,6 @@ module HPath
|
|||||||
,withAbsPath
|
,withAbsPath
|
||||||
,withRelPath
|
,withRelPath
|
||||||
,withFnPath
|
,withFnPath
|
||||||
-- * ByteString operations
|
|
||||||
,fpToString
|
|
||||||
,userStringToFP
|
|
||||||
)
|
)
|
||||||
where
|
where
|
||||||
|
|
||||||
@@ -57,10 +54,10 @@ import Control.Monad.Catch (MonadThrow(..))
|
|||||||
import Data.ByteString(ByteString, stripPrefix)
|
import Data.ByteString(ByteString, stripPrefix)
|
||||||
#else
|
#else
|
||||||
import Data.ByteString(ByteString)
|
import Data.ByteString(ByteString)
|
||||||
|
import qualified Data.List as L
|
||||||
#endif
|
#endif
|
||||||
import qualified Data.ByteString as BS
|
import qualified Data.ByteString as BS
|
||||||
import Data.Data
|
import Data.Data
|
||||||
import qualified Data.List as L
|
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.Word8
|
import Data.Word8
|
||||||
import HPath.Internal
|
import HPath.Internal
|
||||||
@@ -109,19 +106,19 @@ pattern Path x <- (MkPath x)
|
|||||||
--
|
--
|
||||||
-- Throws: 'PathParseException'
|
-- Throws: 'PathParseException'
|
||||||
--
|
--
|
||||||
-- >>> parseAbs "/abc" :: Maybe (Path Abs)
|
-- >>> parseAbs "/abc" :: Maybe (Path Abs)
|
||||||
-- Just "/abc"
|
-- Just "/abc"
|
||||||
-- >>> parseAbs "/" :: Maybe (Path Abs)
|
-- >>> parseAbs "/" :: Maybe (Path Abs)
|
||||||
-- Just "/"
|
-- Just "/"
|
||||||
-- >>> parseAbs "/abc/def" :: Maybe (Path Abs)
|
-- >>> parseAbs "/abc/def" :: Maybe (Path Abs)
|
||||||
-- Just "/abc/def"
|
-- Just "/abc/def"
|
||||||
-- >>> parseAbs "/abc/def/.///" :: Maybe (Path Abs)
|
-- >>> parseAbs "/abc/def/.///" :: Maybe (Path Abs)
|
||||||
-- Just "/abc/def/"
|
-- Just "/abc/def/"
|
||||||
-- >>> parseAbs "abc" :: Maybe (Path Abs)
|
-- >>> parseAbs "abc" :: Maybe (Path Abs)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseAbs "" :: Maybe (Path Abs)
|
-- >>> parseAbs "" :: Maybe (Path Abs)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseAbs "/abc/../foo" :: Maybe (Path Abs)
|
-- >>> parseAbs "/abc/../foo" :: Maybe (Path Abs)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
parseAbs :: MonadThrow m
|
parseAbs :: MonadThrow m
|
||||||
=> ByteString -> m (Path Abs)
|
=> ByteString -> m (Path Abs)
|
||||||
@@ -141,23 +138,23 @@ parseAbs filepath =
|
|||||||
--
|
--
|
||||||
-- Throws: 'PathParseException'
|
-- Throws: 'PathParseException'
|
||||||
--
|
--
|
||||||
-- >>> parseRel "abc" :: Maybe (Path Rel)
|
-- >>> parseRel "abc" :: Maybe (Path Rel)
|
||||||
-- Just "abc"
|
-- Just "abc"
|
||||||
-- >>> parseRel "def/" :: Maybe (Path Rel)
|
-- >>> parseRel "def/" :: Maybe (Path Rel)
|
||||||
-- Just "def/"
|
-- Just "def/"
|
||||||
-- >>> parseRel "abc/def" :: Maybe (Path Rel)
|
-- >>> parseRel "abc/def" :: Maybe (Path Rel)
|
||||||
-- Just "abc/def"
|
-- Just "abc/def"
|
||||||
-- >>> parseRel "abc/def/." :: Maybe (Path Rel)
|
-- >>> parseRel "abc/def/." :: Maybe (Path Rel)
|
||||||
-- Just "abc/def/"
|
-- Just "abc/def/"
|
||||||
-- >>> parseRel "/abc" :: Maybe (Path Rel)
|
-- >>> parseRel "/abc" :: Maybe (Path Rel)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseRel "" :: Maybe (Path Rel)
|
-- >>> parseRel "" :: Maybe (Path Rel)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseRel "abc/../foo" :: Maybe (Path Rel)
|
-- >>> parseRel "abc/../foo" :: Maybe (Path Rel)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseRel "." :: Maybe (Path Rel)
|
-- >>> parseRel "." :: Maybe (Path Rel)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseRel ".." :: Maybe (Path Rel)
|
-- >>> parseRel ".." :: Maybe (Path Rel)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
parseRel :: MonadThrow m
|
parseRel :: MonadThrow m
|
||||||
=> ByteString -> m (Path Rel)
|
=> ByteString -> m (Path Rel)
|
||||||
@@ -176,25 +173,25 @@ parseRel filepath =
|
|||||||
--
|
--
|
||||||
-- Throws: 'PathParseException'
|
-- Throws: 'PathParseException'
|
||||||
--
|
--
|
||||||
-- >>> parseFn "abc" :: Maybe (Path Fn)
|
-- >>> parseFn "abc" :: Maybe (Path Fn)
|
||||||
-- Just "abc"
|
-- Just "abc"
|
||||||
-- >>> parseFn "..." :: Maybe (Path Fn)
|
-- >>> parseFn "..." :: Maybe (Path Fn)
|
||||||
-- Just "..."
|
-- Just "..."
|
||||||
-- >>> parseFn "def/" :: Maybe (Path Fn)
|
-- >>> parseFn "def/" :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseFn "abc/def" :: Maybe (Path Fn)
|
-- >>> parseFn "abc/def" :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseFn "abc/def/." :: Maybe (Path Fn)
|
-- >>> parseFn "abc/def/." :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseFn "/abc" :: Maybe (Path Fn)
|
-- >>> parseFn "/abc" :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseFn "" :: Maybe (Path Fn)
|
-- >>> parseFn "" :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseFn "abc/../foo" :: Maybe (Path Fn)
|
-- >>> parseFn "abc/../foo" :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseFn "." :: Maybe (Path Fn)
|
-- >>> parseFn "." :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> parseFn ".." :: Maybe (Path Fn)
|
-- >>> parseFn ".." :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
parseFn :: MonadThrow m
|
parseFn :: MonadThrow m
|
||||||
=> ByteString -> m (Path Fn)
|
=> ByteString -> m (Path Fn)
|
||||||
@@ -237,13 +234,13 @@ fromRel = toFilePath
|
|||||||
-- because this library is IO-agnostic and makes no assumptions about
|
-- because this library is IO-agnostic and makes no assumptions about
|
||||||
-- file types.
|
-- file types.
|
||||||
--
|
--
|
||||||
-- >>> (MkPath "/") </> (MkPath "file" :: Path Rel)
|
-- >>> (MkPath "/") </> (MkPath "file" :: Path Rel)
|
||||||
-- "/file"
|
-- "/file"
|
||||||
-- >>> (MkPath "/path/to") </> (MkPath "file" :: Path Rel)
|
-- >>> (MkPath "/path/to") </> (MkPath "file" :: Path Rel)
|
||||||
-- "/path/to/file"
|
-- "/path/to/file"
|
||||||
-- >>> (MkPath "/") </> (MkPath "file/lal" :: Path Rel)
|
-- >>> (MkPath "/") </> (MkPath "file/lal" :: Path Rel)
|
||||||
-- "/file/lal"
|
-- "/file/lal"
|
||||||
-- >>> (MkPath "/") </> (MkPath "file/" :: Path Rel)
|
-- >>> (MkPath "/") </> (MkPath "file/" :: Path Rel)
|
||||||
-- "/file/"
|
-- "/file/"
|
||||||
(</>) :: RelC r => Path b -> Path r -> Path b
|
(</>) :: RelC r => Path b -> Path r -> Path b
|
||||||
(</>) (MkPath a) (MkPath b) = MkPath (a' `BS.append` b)
|
(</>) (MkPath a) (MkPath b) = MkPath (a' `BS.append` b)
|
||||||
@@ -257,15 +254,15 @@ fromRel = toFilePath
|
|||||||
--
|
--
|
||||||
-- The bases must match.
|
-- The bases must match.
|
||||||
--
|
--
|
||||||
-- >>> (MkPath "/lal/lad") `stripDir` (MkPath "/lal/lad/fad") :: Maybe (Path Rel)
|
-- >>> (MkPath "/lal/lad") `stripDir` (MkPath "/lal/lad/fad") :: Maybe (Path Rel)
|
||||||
-- Just "fad"
|
-- Just "fad"
|
||||||
-- >>> (MkPath "lal/lad") `stripDir` (MkPath "lal/lad/fad") :: Maybe (Path Rel)
|
-- >>> (MkPath "lal/lad") `stripDir` (MkPath "lal/lad/fad") :: Maybe (Path Rel)
|
||||||
-- Just "fad"
|
-- Just "fad"
|
||||||
-- >>> (MkPath "/") `stripDir` (MkPath "/") :: Maybe (Path Rel)
|
-- >>> (MkPath "/") `stripDir` (MkPath "/") :: Maybe (Path Rel)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> (MkPath "/lal/lad/fad") `stripDir` (MkPath "/lal/lad") :: Maybe (Path Rel)
|
-- >>> (MkPath "/lal/lad/fad") `stripDir` (MkPath "/lal/lad") :: Maybe (Path Rel)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
-- >>> (MkPath "fad") `stripDir` (MkPath "fad") :: Maybe (Path Rel)
|
-- >>> (MkPath "fad") `stripDir` (MkPath "fad") :: Maybe (Path Rel)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
stripDir :: MonadThrow m
|
stripDir :: MonadThrow m
|
||||||
=> Path b -> Path b -> m (Path Rel)
|
=> Path b -> Path b -> m (Path Rel)
|
||||||
@@ -281,15 +278,15 @@ stripDir (MkPath p) (MkPath l) =
|
|||||||
-- | Is p a parent of the given location? Implemented in terms of
|
-- | Is p a parent of the given location? Implemented in terms of
|
||||||
-- 'stripDir'. The bases must match.
|
-- 'stripDir'. The bases must match.
|
||||||
--
|
--
|
||||||
-- >>> (MkPath "/lal/lad") `isParentOf` (MkPath "/lal/lad/fad")
|
-- >>> (MkPath "/lal/lad") `isParentOf` (MkPath "/lal/lad/fad")
|
||||||
-- True
|
-- True
|
||||||
-- >>> (MkPath "lal/lad") `isParentOf` (MkPath "lal/lad/fad")
|
-- >>> (MkPath "lal/lad") `isParentOf` (MkPath "lal/lad/fad")
|
||||||
-- True
|
-- True
|
||||||
-- >>> (MkPath "/") `isParentOf` (MkPath "/")
|
-- >>> (MkPath "/") `isParentOf` (MkPath "/")
|
||||||
-- False
|
-- False
|
||||||
-- >>> (MkPath "/lal/lad/fad") `isParentOf` (MkPath "/lal/lad")
|
-- >>> (MkPath "/lal/lad/fad") `isParentOf` (MkPath "/lal/lad")
|
||||||
-- False
|
-- False
|
||||||
-- >>> (MkPath "fad") `isParentOf` (MkPath "fad")
|
-- >>> (MkPath "fad") `isParentOf` (MkPath "fad")
|
||||||
-- False
|
-- False
|
||||||
isParentOf :: Path b -> Path b -> Bool
|
isParentOf :: Path b -> Path b -> Bool
|
||||||
isParentOf p l = isJust (stripDir p l :: Maybe (Path Rel))
|
isParentOf p l = isJust (stripDir p l :: Maybe (Path Rel))
|
||||||
@@ -333,7 +330,7 @@ dirname (MkPath fp) = MkPath (takeDirectory $ dropTrailingPathSeparator fp)
|
|||||||
--
|
--
|
||||||
-- >>> basename (MkPath "/abc/def/dod") :: Maybe (Path Fn)
|
-- >>> basename (MkPath "/abc/def/dod") :: Maybe (Path Fn)
|
||||||
-- Just "dod"
|
-- Just "dod"
|
||||||
-- >>> basename (MkPath "/") :: Maybe (Path Fn)
|
-- >>> basename (MkPath "/") :: Maybe (Path Fn)
|
||||||
-- Nothing
|
-- Nothing
|
||||||
basename :: MonadThrow m => Path b -> m (Path Fn)
|
basename :: MonadThrow m => Path b -> m (Path Fn)
|
||||||
basename (MkPath l)
|
basename (MkPath l)
|
||||||
|
|||||||
120
src/HPath/IO.hs
120
src/HPath/IO.hs
@@ -27,7 +27,8 @@
|
|||||||
--
|
--
|
||||||
-- Note: `BlockDevice`, `CharacterDevice`, `NamedPipe` and `Socket`
|
-- Note: `BlockDevice`, `CharacterDevice`, `NamedPipe` and `Socket`
|
||||||
-- are not explicitly supported right now. Calling any of these
|
-- are not explicitly supported right now. Calling any of these
|
||||||
-- functions on such a file may throw an exception.
|
-- functions on such a file may throw an exception or just do
|
||||||
|
-- nothing.
|
||||||
|
|
||||||
{-# LANGUAGE PackageImports #-}
|
{-# LANGUAGE PackageImports #-}
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
@@ -55,6 +56,7 @@ module HPath.IO
|
|||||||
-- * File creation
|
-- * File creation
|
||||||
, createRegularFile
|
, createRegularFile
|
||||||
, createDir
|
, createDir
|
||||||
|
, createSymlink
|
||||||
-- * File renaming/moving
|
-- * File renaming/moving
|
||||||
, renameFile
|
, renameFile
|
||||||
, moveFile
|
, moveFile
|
||||||
@@ -79,7 +81,6 @@ import Control.Applicative
|
|||||||
import Control.Exception
|
import Control.Exception
|
||||||
(
|
(
|
||||||
bracket
|
bracket
|
||||||
, bracketOnError
|
|
||||||
, throwIO
|
, throwIO
|
||||||
)
|
)
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
@@ -106,6 +107,8 @@ import Data.Word
|
|||||||
import Foreign.C.Error
|
import Foreign.C.Error
|
||||||
(
|
(
|
||||||
eEXIST
|
eEXIST
|
||||||
|
, eINVAL
|
||||||
|
, eNOSYS
|
||||||
, eNOTEMPTY
|
, eNOTEMPTY
|
||||||
, eXDEV
|
, eXDEV
|
||||||
)
|
)
|
||||||
@@ -135,6 +138,11 @@ import System.IO.Error
|
|||||||
catchIOError
|
catchIOError
|
||||||
, ioeGetErrorType
|
, ioeGetErrorType
|
||||||
)
|
)
|
||||||
|
import System.Linux.Sendfile
|
||||||
|
(
|
||||||
|
sendfileFd
|
||||||
|
, FileRange(..)
|
||||||
|
)
|
||||||
import System.Posix.ByteString
|
import System.Posix.ByteString
|
||||||
(
|
(
|
||||||
exclusive
|
exclusive
|
||||||
@@ -171,6 +179,10 @@ import System.Posix.Files.ByteString
|
|||||||
import qualified System.Posix.Files.ByteString as PF
|
import qualified System.Posix.Files.ByteString as PF
|
||||||
import qualified "unix" System.Posix.IO.ByteString as SPI
|
import qualified "unix" System.Posix.IO.ByteString as SPI
|
||||||
import qualified "unix-bytestring" System.Posix.IO.ByteString as SPB
|
import qualified "unix-bytestring" System.Posix.IO.ByteString as SPB
|
||||||
|
import System.Posix.FD
|
||||||
|
(
|
||||||
|
openFd
|
||||||
|
)
|
||||||
import qualified System.Posix.Directory.Traversals as SPDT
|
import qualified System.Posix.Directory.Traversals as SPDT
|
||||||
import qualified System.Posix.Directory.Foreign as SPDF
|
import qualified System.Posix.Directory.Foreign as SPDF
|
||||||
import qualified System.Posix.Process.ByteString as SPP
|
import qualified System.Posix.Process.ByteString as SPP
|
||||||
@@ -185,6 +197,11 @@ import System.Posix.Types
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-------------
|
||||||
|
--[ Types ]--
|
||||||
|
-------------
|
||||||
|
|
||||||
|
|
||||||
data FileType = Directory
|
data FileType = Directory
|
||||||
| RegularFile
|
| RegularFile
|
||||||
| SymbolicLink
|
| SymbolicLink
|
||||||
@@ -223,8 +240,8 @@ data FileType = Directory
|
|||||||
-- - `PermissionDenied` if source directory can't be opened
|
-- - `PermissionDenied` if source directory can't be opened
|
||||||
-- - `InvalidArgument` if source directory is wrong type (symlink)
|
-- - `InvalidArgument` if source directory is wrong type (symlink)
|
||||||
-- - `InvalidArgument` if source directory is wrong type (regular file)
|
-- - `InvalidArgument` if source directory is wrong type (regular file)
|
||||||
-- - `SameFile` if source and destination are the same file (`HPathIOException`)
|
|
||||||
-- - `AlreadyExists` if destination already exists
|
-- - `AlreadyExists` if destination already exists
|
||||||
|
-- - `SameFile` if source and destination are the same file (`HPathIOException`)
|
||||||
-- - `DestinationInSource` if destination is contained in source (`HPathIOException`)
|
-- - `DestinationInSource` if destination is contained in source (`HPathIOException`)
|
||||||
copyDirRecursive :: Path Abs -- ^ source dir
|
copyDirRecursive :: Path Abs -- ^ source dir
|
||||||
-> Path Abs -- ^ full destination
|
-> Path Abs -- ^ full destination
|
||||||
@@ -252,8 +269,7 @@ copyDirRecursive fromp destdirp
|
|||||||
SymbolicLink -> recreateSymlink f newdest
|
SymbolicLink -> recreateSymlink f newdest
|
||||||
Directory -> go f newdest
|
Directory -> go f newdest
|
||||||
RegularFile -> copyFile f newdest
|
RegularFile -> copyFile f newdest
|
||||||
_ -> ioError $ userError $ "No idea what to do with the" ++
|
_ -> return ()
|
||||||
"given filetype: " ++ show ftype
|
|
||||||
|
|
||||||
|
|
||||||
-- |Like `copyDirRecursive` except it overwrites contents of directories
|
-- |Like `copyDirRecursive` except it overwrites contents of directories
|
||||||
@@ -298,8 +314,7 @@ copyDirRecursiveOverwrite fromp destdirp
|
|||||||
>> recreateSymlink f newdest
|
>> recreateSymlink f newdest
|
||||||
Directory -> go f newdest
|
Directory -> go f newdest
|
||||||
RegularFile -> copyFileOverwrite f newdest
|
RegularFile -> copyFileOverwrite f newdest
|
||||||
_ -> ioError $ userError $ "No idea what to do with the" ++
|
_ -> return ()
|
||||||
"given filetype: " ++ show ftype
|
|
||||||
|
|
||||||
|
|
||||||
-- |Recreate a symlink.
|
-- |Recreate a symlink.
|
||||||
@@ -335,10 +350,10 @@ recreateSymlink symsource newsym
|
|||||||
-- - `PermissionDenied` if source directory can't be opened
|
-- - `PermissionDenied` if source directory can't be opened
|
||||||
-- - `InvalidArgument` if source file is wrong type (symlink)
|
-- - `InvalidArgument` if source file is wrong type (symlink)
|
||||||
-- - `InvalidArgument` if source file is wrong type (directory)
|
-- - `InvalidArgument` if source file is wrong type (directory)
|
||||||
-- - `SameFile` if source and destination are the same file (`HPathIOException`)
|
|
||||||
-- - `AlreadyExists` if destination already exists
|
-- - `AlreadyExists` if destination already exists
|
||||||
|
-- - `SameFile` if source and destination are the same file (`HPathIOException`)
|
||||||
--
|
--
|
||||||
-- Note: calls `sendfile`
|
-- Note: calls `sendfile` and possibly `read`/`write` as fallback
|
||||||
copyFile :: Path Abs -- ^ source file
|
copyFile :: Path Abs -- ^ source file
|
||||||
-> Path Abs -- ^ destination file
|
-> Path Abs -- ^ destination file
|
||||||
-> IO ()
|
-> IO ()
|
||||||
@@ -355,8 +370,7 @@ copyFile from to = do
|
|||||||
--
|
--
|
||||||
-- Safety/reliability concerns:
|
-- Safety/reliability concerns:
|
||||||
--
|
--
|
||||||
-- * not atomic
|
-- * not atomic, since it uses read/write
|
||||||
-- * falls back to delete-copy method with explicit checks
|
|
||||||
--
|
--
|
||||||
-- Throws:
|
-- Throws:
|
||||||
--
|
--
|
||||||
@@ -367,7 +381,7 @@ copyFile from to = do
|
|||||||
-- - `InvalidArgument` if source file is wrong type (directory)
|
-- - `InvalidArgument` if source file is wrong type (directory)
|
||||||
-- - `SameFile` if source and destination are the same file (`HPathIOException`)
|
-- - `SameFile` if source and destination are the same file (`HPathIOException`)
|
||||||
--
|
--
|
||||||
-- Note: calls `sendfile`
|
-- Note: calls `sendfile` and possibly `read`/`write` as fallback
|
||||||
copyFileOverwrite :: Path Abs -- ^ source file
|
copyFileOverwrite :: Path Abs -- ^ source file
|
||||||
-> Path Abs -- ^ destination file
|
-> Path Abs -- ^ destination file
|
||||||
-> IO ()
|
-> IO ()
|
||||||
@@ -395,24 +409,35 @@ _copyFile :: [SPDF.Flags]
|
|||||||
-> IO ()
|
-> IO ()
|
||||||
_copyFile sflags dflags from to
|
_copyFile sflags dflags from to
|
||||||
=
|
=
|
||||||
-- TODO: add sendfile support
|
-- from sendfile(2) manpage:
|
||||||
|
-- Applications may wish to fall back to read(2)/write(2) in the case
|
||||||
|
-- where sendfile() fails with EINVAL or ENOSYS.
|
||||||
withAbsPath to $ \to' -> withAbsPath from $ \from' ->
|
withAbsPath to $ \to' -> withAbsPath from $ \from' ->
|
||||||
void $ fallbackCopy from' to'
|
catchErrno [eINVAL, eNOSYS]
|
||||||
|
(sendFileCopy from' to')
|
||||||
|
(void $ readWriteCopy from' to')
|
||||||
where
|
where
|
||||||
-- low-level copy operation utilizing read(2)/write(2)
|
copyWith copyAction source dest =
|
||||||
-- in case `sendFileCopy` fails/is unsupported
|
bracket (openFd source SPI.ReadOnly sflags Nothing)
|
||||||
fallbackCopy source dest =
|
|
||||||
bracket (SPDT.openFd source SPI.ReadOnly sflags Nothing)
|
|
||||||
SPI.closeFd
|
SPI.closeFd
|
||||||
$ \sfd -> do
|
$ \sfd -> do
|
||||||
fileM <- System.Posix.Files.ByteString.fileMode
|
fileM <- System.Posix.Files.ByteString.fileMode
|
||||||
<$> getFdStatus sfd
|
<$> getFdStatus sfd
|
||||||
bracketeer (SPDT.openFd dest SPI.WriteOnly
|
bracketeer (openFd dest SPI.WriteOnly
|
||||||
dflags $ Just fileM)
|
dflags $ Just fileM)
|
||||||
SPI.closeFd
|
SPI.closeFd
|
||||||
(\fd -> SPI.closeFd fd >> deleteFile to)
|
(\fd -> SPI.closeFd fd >> deleteFile to)
|
||||||
$ \dfd -> allocaBytes (fromIntegral bufSize) $ \buf ->
|
$ \dfd -> copyAction sfd dfd
|
||||||
write' sfd dfd buf 0
|
-- this is low-level stuff utilizing sendfile(2) for speed
|
||||||
|
sendFileCopy :: ByteString -> ByteString -> IO ()
|
||||||
|
sendFileCopy = copyWith
|
||||||
|
(\sfd dfd -> sendfileFd dfd sfd EntireFile $ return ())
|
||||||
|
-- low-level copy operation utilizing read(2)/write(2)
|
||||||
|
-- in case `sendFileCopy` fails/is unsupported
|
||||||
|
readWriteCopy :: ByteString -> ByteString -> IO Int
|
||||||
|
readWriteCopy = copyWith
|
||||||
|
(\sfd dfd -> allocaBytes (fromIntegral bufSize)
|
||||||
|
$ \buf -> write' sfd dfd buf 0)
|
||||||
where
|
where
|
||||||
bufSize :: CSize
|
bufSize :: CSize
|
||||||
bufSize = 8192
|
bufSize = 8192
|
||||||
@@ -442,8 +467,7 @@ easyCopy from to = do
|
|||||||
SymbolicLink -> recreateSymlink from to
|
SymbolicLink -> recreateSymlink from to
|
||||||
RegularFile -> copyFile from to
|
RegularFile -> copyFile from to
|
||||||
Directory -> copyDirRecursive from to
|
Directory -> copyDirRecursive from to
|
||||||
_ -> ioError $ userError $ "No idea what to do with the" ++
|
_ -> return ()
|
||||||
"given filetype: " ++ show ftype
|
|
||||||
|
|
||||||
|
|
||||||
-- |Like `easyCopy` except it overwrites the destination if it already exists.
|
-- |Like `easyCopy` except it overwrites the destination if it already exists.
|
||||||
@@ -459,8 +483,7 @@ easyCopyOverwrite from to = do
|
|||||||
>> recreateSymlink from to
|
>> recreateSymlink from to
|
||||||
RegularFile -> copyFileOverwrite from to
|
RegularFile -> copyFileOverwrite from to
|
||||||
Directory -> copyDirRecursiveOverwrite from to
|
Directory -> copyDirRecursiveOverwrite from to
|
||||||
_ -> ioError $ userError $ "No idea what to do with the" ++
|
_ -> return ()
|
||||||
"given filetype: " ++ show ftype
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -472,7 +495,7 @@ easyCopyOverwrite from to = do
|
|||||||
---------------------
|
---------------------
|
||||||
|
|
||||||
|
|
||||||
-- |Deletes the given file, does not follow symlinks. Raises `eISDIR`
|
-- |Deletes the given file. Raises `eISDIR`
|
||||||
-- if run on a directory. Does not follow symbolic links.
|
-- if run on a directory. Does not follow symbolic links.
|
||||||
--
|
--
|
||||||
-- Throws:
|
-- Throws:
|
||||||
@@ -526,8 +549,7 @@ deleteDirRecursive p =
|
|||||||
SymbolicLink -> deleteFile file
|
SymbolicLink -> deleteFile file
|
||||||
Directory -> deleteDirRecursive file
|
Directory -> deleteDirRecursive file
|
||||||
RegularFile -> deleteFile file
|
RegularFile -> deleteFile file
|
||||||
_ -> ioError $ userError $ "No idea what to do with the" ++
|
_ -> return ()
|
||||||
"given filetype: " ++ show ftype
|
|
||||||
removeDirectory . toFilePath $ p
|
removeDirectory . toFilePath $ p
|
||||||
|
|
||||||
|
|
||||||
@@ -546,8 +568,7 @@ easyDelete p = do
|
|||||||
SymbolicLink -> deleteFile p
|
SymbolicLink -> deleteFile p
|
||||||
Directory -> deleteDirRecursive p
|
Directory -> deleteDirRecursive p
|
||||||
RegularFile -> deleteFile p
|
RegularFile -> deleteFile p
|
||||||
_ -> ioError $ userError $ "No idea what to do with the" ++
|
_ -> return ()
|
||||||
"given filetype: " ++ show ftype
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -607,6 +628,20 @@ createDir :: Path Abs -> IO ()
|
|||||||
createDir dest = createDirectory (fromAbs dest) newDirPerms
|
createDir dest = createDirectory (fromAbs dest) newDirPerms
|
||||||
|
|
||||||
|
|
||||||
|
-- |Create a symlink.
|
||||||
|
--
|
||||||
|
-- Throws:
|
||||||
|
--
|
||||||
|
-- - `PermissionDenied` if output directory cannot be written to
|
||||||
|
-- - `AlreadyExists` if destination file already exists
|
||||||
|
--
|
||||||
|
-- Note: calls `symlink`
|
||||||
|
createSymlink :: Path Abs -- ^ destination file
|
||||||
|
-> ByteString -- ^ path the symlink points to
|
||||||
|
-> IO ()
|
||||||
|
createSymlink dest sympoint
|
||||||
|
= createSymbolicLink sympoint (fromAbs dest)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
----------------------------
|
----------------------------
|
||||||
@@ -629,8 +664,8 @@ createDir dest = createDirectory (fromAbs dest) newDirPerms
|
|||||||
-- - `PermissionDenied` if output directory cannot be written to
|
-- - `PermissionDenied` if output directory cannot be written to
|
||||||
-- - `PermissionDenied` if source directory cannot be opened
|
-- - `PermissionDenied` if source directory cannot be opened
|
||||||
-- - `UnsupportedOperation` if source and destination are on different devices
|
-- - `UnsupportedOperation` if source and destination are on different devices
|
||||||
-- - `FileDoesExist` if destination file already exists
|
-- - `FileDoesExist` if destination file already exists (`HPathIOException`)
|
||||||
-- - `DirDoesExist` if destination directory already exists
|
-- - `DirDoesExist` if destination directory already exists (`HPathIOException`)
|
||||||
-- - `SameFile` if destination and source are the same file (`HPathIOException`)
|
-- - `SameFile` if destination and source are the same file (`HPathIOException`)
|
||||||
--
|
--
|
||||||
-- Note: calls `rename` (but does not allow to rename over existing files)
|
-- Note: calls `rename` (but does not allow to rename over existing files)
|
||||||
@@ -656,8 +691,8 @@ renameFile fromf tof = do
|
|||||||
-- - `NoSuchThing` if source file does not exist
|
-- - `NoSuchThing` if source file does not exist
|
||||||
-- - `PermissionDenied` if output directory cannot be written to
|
-- - `PermissionDenied` if output directory cannot be written to
|
||||||
-- - `PermissionDenied` if source directory cannot be opened
|
-- - `PermissionDenied` if source directory cannot be opened
|
||||||
-- - `FileDoesExist` if destination file already exists
|
-- - `FileDoesExist` if destination file already exists (`HPathIOException`)
|
||||||
-- - `DirDoesExist` if destination directory already exists
|
-- - `DirDoesExist` if destination directory already exists (`HPathIOException`)
|
||||||
-- - `SameFile` if destination and source are the same file (`HPathIOException`)
|
-- - `SameFile` if destination and source are the same file (`HPathIOException`)
|
||||||
--
|
--
|
||||||
-- Note: calls `rename` (but does not allow to rename over existing files)
|
-- Note: calls `rename` (but does not allow to rename over existing files)
|
||||||
@@ -705,8 +740,7 @@ moveFileOverwrite from to = do
|
|||||||
Directory -> do
|
Directory -> do
|
||||||
exists <- doesDirectoryExist to
|
exists <- doesDirectoryExist to
|
||||||
when (exists && writable) (deleteDir to)
|
when (exists && writable) (deleteDir to)
|
||||||
_ -> ioError $ userError $ "Don't know how to handle filetype " ++
|
_ -> return ()
|
||||||
show ft
|
|
||||||
moveFile from to
|
moveFile from to
|
||||||
|
|
||||||
|
|
||||||
@@ -757,14 +791,12 @@ newDirPerms
|
|||||||
getDirsFiles :: Path Abs -- ^ dir to read
|
getDirsFiles :: Path Abs -- ^ dir to read
|
||||||
-> IO [Path Abs]
|
-> IO [Path Abs]
|
||||||
getDirsFiles p =
|
getDirsFiles p =
|
||||||
withAbsPath p $ \fp ->
|
withAbsPath p $ \fp -> do
|
||||||
bracketOnError (SPDT.openFd fp SPI.ReadOnly [SPDF.oNofollow] Nothing)
|
fd <- openFd fp SPI.ReadOnly [SPDF.oNofollow] Nothing
|
||||||
SPI.closeFd
|
return
|
||||||
$ \fd ->
|
. catMaybes
|
||||||
return
|
. fmap (\x -> (</>) p <$> (parseMaybe . snd $ x))
|
||||||
. catMaybes
|
=<< getDirectoryContents' fd
|
||||||
. fmap (\x -> (</>) p <$> (parseMaybe . snd $ x))
|
|
||||||
=<< getDirectoryContents' fd
|
|
||||||
where
|
where
|
||||||
parseMaybe :: ByteString -> Maybe (Path Fn)
|
parseMaybe :: ByteString -> Maybe (Path Fn)
|
||||||
parseMaybe = parseFn
|
parseMaybe = parseFn
|
||||||
|
|||||||
7
src/HPath/IO.hs-boot
Normal file
7
src/HPath/IO.hs-boot
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
module HPath.IO where
|
||||||
|
|
||||||
|
|
||||||
|
import HPath
|
||||||
|
|
||||||
|
canonicalizePath :: Path Abs -> IO (Path Abs)
|
||||||
|
|
||||||
@@ -12,7 +12,44 @@
|
|||||||
{-# LANGUAGE DeriveDataTypeable #-}
|
{-# LANGUAGE DeriveDataTypeable #-}
|
||||||
{-# LANGUAGE ScopedTypeVariables #-}
|
{-# LANGUAGE ScopedTypeVariables #-}
|
||||||
|
|
||||||
module HPath.IO.Errors where
|
module HPath.IO.Errors
|
||||||
|
(
|
||||||
|
-- * Types
|
||||||
|
HPathIOException(..)
|
||||||
|
|
||||||
|
-- * Exception identifiers
|
||||||
|
, isFileDoesNotExist
|
||||||
|
, isDirDoesNotExist
|
||||||
|
, isSameFile
|
||||||
|
, isDestinationInSource
|
||||||
|
, isFileDoesExist
|
||||||
|
, isDirDoesExist
|
||||||
|
, isInvalidOperation
|
||||||
|
, isCan'tOpenDirectory
|
||||||
|
, isCopyFailed
|
||||||
|
|
||||||
|
-- * Path based functions
|
||||||
|
, throwFileDoesExist
|
||||||
|
, throwDirDoesExist
|
||||||
|
, throwFileDoesNotExist
|
||||||
|
, throwDirDoesNotExist
|
||||||
|
, throwSameFile
|
||||||
|
, sameFile
|
||||||
|
, throwDestinationInSource
|
||||||
|
, doesFileExist
|
||||||
|
, doesDirectoryExist
|
||||||
|
, isWritable
|
||||||
|
, canOpenDirectory
|
||||||
|
, throwCantOpenDirectory
|
||||||
|
|
||||||
|
-- * Error handling functions
|
||||||
|
, catchErrno
|
||||||
|
, rethrowErrnoAs
|
||||||
|
, handleIOError
|
||||||
|
, bracketeer
|
||||||
|
, reactOnError
|
||||||
|
)
|
||||||
|
where
|
||||||
|
|
||||||
|
|
||||||
import Control.Applicative
|
import Control.Applicative
|
||||||
@@ -29,6 +66,10 @@ import Data.ByteString
|
|||||||
(
|
(
|
||||||
ByteString
|
ByteString
|
||||||
)
|
)
|
||||||
|
import Data.ByteString.UTF8
|
||||||
|
(
|
||||||
|
toString
|
||||||
|
)
|
||||||
import Data.Data
|
import Data.Data
|
||||||
(
|
(
|
||||||
Data(..)
|
Data(..)
|
||||||
@@ -44,6 +85,10 @@ import GHC.IO.Exception
|
|||||||
IOErrorType
|
IOErrorType
|
||||||
)
|
)
|
||||||
import HPath
|
import HPath
|
||||||
|
import {-# SOURCE #-} HPath.IO
|
||||||
|
(
|
||||||
|
canonicalizePath
|
||||||
|
)
|
||||||
import HPath.IO.Utils
|
import HPath.IO.Utils
|
||||||
import System.IO.Error
|
import System.IO.Error
|
||||||
(
|
(
|
||||||
@@ -62,47 +107,32 @@ import qualified System.Posix.Files.ByteString as PF
|
|||||||
|
|
||||||
data HPathIOException = FileDoesNotExist ByteString
|
data HPathIOException = FileDoesNotExist ByteString
|
||||||
| DirDoesNotExist ByteString
|
| DirDoesNotExist ByteString
|
||||||
| PathNotAbsolute ByteString
|
|
||||||
| FileNotExecutable ByteString
|
|
||||||
| SameFile ByteString ByteString
|
| SameFile ByteString ByteString
|
||||||
| NotAFile ByteString
|
|
||||||
| NotADir ByteString
|
|
||||||
| DestinationInSource ByteString ByteString
|
| DestinationInSource ByteString ByteString
|
||||||
| FileDoesExist ByteString
|
| FileDoesExist ByteString
|
||||||
| DirDoesExist ByteString
|
| DirDoesExist ByteString
|
||||||
| IsSymlink ByteString
|
|
||||||
| InvalidOperation String
|
| InvalidOperation String
|
||||||
| InvalidFileName
|
|
||||||
| Can'tOpenDirectory ByteString
|
| Can'tOpenDirectory ByteString
|
||||||
| CopyFailed String
|
| CopyFailed String
|
||||||
| MoveFailed String
|
|
||||||
deriving (Typeable, Eq, Data)
|
deriving (Typeable, Eq, Data)
|
||||||
|
|
||||||
|
|
||||||
instance Show HPathIOException where
|
instance Show HPathIOException where
|
||||||
show (FileDoesNotExist fp) = "File does not exist:" ++ fpToString fp
|
show (FileDoesNotExist fp) = "File does not exist:" ++ toString fp
|
||||||
show (DirDoesNotExist fp) = "Directory does not exist: "
|
show (DirDoesNotExist fp) = "Directory does not exist: "
|
||||||
++ fpToString fp
|
++ toString fp
|
||||||
show (PathNotAbsolute fp) = "Path not absolute: " ++ fpToString fp
|
show (SameFile fp1 fp2) = toString fp1
|
||||||
show (FileNotExecutable fp) = "File not executable: "
|
++ " and " ++ toString fp2
|
||||||
++ fpToString fp
|
|
||||||
show (SameFile fp1 fp2) = fpToString fp1
|
|
||||||
++ " and " ++ fpToString fp2
|
|
||||||
++ " are the same file!"
|
++ " are the same file!"
|
||||||
show (NotAFile fp) = "Not a file: " ++ fpToString fp
|
show (DestinationInSource fp1 fp2) = toString fp1
|
||||||
show (NotADir fp) = "Not a directory: " ++ fpToString fp
|
|
||||||
show (DestinationInSource fp1 fp2) = fpToString fp1
|
|
||||||
++ " is contained in "
|
++ " is contained in "
|
||||||
++ fpToString fp2
|
++ toString fp2
|
||||||
show (FileDoesExist fp) = "File does exist: " ++ fpToString fp
|
show (FileDoesExist fp) = "File does exist: " ++ toString fp
|
||||||
show (DirDoesExist fp) = "Directory does exist: " ++ fpToString fp
|
show (DirDoesExist fp) = "Directory does exist: " ++ toString fp
|
||||||
show (IsSymlink fp) = "Is a symlink: " ++ fpToString fp
|
|
||||||
show (InvalidOperation str) = "Invalid operation: " ++ str
|
show (InvalidOperation str) = "Invalid operation: " ++ str
|
||||||
show InvalidFileName = "Invalid file name!"
|
|
||||||
show (Can'tOpenDirectory fp) = "Can't open directory: "
|
show (Can'tOpenDirectory fp) = "Can't open directory: "
|
||||||
++ fpToString fp
|
++ toString fp
|
||||||
show (CopyFailed str) = "Copying failed: " ++ str
|
show (CopyFailed str) = "Copying failed: " ++ str
|
||||||
show (MoveFailed str) = "Moving failed: " ++ str
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -110,24 +140,23 @@ instance Exception HPathIOException
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
isDestinationInSource :: HPathIOException -> Bool
|
|
||||||
isDestinationInSource (DestinationInSource _ _) = True
|
|
||||||
isDestinationInSource _ = False
|
|
||||||
|
|
||||||
|
|
||||||
isSameFile :: HPathIOException -> Bool
|
-----------------------------
|
||||||
isSameFile (SameFile _ _) = True
|
--[ Exception identifiers ]--
|
||||||
isSameFile _ = False
|
-----------------------------
|
||||||
|
|
||||||
|
isFileDoesNotExist, isDirDoesNotExist, isSameFile, isDestinationInSource, isFileDoesExist, isDirDoesExist, isInvalidOperation, isCan'tOpenDirectory, isCopyFailed :: HPathIOException -> Bool
|
||||||
|
isFileDoesNotExist ex = toConstr (ex :: HPathIOException) == toConstr FileDoesNotExist{}
|
||||||
|
isDirDoesNotExist ex = toConstr (ex :: HPathIOException) == toConstr DirDoesNotExist{}
|
||||||
|
isSameFile ex = toConstr (ex :: HPathIOException) == toConstr SameFile{}
|
||||||
|
isDestinationInSource ex = toConstr (ex :: HPathIOException) == toConstr DestinationInSource{}
|
||||||
|
isFileDoesExist ex = toConstr (ex :: HPathIOException) == toConstr FileDoesExist{}
|
||||||
|
isDirDoesExist ex = toConstr (ex :: HPathIOException) == toConstr DirDoesExist{}
|
||||||
|
isInvalidOperation ex = toConstr (ex :: HPathIOException) == toConstr InvalidOperation{}
|
||||||
|
isCan'tOpenDirectory ex = toConstr (ex :: HPathIOException) == toConstr Can'tOpenDirectory{}
|
||||||
|
isCopyFailed ex = toConstr (ex :: HPathIOException) == toConstr CopyFailed{}
|
||||||
|
|
||||||
isFileDoesExist :: HPathIOException -> Bool
|
|
||||||
isFileDoesExist (FileDoesExist _) = True
|
|
||||||
isFileDoesExist _ = False
|
|
||||||
|
|
||||||
|
|
||||||
isDirDoesExist :: HPathIOException -> Bool
|
|
||||||
isDirDoesExist (DirDoesExist _) = True
|
|
||||||
isDirDoesExist _ = False
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -184,18 +213,18 @@ sameFile fp1 fp2 =
|
|||||||
else return False
|
else return False
|
||||||
|
|
||||||
|
|
||||||
|
-- TODO: make this more robust when destination does not exist
|
||||||
-- |Checks whether the destination directory is contained
|
-- |Checks whether the destination directory is contained
|
||||||
-- within the source directory by comparing the device+file ID of the
|
-- within the source directory by comparing the device+file ID of the
|
||||||
-- source directory with all device+file IDs of the parent directories
|
-- source directory with all device+file IDs of the parent directories
|
||||||
-- of the destination.
|
-- of the destination.
|
||||||
throwDestinationInSource :: Path Abs -- ^ source dir
|
throwDestinationInSource :: Path Abs -- ^ source dir
|
||||||
-> Path Abs -- ^ full destination, `dirname dest`
|
-> Path Abs -- ^ full destination, @dirname dest@
|
||||||
-- must exist
|
-- must exist
|
||||||
-> IO ()
|
-> IO ()
|
||||||
throwDestinationInSource source dest = do
|
throwDestinationInSource source dest = do
|
||||||
dest' <- (\x -> maybe x (\y -> x </> y) $ basename dest)
|
dest' <- (\x -> maybe x (\y -> x </> y) $ basename dest)
|
||||||
{- <$> (canonicalizePath $ P.dirname dest) -}
|
<$> (canonicalizePath $ dirname dest)
|
||||||
<$> (return $ dirname dest)
|
|
||||||
dids <- forM (getAllParents dest') $ \p -> do
|
dids <- forM (getAllParents dest') $ \p -> do
|
||||||
fs <- PF.getSymbolicLinkStatus (fromAbs p)
|
fs <- PF.getSymbolicLinkStatus (fromAbs p)
|
||||||
return (PF.deviceID fs, PF.fileID fs)
|
return (PF.deviceID fs, PF.fileID fs)
|
||||||
@@ -242,7 +271,7 @@ canOpenDirectory fp =
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
-- |Throws a `Can'tOpenDirectory` FmIOException if the directory at the given
|
-- |Throws a `Can'tOpenDirectory` HPathIOException if the directory at the given
|
||||||
-- path cannot be opened.
|
-- path cannot be opened.
|
||||||
throwCantOpenDirectory :: Path Abs -> IO ()
|
throwCantOpenDirectory :: Path Abs -> IO ()
|
||||||
throwCantOpenDirectory fp =
|
throwCantOpenDirectory fp =
|
||||||
@@ -308,8 +337,8 @@ bracketeer before after afterEx thing =
|
|||||||
|
|
||||||
|
|
||||||
reactOnError :: IO a
|
reactOnError :: IO a
|
||||||
-> [(IOErrorType, IO a)] -- ^ reaction on IO errors
|
-> [(IOErrorType, IO a)] -- ^ reaction on IO errors
|
||||||
-> [(HPathIOException, IO a)] -- ^ reaction on FmIOException
|
-> [(HPathIOException, IO a)] -- ^ reaction on HPathIOException
|
||||||
-> IO a
|
-> IO a
|
||||||
reactOnError a ios fmios =
|
reactOnError a ios fmios =
|
||||||
a `catches` [iohandler, fmiohandler]
|
a `catches` [iohandler, fmiohandler]
|
||||||
|
|||||||
@@ -1,9 +1,24 @@
|
|||||||
|
-- |
|
||||||
|
-- Module : System.Posix.Directory.Traversals
|
||||||
|
-- Copyright : © 2016 Julian Ospald
|
||||||
|
-- License : BSD3
|
||||||
|
--
|
||||||
|
-- Maintainer : Julian Ospald <hasufell@posteo.de>
|
||||||
|
-- Stability : experimental
|
||||||
|
-- Portability : portable
|
||||||
|
--
|
||||||
|
-- Traversal and read operations on directories.
|
||||||
|
|
||||||
|
|
||||||
{-# LANGUAGE ForeignFunctionInterface #-}
|
{-# LANGUAGE ForeignFunctionInterface #-}
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE PackageImports #-}
|
||||||
{-# LANGUAGE TupleSections #-}
|
{-# LANGUAGE TupleSections #-}
|
||||||
{-# LANGUAGE ViewPatterns #-}
|
{-# LANGUAGE ViewPatterns #-}
|
||||||
|
|
||||||
{-# OPTIONS_GHC -Wall #-}
|
{-# OPTIONS_GHC -Wall #-}
|
||||||
|
|
||||||
|
|
||||||
module System.Posix.Directory.Traversals (
|
module System.Posix.Directory.Traversals (
|
||||||
|
|
||||||
getDirectoryContents
|
getDirectoryContents
|
||||||
@@ -17,7 +32,7 @@ module System.Posix.Directory.Traversals (
|
|||||||
, readDirEnt
|
, readDirEnt
|
||||||
, packDirStream
|
, packDirStream
|
||||||
, unpackDirStream
|
, unpackDirStream
|
||||||
, openFd
|
, fdOpendir
|
||||||
|
|
||||||
, realpath
|
, realpath
|
||||||
) where
|
) where
|
||||||
@@ -36,6 +51,7 @@ import System.Posix.Directory.ByteString as PosixBS
|
|||||||
import System.Posix.Files.ByteString
|
import System.Posix.Files.ByteString
|
||||||
|
|
||||||
import System.IO.Unsafe
|
import System.IO.Unsafe
|
||||||
|
import "unix" System.Posix.IO.ByteString (closeFd)
|
||||||
import Unsafe.Coerce (unsafeCoerce)
|
import Unsafe.Coerce (unsafeCoerce)
|
||||||
import Foreign.C.Error
|
import Foreign.C.Error
|
||||||
import Foreign.C.String
|
import Foreign.C.String
|
||||||
@@ -54,6 +70,8 @@ import Foreign.Storable
|
|||||||
-- Upon entering a directory, 'allDirectoryContents' will get all entries
|
-- Upon entering a directory, 'allDirectoryContents' will get all entries
|
||||||
-- strictly. However the returned list is lazy in that directories will only
|
-- strictly. However the returned list is lazy in that directories will only
|
||||||
-- be accessed on demand.
|
-- be accessed on demand.
|
||||||
|
--
|
||||||
|
-- Follows symbolic links for the input dir.
|
||||||
allDirectoryContents :: RawFilePath -> IO [RawFilePath]
|
allDirectoryContents :: RawFilePath -> IO [RawFilePath]
|
||||||
allDirectoryContents topdir = do
|
allDirectoryContents topdir = do
|
||||||
namesAndTypes <- getDirectoryContents topdir
|
namesAndTypes <- getDirectoryContents topdir
|
||||||
@@ -71,6 +89,8 @@ allDirectoryContents topdir = do
|
|||||||
return (topdir : concat paths)
|
return (topdir : concat paths)
|
||||||
|
|
||||||
-- | Get all files from a directory and its subdirectories strictly.
|
-- | Get all files from a directory and its subdirectories strictly.
|
||||||
|
--
|
||||||
|
-- Follows symbolic links for the input dir.
|
||||||
allDirectoryContents' :: RawFilePath -> IO [RawFilePath]
|
allDirectoryContents' :: RawFilePath -> IO [RawFilePath]
|
||||||
allDirectoryContents' = fmap reverse . traverseDirectory (\acc fp -> return (fp:acc)) []
|
allDirectoryContents' = fmap reverse . traverseDirectory (\acc fp -> return (fp:acc)) []
|
||||||
-- this uses traverseDirectory because it's more efficient than forcing the
|
-- this uses traverseDirectory because it's more efficient than forcing the
|
||||||
@@ -80,6 +100,8 @@ allDirectoryContents' = fmap reverse . traverseDirectory (\acc fp -> return (fp:
|
|||||||
-- files/subdirectories.
|
-- files/subdirectories.
|
||||||
--
|
--
|
||||||
-- This function allows for memory-efficient traversals.
|
-- This function allows for memory-efficient traversals.
|
||||||
|
--
|
||||||
|
-- Follows symbolic links for the input dir.
|
||||||
traverseDirectory :: (s -> RawFilePath -> IO s) -> s -> RawFilePath -> IO s
|
traverseDirectory :: (s -> RawFilePath -> IO s) -> s -> RawFilePath -> IO s
|
||||||
traverseDirectory act s0 topdir = toploop
|
traverseDirectory act s0 topdir = toploop
|
||||||
where
|
where
|
||||||
@@ -103,17 +125,17 @@ actOnDirContents :: RawFilePath
|
|||||||
-> IO b
|
-> IO b
|
||||||
actOnDirContents pathRelToTop b f =
|
actOnDirContents pathRelToTop b f =
|
||||||
modifyIOError ((`ioeSetFileName` (BS.unpack pathRelToTop)) .
|
modifyIOError ((`ioeSetFileName` (BS.unpack pathRelToTop)) .
|
||||||
(`ioeSetLocation` "findBSTypRel")) $ do
|
(`ioeSetLocation` "findBSTypRel")) $
|
||||||
bracket
|
bracket
|
||||||
(openDirStream pathRelToTop)
|
(openDirStream pathRelToTop)
|
||||||
(Posix.closeDirStream)
|
Posix.closeDirStream
|
||||||
(\dirp -> loop dirp b)
|
(\dirp -> loop dirp b)
|
||||||
where
|
where
|
||||||
loop dirp b' = do
|
loop dirp b' = do
|
||||||
(typ,e) <- readDirEnt dirp
|
(typ,e) <- readDirEnt dirp
|
||||||
if (e == "")
|
if (e == "")
|
||||||
then return b'
|
then return b'
|
||||||
else do
|
else
|
||||||
if (e == "." || e == "..")
|
if (e == "." || e == "..")
|
||||||
then loop dirp b'
|
then loop dirp b'
|
||||||
else f typ (pathRelToTop </> e) b' >>= loop dirp
|
else f typ (pathRelToTop </> e) b' >>= loop dirp
|
||||||
@@ -154,9 +176,6 @@ foreign import ccall "realpath"
|
|||||||
foreign import ccall unsafe "fdopendir"
|
foreign import ccall unsafe "fdopendir"
|
||||||
c_fdopendir :: Posix.Fd -> IO (Ptr ())
|
c_fdopendir :: Posix.Fd -> IO (Ptr ())
|
||||||
|
|
||||||
foreign import ccall unsafe "open"
|
|
||||||
c_open :: CString -> CInt -> Posix.CMode -> IO CInt
|
|
||||||
|
|
||||||
----------------------------------------------------------
|
----------------------------------------------------------
|
||||||
-- less dodgy but still lower-level
|
-- less dodgy but still lower-level
|
||||||
|
|
||||||
@@ -189,81 +208,53 @@ readDirEnt (unpackDirStream -> dirp) =
|
|||||||
else throwErrno "readDirEnt"
|
else throwErrno "readDirEnt"
|
||||||
|
|
||||||
|
|
||||||
|
-- |Gets all directory contents (not recursively).
|
||||||
getDirectoryContents :: RawFilePath -> IO [(DirType, RawFilePath)]
|
getDirectoryContents :: RawFilePath -> IO [(DirType, RawFilePath)]
|
||||||
getDirectoryContents path =
|
getDirectoryContents path =
|
||||||
modifyIOError ((`ioeSetFileName` (BS.unpack path)) .
|
modifyIOError ((`ioeSetFileName` (BS.unpack path)) .
|
||||||
(`ioeSetLocation` "System.Posix.Directory.Traversals.getDirectoryContents")) $ do
|
(`ioeSetLocation` "System.Posix.Directory.Traversals.getDirectoryContents")) $
|
||||||
bracket
|
bracket
|
||||||
(PosixBS.openDirStream path)
|
(PosixBS.openDirStream path)
|
||||||
PosixBS.closeDirStream
|
PosixBS.closeDirStream
|
||||||
loop
|
_dirloop
|
||||||
where
|
|
||||||
loop dirp = do
|
|
||||||
t@(_typ,e) <- readDirEnt dirp
|
|
||||||
if BS.null e then return [] else do
|
|
||||||
es <- loop dirp
|
|
||||||
return (t:es)
|
|
||||||
|
|
||||||
|
|
||||||
|
-- |Binding to @fdopendir(3)@.
|
||||||
fdOpendir :: Posix.Fd -> IO DirStream
|
fdOpendir :: Posix.Fd -> IO DirStream
|
||||||
fdOpendir fd =
|
fdOpendir fd =
|
||||||
packDirStream <$> throwErrnoIfNull "fdOpendir" (c_fdopendir fd)
|
packDirStream <$> throwErrnoIfNull "fdOpendir" (c_fdopendir fd)
|
||||||
|
|
||||||
|
|
||||||
|
-- |Like `getDirectoryContents` except for a file descriptor.
|
||||||
|
--
|
||||||
|
-- To avoid complicated error checks, the file descriptor is
|
||||||
|
-- __always__ closed, even if `fdOpendir` fails. Usually, this
|
||||||
|
-- only happens on successful `fdOpendir` and after the directory
|
||||||
|
-- stream is closed. Also see the manpage of @fdopendir(3)@ for
|
||||||
|
-- more details.
|
||||||
getDirectoryContents' :: Posix.Fd -> IO [(DirType, RawFilePath)]
|
getDirectoryContents' :: Posix.Fd -> IO [(DirType, RawFilePath)]
|
||||||
getDirectoryContents' fd =
|
getDirectoryContents' fd = do
|
||||||
bracket
|
dirstream <- fdOpendir fd `catchIOError` \e -> do
|
||||||
(fdOpendir fd)
|
closeFd fd
|
||||||
PosixBS.closeDirStream
|
ioError e
|
||||||
loop
|
-- closeDirStream closes the filedescriptor
|
||||||
where
|
finally (_dirloop dirstream) (PosixBS.closeDirStream dirstream)
|
||||||
loop dirp = do
|
|
||||||
t@(_typ,e) <- readDirEnt dirp
|
|
||||||
if BS.null e then return [] else do
|
|
||||||
es <- loop dirp
|
|
||||||
return (t:es)
|
|
||||||
|
|
||||||
|
|
||||||
open_ :: CString
|
_dirloop :: DirStream -> IO [(DirType, RawFilePath)]
|
||||||
-> Posix.OpenMode
|
{-# INLINE _dirloop #-}
|
||||||
-> [Flags]
|
_dirloop dirp = do
|
||||||
-> Maybe Posix.FileMode
|
t@(_typ,e) <- readDirEnt dirp
|
||||||
-> IO Posix.Fd
|
if BS.null e then return [] else do
|
||||||
open_ str how optional_flags maybe_mode = do
|
es <- _dirloop dirp
|
||||||
fd <- c_open str all_flags mode_w
|
return (t:es)
|
||||||
return (Posix.Fd fd)
|
|
||||||
where
|
|
||||||
all_flags = unionFlags $ optional_flags ++ [open_mode] ++ creat
|
|
||||||
|
|
||||||
|
|
||||||
(creat, mode_w) = case maybe_mode of
|
|
||||||
Nothing -> ([],0)
|
|
||||||
Just x -> ([oCreat], x)
|
|
||||||
|
|
||||||
open_mode = case how of
|
|
||||||
Posix.ReadOnly -> oRdonly
|
|
||||||
Posix.WriteOnly -> oWronly
|
|
||||||
Posix.ReadWrite -> oRdwr
|
|
||||||
|
|
||||||
|
|
||||||
-- |Open and optionally create this file. See 'System.Posix.Files'
|
|
||||||
-- for information on how to use the 'FileMode' type.
|
|
||||||
openFd :: RawFilePath
|
|
||||||
-> Posix.OpenMode
|
|
||||||
-> [Flags]
|
|
||||||
-> Maybe Posix.FileMode
|
|
||||||
-> IO Posix.Fd
|
|
||||||
openFd name how optional_flags maybe_mode =
|
|
||||||
withFilePath name $ \str ->
|
|
||||||
throwErrnoPathIfMinus1Retry "openFd" name $
|
|
||||||
open_ str how optional_flags maybe_mode
|
|
||||||
|
|
||||||
|
|
||||||
-- | return the canonicalized absolute pathname
|
-- | return the canonicalized absolute pathname
|
||||||
--
|
--
|
||||||
-- like canonicalizePath, but uses realpath(3)
|
-- like canonicalizePath, but uses @realpath(3)@
|
||||||
realpath :: RawFilePath -> IO RawFilePath
|
realpath :: RawFilePath -> IO RawFilePath
|
||||||
realpath inp = do
|
realpath inp =
|
||||||
allocaBytes pathMax $ \tmp -> do
|
allocaBytes pathMax $ \tmp -> do
|
||||||
void $ BS.useAsCString inp $ \cstr -> throwErrnoIfNull "realpath" $ c_realpath cstr tmp
|
void $ BS.useAsCString inp $ \cstr -> throwErrnoIfNull "realpath" $ c_realpath cstr tmp
|
||||||
BS.packCString tmp
|
BS.packCString tmp
|
||||||
|
|||||||
75
src/System/Posix/FD.hs
Normal file
75
src/System/Posix/FD.hs
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
-- |
|
||||||
|
-- Module : System.Posix.FD
|
||||||
|
-- Copyright : © 2016 Julian Ospald
|
||||||
|
-- License : BSD3
|
||||||
|
--
|
||||||
|
-- Maintainer : Julian Ospald <hasufell@posteo.de>
|
||||||
|
-- Stability : experimental
|
||||||
|
-- Portability : portable
|
||||||
|
--
|
||||||
|
-- Provides an alternative for `System.Posix.IO.ByteString.openFd`
|
||||||
|
-- which gives us more control on what status flags to pass to the
|
||||||
|
-- low-level @open(2)@ call, in contrast to the unix package.
|
||||||
|
|
||||||
|
|
||||||
|
{-# LANGUAGE ForeignFunctionInterface #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE TupleSections #-}
|
||||||
|
|
||||||
|
{-# OPTIONS_GHC -Wall #-}
|
||||||
|
|
||||||
|
|
||||||
|
module System.Posix.FD (
|
||||||
|
openFd
|
||||||
|
) where
|
||||||
|
|
||||||
|
|
||||||
|
import Foreign.C.String
|
||||||
|
import Foreign.C.Types
|
||||||
|
import System.Posix.Directory.Foreign
|
||||||
|
import qualified System.Posix as Posix
|
||||||
|
import System.Posix.ByteString.FilePath
|
||||||
|
|
||||||
|
|
||||||
|
foreign import ccall unsafe "open"
|
||||||
|
c_open :: CString -> CInt -> Posix.CMode -> IO CInt
|
||||||
|
|
||||||
|
|
||||||
|
open_ :: CString
|
||||||
|
-> Posix.OpenMode
|
||||||
|
-> [Flags]
|
||||||
|
-> Maybe Posix.FileMode
|
||||||
|
-> IO Posix.Fd
|
||||||
|
open_ str how optional_flags maybe_mode = do
|
||||||
|
fd <- c_open str all_flags mode_w
|
||||||
|
return (Posix.Fd fd)
|
||||||
|
where
|
||||||
|
all_flags = unionFlags $ optional_flags ++ [open_mode] ++ creat
|
||||||
|
|
||||||
|
|
||||||
|
(creat, mode_w) = case maybe_mode of
|
||||||
|
Nothing -> ([],0)
|
||||||
|
Just x -> ([oCreat], x)
|
||||||
|
|
||||||
|
open_mode = case how of
|
||||||
|
Posix.ReadOnly -> oRdonly
|
||||||
|
Posix.WriteOnly -> oWronly
|
||||||
|
Posix.ReadWrite -> oRdwr
|
||||||
|
|
||||||
|
|
||||||
|
-- |Open and optionally create this file. See 'System.Posix.Files'
|
||||||
|
-- for information on how to use the 'FileMode' type.
|
||||||
|
--
|
||||||
|
-- Note that passing @Just x@ as the 4th argument triggers the
|
||||||
|
-- `oCreat` status flag, which must be set when you pass in `oExcl`
|
||||||
|
-- to the status flags. Also see the manpage for @open(2)@.
|
||||||
|
openFd :: RawFilePath
|
||||||
|
-> Posix.OpenMode
|
||||||
|
-> [Flags] -- ^ status flags of @open(2)@
|
||||||
|
-> Maybe Posix.FileMode -- ^ @Just x@ => creates the file with the given modes, Nothing => the file must exist.
|
||||||
|
-> IO Posix.Fd
|
||||||
|
openFd name how optional_flags maybe_mode =
|
||||||
|
withFilePath name $ \str ->
|
||||||
|
throwErrnoPathIfMinus1Retry "openFd" name $
|
||||||
|
open_ str how optional_flags maybe_mode
|
||||||
|
|
||||||
@@ -11,6 +11,8 @@
|
|||||||
--
|
--
|
||||||
-- Not all functions of "System.FilePath" are implemented yet. Feel free to contribute!
|
-- Not all functions of "System.FilePath" are implemented yet. Feel free to contribute!
|
||||||
|
|
||||||
|
|
||||||
|
{-# LANGUAGE CPP #-}
|
||||||
{-# LANGUAGE TupleSections #-}
|
{-# LANGUAGE TupleSections #-}
|
||||||
|
|
||||||
{-# OPTIONS_GHC -Wall #-}
|
{-# OPTIONS_GHC -Wall #-}
|
||||||
@@ -18,7 +20,7 @@
|
|||||||
|
|
||||||
module System.Posix.FilePath (
|
module System.Posix.FilePath (
|
||||||
|
|
||||||
-- * Separators
|
-- * Separator predicates
|
||||||
pathSeparator
|
pathSeparator
|
||||||
, isPathSeparator
|
, isPathSeparator
|
||||||
, searchPathSeparator
|
, searchPathSeparator
|
||||||
@@ -26,7 +28,11 @@ module System.Posix.FilePath (
|
|||||||
, extSeparator
|
, extSeparator
|
||||||
, isExtSeparator
|
, isExtSeparator
|
||||||
|
|
||||||
-- * File extensions
|
-- * $PATH methods
|
||||||
|
, splitSearchPath
|
||||||
|
, getSearchPath
|
||||||
|
|
||||||
|
-- * Extension functions
|
||||||
, splitExtension
|
, splitExtension
|
||||||
, takeExtension
|
, takeExtension
|
||||||
, replaceExtension
|
, replaceExtension
|
||||||
@@ -37,8 +43,9 @@ module System.Posix.FilePath (
|
|||||||
, splitExtensions
|
, splitExtensions
|
||||||
, dropExtensions
|
, dropExtensions
|
||||||
, takeExtensions
|
, takeExtensions
|
||||||
|
, stripExtension
|
||||||
|
|
||||||
-- * Filenames/Directory names
|
-- * Filename\/directory functions
|
||||||
, splitFileName
|
, splitFileName
|
||||||
, takeFileName
|
, takeFileName
|
||||||
, replaceFileName
|
, replaceFileName
|
||||||
@@ -47,92 +54,135 @@ module System.Posix.FilePath (
|
|||||||
, replaceBaseName
|
, replaceBaseName
|
||||||
, takeDirectory
|
, takeDirectory
|
||||||
, replaceDirectory
|
, replaceDirectory
|
||||||
|
|
||||||
-- * Path combinators and splitters
|
|
||||||
, combine
|
, combine
|
||||||
, (</>)
|
, (</>)
|
||||||
, splitPath
|
, splitPath
|
||||||
, joinPath
|
, joinPath
|
||||||
, splitDirectories
|
, splitDirectories
|
||||||
|
|
||||||
-- * Path conversions
|
-- * Trailing slash functions
|
||||||
, normalise
|
|
||||||
|
|
||||||
-- * Trailing path separator
|
|
||||||
, hasTrailingPathSeparator
|
, hasTrailingPathSeparator
|
||||||
, addTrailingPathSeparator
|
, addTrailingPathSeparator
|
||||||
, dropTrailingPathSeparator
|
, dropTrailingPathSeparator
|
||||||
|
|
||||||
-- * Queries
|
-- * File name manipulations
|
||||||
|
, normalise
|
||||||
|
, makeRelative
|
||||||
|
, equalFilePath
|
||||||
, isRelative
|
, isRelative
|
||||||
, isAbsolute
|
, isAbsolute
|
||||||
, isValid
|
, isValid
|
||||||
|
, makeValid
|
||||||
, isFileName
|
, isFileName
|
||||||
, hasParentDir
|
, hasParentDir
|
||||||
, equalFilePath
|
|
||||||
, hiddenFile
|
, hiddenFile
|
||||||
|
|
||||||
-- * Type conversion
|
|
||||||
, fpToString
|
|
||||||
, userStringToFP
|
|
||||||
|
|
||||||
, module System.Posix.ByteString.FilePath
|
, module System.Posix.ByteString.FilePath
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Data.ByteString (ByteString)
|
import Data.ByteString (ByteString)
|
||||||
import qualified Data.ByteString as BS
|
import qualified Data.ByteString as BS
|
||||||
import Data.ByteString.UTF8 (fromString, toString)
|
import Data.String (fromString)
|
||||||
import System.Posix.ByteString.FilePath
|
import System.Posix.ByteString.FilePath
|
||||||
|
import qualified System.Posix.Env.ByteString as PE
|
||||||
|
|
||||||
import Data.Maybe (isJust)
|
import Data.Maybe (isJust)
|
||||||
import Data.Word8
|
import Data.Word8
|
||||||
|
#if !MIN_VERSION_bytestring(0,10,8)
|
||||||
|
import qualified Data.List as L
|
||||||
|
#endif
|
||||||
import Control.Arrow (second)
|
import Control.Arrow (second)
|
||||||
|
|
||||||
-- $setup
|
-- $setup
|
||||||
-- >>> import Data.Char
|
-- >>> import Data.Char
|
||||||
|
-- >>> import Data.Maybe
|
||||||
-- >>> import Test.QuickCheck
|
-- >>> import Test.QuickCheck
|
||||||
-- >>> import Control.Applicative
|
-- >>> import Control.Applicative
|
||||||
-- >>> import qualified Data.ByteString as BS
|
-- >>> import qualified Data.ByteString as BS
|
||||||
-- >>> import Data.ByteString (ByteString)
|
|
||||||
-- >>> instance Arbitrary ByteString where arbitrary = BS.pack <$> arbitrary
|
-- >>> instance Arbitrary ByteString where arbitrary = BS.pack <$> arbitrary
|
||||||
-- >>> instance CoArbitrary ByteString where coarbitrary = coarbitrary . BS.unpack
|
-- >>> instance CoArbitrary ByteString where coarbitrary = coarbitrary . BS.unpack
|
||||||
--
|
--
|
||||||
-- >>> let _chr :: Word8 -> Char; _chr = chr . fromIntegral
|
-- >>> let _chr :: Word8 -> Char; _chr = chr . fromIntegral
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
------------------------
|
||||||
|
-- Separator predicates
|
||||||
|
|
||||||
|
|
||||||
-- | Path separator character
|
-- | Path separator character
|
||||||
pathSeparator :: Word8
|
pathSeparator :: Word8
|
||||||
pathSeparator = _slash
|
pathSeparator = _slash
|
||||||
|
|
||||||
|
|
||||||
-- | Check if a character is the path separator
|
-- | Check if a character is the path separator
|
||||||
--
|
--
|
||||||
-- prop> \n -> (_chr n == '/') == isPathSeparator n
|
-- prop> \n -> (_chr n == '/') == isPathSeparator n
|
||||||
isPathSeparator :: Word8 -> Bool
|
isPathSeparator :: Word8 -> Bool
|
||||||
isPathSeparator = (== pathSeparator)
|
isPathSeparator = (== pathSeparator)
|
||||||
|
|
||||||
|
|
||||||
-- | Search path separator
|
-- | Search path separator
|
||||||
searchPathSeparator :: Word8
|
searchPathSeparator :: Word8
|
||||||
searchPathSeparator = _colon
|
searchPathSeparator = _colon
|
||||||
|
|
||||||
|
|
||||||
-- | Check if a character is the search path separator
|
-- | Check if a character is the search path separator
|
||||||
--
|
--
|
||||||
-- prop> \n -> (_chr n == ':') == isSearchPathSeparator n
|
-- prop> \n -> (_chr n == ':') == isSearchPathSeparator n
|
||||||
isSearchPathSeparator :: Word8 -> Bool
|
isSearchPathSeparator :: Word8 -> Bool
|
||||||
isSearchPathSeparator = (== searchPathSeparator)
|
isSearchPathSeparator = (== searchPathSeparator)
|
||||||
|
|
||||||
|
|
||||||
-- | File extension separator
|
-- | File extension separator
|
||||||
extSeparator :: Word8
|
extSeparator :: Word8
|
||||||
extSeparator = _period
|
extSeparator = _period
|
||||||
|
|
||||||
|
|
||||||
-- | Check if a character is the file extension separator
|
-- | Check if a character is the file extension separator
|
||||||
--
|
--
|
||||||
-- prop> \n -> (_chr n == '.') == isExtSeparator n
|
-- prop> \n -> (_chr n == '.') == isExtSeparator n
|
||||||
isExtSeparator :: Word8 -> Bool
|
isExtSeparator :: Word8 -> Bool
|
||||||
isExtSeparator = (== extSeparator)
|
isExtSeparator = (== extSeparator)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
------------------------
|
------------------------
|
||||||
-- extension stuff
|
-- $PATH methods
|
||||||
|
|
||||||
|
|
||||||
|
-- | Take a ByteString, split it on the 'searchPathSeparator'.
|
||||||
|
-- Blank items are converted to @.@.
|
||||||
|
--
|
||||||
|
-- Follows the recommendations in
|
||||||
|
-- <http://www.opengroup.org/onlinepubs/009695399/basedefs/xbd_chap08.html>
|
||||||
|
--
|
||||||
|
-- >>> splitSearchPath "File1:File2:File3"
|
||||||
|
-- ["File1","File2","File3"]
|
||||||
|
-- >>> splitSearchPath "File1::File2:File3"
|
||||||
|
-- ["File1",".","File2","File3"]
|
||||||
|
-- >>> splitSearchPath ""
|
||||||
|
-- ["."]
|
||||||
|
splitSearchPath :: ByteString -> [RawFilePath]
|
||||||
|
splitSearchPath = f
|
||||||
|
where
|
||||||
|
f bs = let (pre, post) = BS.break isSearchPathSeparator bs
|
||||||
|
in if BS.null post
|
||||||
|
then g pre
|
||||||
|
else g pre ++ f (BS.tail post)
|
||||||
|
g x
|
||||||
|
| BS.null x = [BS.singleton _period]
|
||||||
|
| otherwise = [x]
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get a list of 'RawFilePath's in the $PATH variable.
|
||||||
|
getSearchPath :: IO [RawFilePath]
|
||||||
|
getSearchPath = fmap (maybe [] splitSearchPath) (PE.getEnv $ fromString "PATH")
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
------------------------
|
||||||
|
-- Extension functions
|
||||||
|
|
||||||
-- | Split a 'RawFilePath' into a path+filename and extension
|
-- | Split a 'RawFilePath' into a path+filename and extension
|
||||||
--
|
--
|
||||||
@@ -152,6 +202,7 @@ splitExtension x = if BS.null basename
|
|||||||
(path,file) = splitFileNameRaw x
|
(path,file) = splitFileNameRaw x
|
||||||
(basename,fileExt) = BS.breakEnd isExtSeparator file
|
(basename,fileExt) = BS.breakEnd isExtSeparator file
|
||||||
|
|
||||||
|
|
||||||
-- | Get the final extension from a 'RawFilePath'
|
-- | Get the final extension from a 'RawFilePath'
|
||||||
--
|
--
|
||||||
-- >>> takeExtension "file.exe"
|
-- >>> takeExtension "file.exe"
|
||||||
@@ -163,12 +214,14 @@ splitExtension x = if BS.null basename
|
|||||||
takeExtension :: RawFilePath -> ByteString
|
takeExtension :: RawFilePath -> ByteString
|
||||||
takeExtension = snd . splitExtension
|
takeExtension = snd . splitExtension
|
||||||
|
|
||||||
|
|
||||||
-- | Change a file's extension
|
-- | Change a file's extension
|
||||||
--
|
--
|
||||||
-- prop> \path -> let ext = takeExtension path in replaceExtension path ext == path
|
-- prop> \path -> let ext = takeExtension path in replaceExtension path ext == path
|
||||||
replaceExtension :: RawFilePath -> ByteString -> RawFilePath
|
replaceExtension :: RawFilePath -> ByteString -> RawFilePath
|
||||||
replaceExtension path ext = dropExtension path <.> ext
|
replaceExtension path ext = dropExtension path <.> ext
|
||||||
|
|
||||||
|
|
||||||
-- | Drop the final extension from a 'RawFilePath'
|
-- | Drop the final extension from a 'RawFilePath'
|
||||||
--
|
--
|
||||||
-- >>> dropExtension "file.exe"
|
-- >>> dropExtension "file.exe"
|
||||||
@@ -180,6 +233,7 @@ replaceExtension path ext = dropExtension path <.> ext
|
|||||||
dropExtension :: RawFilePath -> RawFilePath
|
dropExtension :: RawFilePath -> RawFilePath
|
||||||
dropExtension = fst . splitExtension
|
dropExtension = fst . splitExtension
|
||||||
|
|
||||||
|
|
||||||
-- | Add an extension to a 'RawFilePath'
|
-- | Add an extension to a 'RawFilePath'
|
||||||
--
|
--
|
||||||
-- >>> addExtension "file" ".exe"
|
-- >>> addExtension "file" ".exe"
|
||||||
@@ -195,10 +249,6 @@ addExtension file ext
|
|||||||
| otherwise = BS.intercalate (BS.singleton extSeparator) [file, ext]
|
| otherwise = BS.intercalate (BS.singleton extSeparator) [file, ext]
|
||||||
|
|
||||||
|
|
||||||
-- | Operator version of 'addExtension'
|
|
||||||
(<.>) :: RawFilePath -> ByteString -> RawFilePath
|
|
||||||
(<.>) = addExtension
|
|
||||||
|
|
||||||
-- | Check if a 'RawFilePath' has an extension
|
-- | Check if a 'RawFilePath' has an extension
|
||||||
--
|
--
|
||||||
-- >>> hasExtension "file"
|
-- >>> hasExtension "file"
|
||||||
@@ -210,7 +260,13 @@ addExtension file ext
|
|||||||
hasExtension :: RawFilePath -> Bool
|
hasExtension :: RawFilePath -> Bool
|
||||||
hasExtension = isJust . BS.elemIndex extSeparator . takeFileName
|
hasExtension = isJust . BS.elemIndex extSeparator . takeFileName
|
||||||
|
|
||||||
-- | Split a 'RawFilePath' on the first extension
|
|
||||||
|
-- | Operator version of 'addExtension'
|
||||||
|
(<.>) :: RawFilePath -> ByteString -> RawFilePath
|
||||||
|
(<.>) = addExtension
|
||||||
|
|
||||||
|
|
||||||
|
-- | Split a 'RawFilePath' on the first extension.
|
||||||
--
|
--
|
||||||
-- >>> splitExtensions "/path/file.tar.gz"
|
-- >>> splitExtensions "/path/file.tar.gz"
|
||||||
-- ("/path/file",".tar.gz")
|
-- ("/path/file",".tar.gz")
|
||||||
@@ -224,6 +280,7 @@ splitExtensions x = if BS.null basename
|
|||||||
(path,file) = splitFileNameRaw x
|
(path,file) = splitFileNameRaw x
|
||||||
(basename,fileExt) = BS.break isExtSeparator file
|
(basename,fileExt) = BS.break isExtSeparator file
|
||||||
|
|
||||||
|
|
||||||
-- | Remove all extensions from a 'RawFilePath'
|
-- | Remove all extensions from a 'RawFilePath'
|
||||||
--
|
--
|
||||||
-- >>> dropExtensions "/path/file.tar.gz"
|
-- >>> dropExtensions "/path/file.tar.gz"
|
||||||
@@ -231,6 +288,7 @@ splitExtensions x = if BS.null basename
|
|||||||
dropExtensions :: RawFilePath -> RawFilePath
|
dropExtensions :: RawFilePath -> RawFilePath
|
||||||
dropExtensions = fst . splitExtensions
|
dropExtensions = fst . splitExtensions
|
||||||
|
|
||||||
|
|
||||||
-- | Take all extensions from a 'RawFilePath'
|
-- | Take all extensions from a 'RawFilePath'
|
||||||
--
|
--
|
||||||
-- >>> takeExtensions "/path/file.tar.gz"
|
-- >>> takeExtensions "/path/file.tar.gz"
|
||||||
@@ -238,8 +296,48 @@ dropExtensions = fst . splitExtensions
|
|||||||
takeExtensions :: RawFilePath -> ByteString
|
takeExtensions :: RawFilePath -> ByteString
|
||||||
takeExtensions = snd . splitExtensions
|
takeExtensions = snd . splitExtensions
|
||||||
|
|
||||||
|
|
||||||
|
-- | Drop the given extension from a FilePath, and the @\".\"@ preceding it.
|
||||||
|
-- Returns 'Nothing' if the FilePath does not have the given extension, or
|
||||||
|
-- 'Just' and the part before the extension if it does.
|
||||||
|
--
|
||||||
|
-- This function can be more predictable than 'dropExtensions',
|
||||||
|
-- especially if the filename might itself contain @.@ characters.
|
||||||
|
--
|
||||||
|
-- >>> stripExtension "hs.o" "foo.x.hs.o"
|
||||||
|
-- Just "foo.x"
|
||||||
|
-- >>> stripExtension "hi.o" "foo.x.hs.o"
|
||||||
|
-- Nothing
|
||||||
|
-- >>> stripExtension ".c.d" "a.b.c.d"
|
||||||
|
-- Just "a.b"
|
||||||
|
-- >>> stripExtension ".c.d" "a.b..c.d"
|
||||||
|
-- Just "a.b."
|
||||||
|
-- >>> stripExtension "baz" "foo.bar"
|
||||||
|
-- Nothing
|
||||||
|
-- >>> stripExtension "bar" "foobar"
|
||||||
|
-- Nothing
|
||||||
|
--
|
||||||
|
-- prop> \path -> stripExtension "" path == Just path
|
||||||
|
-- prop> \path -> dropExtension path == fromJust (stripExtension (takeExtension path) path)
|
||||||
|
-- prop> \path -> dropExtensions path == fromJust (stripExtension (takeExtensions path) path)
|
||||||
|
stripExtension :: ByteString -> RawFilePath -> Maybe RawFilePath
|
||||||
|
stripExtension bs path
|
||||||
|
| BS.null bs = Just path
|
||||||
|
| otherwise = stripSuffix' dotExt path
|
||||||
|
where
|
||||||
|
dotExt = if isExtSeparator $ BS.head bs
|
||||||
|
then bs
|
||||||
|
else extSeparator `BS.cons` bs
|
||||||
|
#if MIN_VERSION_bytestring(0,10,8)
|
||||||
|
stripSuffix' = BS.stripSuffix
|
||||||
|
#else
|
||||||
|
stripSuffix' xs ys = fmap (BS.pack . reverse) $ L.stripPrefix (reverse $ BS.unpack xs) (reverse $ BS.unpack ys)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
------------------------
|
------------------------
|
||||||
-- more stuff
|
-- Filename/directory functions
|
||||||
|
|
||||||
|
|
||||||
-- | Split a 'RawFilePath' into (path,file). 'combine' is the inverse
|
-- | Split a 'RawFilePath' into (path,file). 'combine' is the inverse
|
||||||
--
|
--
|
||||||
@@ -269,12 +367,14 @@ splitFileName x = if BS.null path
|
|||||||
takeFileName :: RawFilePath -> RawFilePath
|
takeFileName :: RawFilePath -> RawFilePath
|
||||||
takeFileName = snd . splitFileName
|
takeFileName = snd . splitFileName
|
||||||
|
|
||||||
|
|
||||||
-- | Change the file name
|
-- | Change the file name
|
||||||
--
|
--
|
||||||
-- prop> \path -> replaceFileName path (takeFileName path) == path
|
-- prop> \path -> replaceFileName path (takeFileName path) == path
|
||||||
replaceFileName :: RawFilePath -> ByteString -> RawFilePath
|
replaceFileName :: RawFilePath -> ByteString -> RawFilePath
|
||||||
replaceFileName x y = fst (splitFileNameRaw x) </> y
|
replaceFileName x y = fst (splitFileNameRaw x) </> y
|
||||||
|
|
||||||
|
|
||||||
-- | Drop the file name
|
-- | Drop the file name
|
||||||
--
|
--
|
||||||
-- >>> dropFileName "path/file.txt"
|
-- >>> dropFileName "path/file.txt"
|
||||||
@@ -284,6 +384,7 @@ replaceFileName x y = fst (splitFileNameRaw x) </> y
|
|||||||
dropFileName :: RawFilePath -> RawFilePath
|
dropFileName :: RawFilePath -> RawFilePath
|
||||||
dropFileName = fst . splitFileName
|
dropFileName = fst . splitFileName
|
||||||
|
|
||||||
|
|
||||||
-- | Get the file name, without a trailing extension
|
-- | Get the file name, without a trailing extension
|
||||||
--
|
--
|
||||||
-- >>> takeBaseName "path/file.tar.gz"
|
-- >>> takeBaseName "path/file.tar.gz"
|
||||||
@@ -293,6 +394,7 @@ dropFileName = fst . splitFileName
|
|||||||
takeBaseName :: RawFilePath -> ByteString
|
takeBaseName :: RawFilePath -> ByteString
|
||||||
takeBaseName = dropExtension . takeFileName
|
takeBaseName = dropExtension . takeFileName
|
||||||
|
|
||||||
|
|
||||||
-- | Change the base name
|
-- | Change the base name
|
||||||
--
|
--
|
||||||
-- >>> replaceBaseName "path/file.tar.gz" "bob"
|
-- >>> replaceBaseName "path/file.tar.gz" "bob"
|
||||||
@@ -305,6 +407,7 @@ replaceBaseName path name = combineRaw dir (name <.> ext)
|
|||||||
(dir,file) = splitFileNameRaw path
|
(dir,file) = splitFileNameRaw path
|
||||||
ext = takeExtension file
|
ext = takeExtension file
|
||||||
|
|
||||||
|
|
||||||
-- | Get the directory, moving up one level if it's already a directory
|
-- | Get the directory, moving up one level if it's already a directory
|
||||||
--
|
--
|
||||||
-- >>> takeDirectory "path/file.txt"
|
-- >>> takeDirectory "path/file.txt"
|
||||||
@@ -324,12 +427,14 @@ takeDirectory x = case () of
|
|||||||
res = fst $ BS.spanEnd isPathSeparator file
|
res = fst $ BS.spanEnd isPathSeparator file
|
||||||
file = dropFileName x
|
file = dropFileName x
|
||||||
|
|
||||||
|
|
||||||
-- | Change the directory component of a 'RawFilePath'
|
-- | Change the directory component of a 'RawFilePath'
|
||||||
--
|
--
|
||||||
-- prop> \path -> replaceDirectory path (takeDirectory path) `equalFilePath` path || takeDirectory path == "."
|
-- prop> \path -> replaceDirectory path (takeDirectory path) `equalFilePath` path || takeDirectory path == "."
|
||||||
replaceDirectory :: RawFilePath -> ByteString -> RawFilePath
|
replaceDirectory :: RawFilePath -> ByteString -> RawFilePath
|
||||||
replaceDirectory file dir = combineRaw dir (takeFileName file)
|
replaceDirectory file dir = combineRaw dir (takeFileName file)
|
||||||
|
|
||||||
|
|
||||||
-- | Join two paths together
|
-- | Join two paths together
|
||||||
--
|
--
|
||||||
-- >>> combine "/" "file"
|
-- >>> combine "/" "file"
|
||||||
@@ -342,6 +447,7 @@ combine :: RawFilePath -> RawFilePath -> RawFilePath
|
|||||||
combine a b | not (BS.null b) && isPathSeparator (BS.head b) = b
|
combine a b | not (BS.null b) && isPathSeparator (BS.head b) = b
|
||||||
| otherwise = combineRaw a b
|
| otherwise = combineRaw a b
|
||||||
|
|
||||||
|
|
||||||
-- | Operator version of combine
|
-- | Operator version of combine
|
||||||
(</>) :: RawFilePath -> RawFilePath -> RawFilePath
|
(</>) :: RawFilePath -> RawFilePath -> RawFilePath
|
||||||
(</>) = combine
|
(</>) = combine
|
||||||
@@ -363,6 +469,17 @@ splitPath = splitter
|
|||||||
Nothing -> [x]
|
Nothing -> [x]
|
||||||
Just runlen -> uncurry (:) . second splitter $ BS.splitAt (ix+1+runlen) x
|
Just runlen -> uncurry (:) . second splitter $ BS.splitAt (ix+1+runlen) x
|
||||||
|
|
||||||
|
|
||||||
|
-- | Join a split path back together
|
||||||
|
--
|
||||||
|
-- prop> \path -> joinPath (splitPath path) == path
|
||||||
|
--
|
||||||
|
-- >>> joinPath ["path","to","file.txt"]
|
||||||
|
-- "path/to/file.txt"
|
||||||
|
joinPath :: [RawFilePath] -> RawFilePath
|
||||||
|
joinPath = foldr (</>) BS.empty
|
||||||
|
|
||||||
|
|
||||||
-- | Like 'splitPath', but without trailing slashes
|
-- | Like 'splitPath', but without trailing slashes
|
||||||
--
|
--
|
||||||
-- >>> splitDirectories "/path/to/file.txt"
|
-- >>> splitDirectories "/path/to/file.txt"
|
||||||
@@ -378,14 +495,60 @@ splitDirectories x
|
|||||||
where
|
where
|
||||||
splitter = filter (not . BS.null) . BS.split pathSeparator
|
splitter = filter (not . BS.null) . BS.split pathSeparator
|
||||||
|
|
||||||
-- | Join a split path back together
|
|
||||||
|
|
||||||
|
------------------------
|
||||||
|
-- Trailing slash functions
|
||||||
|
|
||||||
|
-- | Check if the last character of a 'RawFilePath' is '/'.
|
||||||
--
|
--
|
||||||
-- prop> \path -> joinPath (splitPath path) == path
|
-- >>> hasTrailingPathSeparator "/path/"
|
||||||
|
-- True
|
||||||
|
-- >>> hasTrailingPathSeparator "/"
|
||||||
|
-- True
|
||||||
|
-- >>> hasTrailingPathSeparator "/path"
|
||||||
|
-- False
|
||||||
|
hasTrailingPathSeparator :: RawFilePath -> Bool
|
||||||
|
hasTrailingPathSeparator x
|
||||||
|
| BS.null x = False
|
||||||
|
| otherwise = isPathSeparator $ BS.last x
|
||||||
|
|
||||||
|
|
||||||
|
-- | Add a trailing path separator.
|
||||||
--
|
--
|
||||||
-- >>> joinPath ["path","to","file.txt"]
|
-- >>> addTrailingPathSeparator "/path"
|
||||||
-- "path/to/file.txt"
|
-- "/path/"
|
||||||
joinPath :: [RawFilePath] -> RawFilePath
|
-- >>> addTrailingPathSeparator "/path/"
|
||||||
joinPath = foldr (</>) BS.empty
|
-- "/path/"
|
||||||
|
-- >>> addTrailingPathSeparator "/"
|
||||||
|
-- "/"
|
||||||
|
addTrailingPathSeparator :: RawFilePath -> RawFilePath
|
||||||
|
addTrailingPathSeparator x = if hasTrailingPathSeparator x
|
||||||
|
then x
|
||||||
|
else x `BS.snoc` pathSeparator
|
||||||
|
|
||||||
|
|
||||||
|
-- | Remove a trailing path separator
|
||||||
|
--
|
||||||
|
-- >>> dropTrailingPathSeparator "/path/"
|
||||||
|
-- "/path"
|
||||||
|
-- >>> dropTrailingPathSeparator "/path////"
|
||||||
|
-- "/path"
|
||||||
|
-- >>> dropTrailingPathSeparator "/"
|
||||||
|
-- "/"
|
||||||
|
-- >>> dropTrailingPathSeparator "//"
|
||||||
|
-- "/"
|
||||||
|
dropTrailingPathSeparator :: RawFilePath -> RawFilePath
|
||||||
|
dropTrailingPathSeparator x
|
||||||
|
| x == BS.singleton pathSeparator = x
|
||||||
|
| otherwise = if hasTrailingPathSeparator x
|
||||||
|
then dropTrailingPathSeparator $ BS.init x
|
||||||
|
else x
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
------------------------
|
||||||
|
-- File name manipulations
|
||||||
|
|
||||||
|
|
||||||
-- |Normalise a file.
|
-- |Normalise a file.
|
||||||
@@ -442,139 +605,47 @@ normalise filepath =
|
|||||||
dropDots = filter (BS.singleton _period /=)
|
dropDots = filter (BS.singleton _period /=)
|
||||||
|
|
||||||
|
|
||||||
------------------------
|
|
||||||
-- trailing path separators
|
|
||||||
|
|
||||||
-- | Check if the last character of a 'RawFilePath' is '/'.
|
-- | Contract a filename, based on a relative path. Note that the resulting
|
||||||
|
-- path will never introduce @..@ paths, as the presence of symlinks
|
||||||
|
-- means @..\/b@ may not reach @a\/b@ if it starts from @a\/c@. For a
|
||||||
|
-- worked example see
|
||||||
|
-- <http://neilmitchell.blogspot.co.uk/2015/10/filepaths-are-subtle-symlinks-are-hard.html this blog post>.
|
||||||
--
|
--
|
||||||
-- >>> hasTrailingPathSeparator "/path/"
|
-- >>> makeRelative "/directory" "/directory/file.ext"
|
||||||
-- True
|
-- "file.ext"
|
||||||
-- >>> hasTrailingPathSeparator "/"
|
-- >>> makeRelative "/Home" "/home/bob"
|
||||||
-- True
|
-- "/home/bob"
|
||||||
-- >>> hasTrailingPathSeparator "/path"
|
-- >>> makeRelative "/home/" "/home/bob/foo/bar"
|
||||||
-- False
|
-- "bob/foo/bar"
|
||||||
hasTrailingPathSeparator :: RawFilePath -> Bool
|
-- >>> makeRelative "/fred" "bob"
|
||||||
hasTrailingPathSeparator x
|
-- "bob"
|
||||||
| BS.null x = False
|
-- >>> makeRelative "/file/test" "/file/test/fred"
|
||||||
| otherwise = isPathSeparator $ BS.last x
|
-- "fred"
|
||||||
|
-- >>> makeRelative "/file/test" "/file/test/fred/"
|
||||||
-- | Add a trailing path separator.
|
-- "fred/"
|
||||||
|
-- >>> makeRelative "some/path" "some/path/a/b/c"
|
||||||
|
-- "a/b/c"
|
||||||
--
|
--
|
||||||
-- >>> addTrailingPathSeparator "/path"
|
-- prop> \p -> makeRelative p p == "."
|
||||||
-- "/path/"
|
-- prop> \p -> makeRelative (takeDirectory p) p `equalFilePath` takeFileName p
|
||||||
-- >>> addTrailingPathSeparator "/path/"
|
-- prop \x y -> equalFilePath x y || (isRelative x && makeRelative y x == x) || equalFilePath (y </> makeRelative y x) x
|
||||||
-- "/path/"
|
makeRelative :: RawFilePath -> RawFilePath -> RawFilePath
|
||||||
-- >>> addTrailingPathSeparator "/"
|
makeRelative root path
|
||||||
-- "/"
|
| equalFilePath root path = BS.singleton _period
|
||||||
addTrailingPathSeparator :: RawFilePath -> RawFilePath
|
| takeAbs root /= takeAbs path = path
|
||||||
addTrailingPathSeparator x = if hasTrailingPathSeparator x
|
| otherwise = f (dropAbs root) (dropAbs path)
|
||||||
then x
|
|
||||||
else x `BS.snoc` pathSeparator
|
|
||||||
|
|
||||||
-- | Remove a trailing path separator
|
|
||||||
--
|
|
||||||
-- >>> dropTrailingPathSeparator "/path/"
|
|
||||||
-- "/path"
|
|
||||||
-- >>> dropTrailingPathSeparator "/path////"
|
|
||||||
-- "/path"
|
|
||||||
-- >>> dropTrailingPathSeparator "/"
|
|
||||||
-- "/"
|
|
||||||
-- >>> dropTrailingPathSeparator "//"
|
|
||||||
-- "/"
|
|
||||||
dropTrailingPathSeparator :: RawFilePath -> RawFilePath
|
|
||||||
dropTrailingPathSeparator x
|
|
||||||
| x == BS.singleton pathSeparator = x
|
|
||||||
| otherwise = if hasTrailingPathSeparator x
|
|
||||||
then dropTrailingPathSeparator $ BS.init x
|
|
||||||
else x
|
|
||||||
|
|
||||||
------------------------
|
|
||||||
-- Filename/system stuff
|
|
||||||
|
|
||||||
-- | Check if a path is absolute
|
|
||||||
--
|
|
||||||
-- >>> isAbsolute "/path"
|
|
||||||
-- True
|
|
||||||
-- >>> isAbsolute "path"
|
|
||||||
-- False
|
|
||||||
-- >>> isAbsolute ""
|
|
||||||
-- False
|
|
||||||
isAbsolute :: RawFilePath -> Bool
|
|
||||||
isAbsolute x
|
|
||||||
| BS.length x > 0 = isPathSeparator (BS.head x)
|
|
||||||
| otherwise = False
|
|
||||||
|
|
||||||
-- | Check if a path is relative
|
|
||||||
--
|
|
||||||
-- prop> \path -> isRelative path /= isAbsolute path
|
|
||||||
isRelative :: RawFilePath -> Bool
|
|
||||||
isRelative = not . isAbsolute
|
|
||||||
|
|
||||||
-- | Is a FilePath valid, i.e. could you create a file like it?
|
|
||||||
--
|
|
||||||
-- >>> isValid ""
|
|
||||||
-- False
|
|
||||||
-- >>> isValid "\0"
|
|
||||||
-- False
|
|
||||||
-- >>> isValid "/random_ path:*"
|
|
||||||
-- True
|
|
||||||
isValid :: RawFilePath -> Bool
|
|
||||||
isValid filepath
|
|
||||||
| BS.null filepath = False
|
|
||||||
| _nul `BS.elem` filepath = False
|
|
||||||
| otherwise = True
|
|
||||||
|
|
||||||
-- | Is the given filename a valid filename?
|
|
||||||
--
|
|
||||||
-- >>> isFileName "lal"
|
|
||||||
-- True
|
|
||||||
-- >>> isFileName "."
|
|
||||||
-- True
|
|
||||||
-- >>> isFileName ".."
|
|
||||||
-- True
|
|
||||||
-- >>> isFileName ""
|
|
||||||
-- False
|
|
||||||
-- >>> isFileName "\0"
|
|
||||||
-- False
|
|
||||||
-- >>> isFileName "/random_ path:*"
|
|
||||||
-- False
|
|
||||||
isFileName :: ByteString -> Bool
|
|
||||||
isFileName filepath =
|
|
||||||
not (BS.singleton pathSeparator `BS.isInfixOf` filepath) &&
|
|
||||||
not (BS.null filepath) &&
|
|
||||||
not (_nul `BS.elem` filepath)
|
|
||||||
|
|
||||||
-- | Helper function: check if the filepath has any parent directories in it.
|
|
||||||
--
|
|
||||||
-- >>> hasParentDir "/.."
|
|
||||||
-- True
|
|
||||||
-- >>> hasParentDir "foo/bar/.."
|
|
||||||
-- True
|
|
||||||
-- >>> hasParentDir "foo/../bar/."
|
|
||||||
-- True
|
|
||||||
-- >>> hasParentDir "foo/bar"
|
|
||||||
-- False
|
|
||||||
-- >>> hasParentDir "foo"
|
|
||||||
-- False
|
|
||||||
-- >>> hasParentDir ""
|
|
||||||
-- False
|
|
||||||
-- >>> hasParentDir ".."
|
|
||||||
-- False
|
|
||||||
hasParentDir :: ByteString -> Bool
|
|
||||||
hasParentDir filepath =
|
|
||||||
((pathSeparator `BS.cons` pathDoubleDot)
|
|
||||||
`BS.isSuffixOf` filepath
|
|
||||||
) ||
|
|
||||||
((BS.singleton pathSeparator
|
|
||||||
`BS.append` pathDoubleDot
|
|
||||||
`BS.append` BS.singleton pathSeparator
|
|
||||||
) `BS.isInfixOf` filepath
|
|
||||||
) ||
|
|
||||||
((pathDoubleDot `BS.append` BS.singleton pathSeparator
|
|
||||||
) `BS.isPrefixOf` filepath
|
|
||||||
)
|
|
||||||
where
|
where
|
||||||
pathDoubleDot = BS.pack [_period, _period]
|
f x y
|
||||||
|
| BS.null x = BS.dropWhile isPathSeparator y
|
||||||
|
| otherwise = let (x1,x2) = g x
|
||||||
|
(y1,y2) = g y
|
||||||
|
in if equalFilePath x1 y1 then f x2 y2 else path
|
||||||
|
g x = (BS.dropWhile isPathSeparator a, BS.dropWhile isPathSeparator b)
|
||||||
|
where (a, b) = BS.break isPathSeparator $ BS.dropWhile isPathSeparator x
|
||||||
|
dropAbs x = snd $ BS.span (== _slash) x
|
||||||
|
takeAbs x = fst $ BS.span (== _slash) x
|
||||||
|
|
||||||
|
|
||||||
-- |Equality of two filepaths. The filepaths are normalised
|
-- |Equality of two filepaths. The filepaths are normalised
|
||||||
-- and trailing path separators are dropped.
|
-- and trailing path separators are dropped.
|
||||||
@@ -585,6 +656,8 @@ hasParentDir filepath =
|
|||||||
-- True
|
-- True
|
||||||
-- >>> equalFilePath "foo" "./foo"
|
-- >>> equalFilePath "foo" "./foo"
|
||||||
-- True
|
-- True
|
||||||
|
-- >>> equalFilePath "" ""
|
||||||
|
-- True
|
||||||
-- >>> equalFilePath "foo" "/foo"
|
-- >>> equalFilePath "foo" "/foo"
|
||||||
-- False
|
-- False
|
||||||
-- >>> equalFilePath "foo" "FOO"
|
-- >>> equalFilePath "foo" "FOO"
|
||||||
@@ -599,37 +672,138 @@ equalFilePath p1 p2 = f p1 == f p2
|
|||||||
f x = dropTrailingPathSeparator $ normalise x
|
f x = dropTrailingPathSeparator $ normalise x
|
||||||
|
|
||||||
|
|
||||||
|
-- | Check if a path is relative
|
||||||
|
--
|
||||||
|
-- prop> \path -> isRelative path /= isAbsolute path
|
||||||
|
isRelative :: RawFilePath -> Bool
|
||||||
|
isRelative = not . isAbsolute
|
||||||
|
|
||||||
|
|
||||||
|
-- | Check if a path is absolute
|
||||||
|
--
|
||||||
|
-- >>> isAbsolute "/path"
|
||||||
|
-- True
|
||||||
|
-- >>> isAbsolute "path"
|
||||||
|
-- False
|
||||||
|
-- >>> isAbsolute ""
|
||||||
|
-- False
|
||||||
|
isAbsolute :: RawFilePath -> Bool
|
||||||
|
isAbsolute x
|
||||||
|
| BS.length x > 0 = isPathSeparator (BS.head x)
|
||||||
|
| otherwise = False
|
||||||
|
|
||||||
|
|
||||||
|
-- | Is a FilePath valid, i.e. could you create a file like it?
|
||||||
|
--
|
||||||
|
-- >>> isValid ""
|
||||||
|
-- False
|
||||||
|
-- >>> isValid "\0"
|
||||||
|
-- False
|
||||||
|
-- >>> isValid "/random_ path:*"
|
||||||
|
-- True
|
||||||
|
isValid :: RawFilePath -> Bool
|
||||||
|
isValid filepath
|
||||||
|
| BS.null filepath = False
|
||||||
|
| _nul `BS.elem` filepath = False
|
||||||
|
| otherwise = True
|
||||||
|
|
||||||
|
|
||||||
|
-- | Take a FilePath and make it valid; does not change already valid FilePaths.
|
||||||
|
--
|
||||||
|
-- >>> makeValid ""
|
||||||
|
-- "_"
|
||||||
|
-- >>> makeValid "file\0name"
|
||||||
|
-- "file_name"
|
||||||
|
--
|
||||||
|
-- prop> \p -> if isValid p then makeValid p == p else makeValid p /= p
|
||||||
|
-- prop> \p -> isValid (makeValid p)
|
||||||
|
makeValid :: RawFilePath -> RawFilePath
|
||||||
|
makeValid path
|
||||||
|
| BS.null path = BS.singleton _underscore
|
||||||
|
| otherwise = BS.map (\x -> if x == _nul then _underscore else x) path
|
||||||
|
|
||||||
|
|
||||||
|
-- | Is the given path a valid filename? This includes
|
||||||
|
-- "." and "..".
|
||||||
|
--
|
||||||
|
-- >>> isFileName "lal"
|
||||||
|
-- True
|
||||||
|
-- >>> isFileName "."
|
||||||
|
-- True
|
||||||
|
-- >>> isFileName ".."
|
||||||
|
-- True
|
||||||
|
-- >>> isFileName ""
|
||||||
|
-- False
|
||||||
|
-- >>> isFileName "\0"
|
||||||
|
-- False
|
||||||
|
-- >>> isFileName "/random_ path:*"
|
||||||
|
-- False
|
||||||
|
isFileName :: RawFilePath -> Bool
|
||||||
|
isFileName filepath =
|
||||||
|
not (BS.singleton pathSeparator `BS.isInfixOf` filepath) &&
|
||||||
|
not (BS.null filepath) &&
|
||||||
|
not (_nul `BS.elem` filepath)
|
||||||
|
|
||||||
|
|
||||||
|
-- | Check if the filepath has any parent directories in it.
|
||||||
|
--
|
||||||
|
-- >>> hasParentDir "/.."
|
||||||
|
-- True
|
||||||
|
-- >>> hasParentDir "foo/bar/.."
|
||||||
|
-- True
|
||||||
|
-- >>> hasParentDir "foo/../bar/."
|
||||||
|
-- True
|
||||||
|
-- >>> hasParentDir "foo/bar"
|
||||||
|
-- False
|
||||||
|
-- >>> hasParentDir "foo"
|
||||||
|
-- False
|
||||||
|
-- >>> hasParentDir ""
|
||||||
|
-- False
|
||||||
|
-- >>> hasParentDir ".."
|
||||||
|
-- False
|
||||||
|
hasParentDir :: RawFilePath -> Bool
|
||||||
|
hasParentDir filepath =
|
||||||
|
(pathSeparator `BS.cons` pathDoubleDot)
|
||||||
|
`BS.isSuffixOf` filepath
|
||||||
|
||
|
||||||
|
(BS.singleton pathSeparator
|
||||||
|
`BS.append` pathDoubleDot
|
||||||
|
`BS.append` BS.singleton pathSeparator)
|
||||||
|
`BS.isInfixOf` filepath
|
||||||
|
||
|
||||||
|
(pathDoubleDot `BS.append` BS.singleton pathSeparator)
|
||||||
|
`BS.isPrefixOf` filepath
|
||||||
|
where
|
||||||
|
pathDoubleDot = BS.pack [_period, _period]
|
||||||
|
|
||||||
|
|
||||||
-- | Whether the file is a hidden file.
|
-- | Whether the file is a hidden file.
|
||||||
--
|
--
|
||||||
-- >>> hiddenFile ".foo"
|
-- >>> hiddenFile ".foo"
|
||||||
-- True
|
-- True
|
||||||
-- >>> hiddenFile "..foo.bar"
|
-- >>> hiddenFile "..foo.bar"
|
||||||
-- True
|
-- True
|
||||||
|
-- >>> hiddenFile "some/path/.bar"
|
||||||
|
-- True
|
||||||
-- >>> hiddenFile "..."
|
-- >>> hiddenFile "..."
|
||||||
-- True
|
-- True
|
||||||
-- >>> hiddenFile "dod"
|
|
||||||
-- False
|
|
||||||
-- >>> hiddenFile "dod.bar"
|
-- >>> hiddenFile "dod.bar"
|
||||||
-- False
|
-- False
|
||||||
|
-- >>> hiddenFile "."
|
||||||
|
-- False
|
||||||
|
-- >>> hiddenFile ".."
|
||||||
|
-- False
|
||||||
|
-- >>> hiddenFile ""
|
||||||
|
-- False
|
||||||
hiddenFile :: RawFilePath -> Bool
|
hiddenFile :: RawFilePath -> Bool
|
||||||
hiddenFile fp
|
hiddenFile fp
|
||||||
| fp == BS.pack [_period, _period] = False
|
| fn == BS.pack [_period, _period] = False
|
||||||
| fp == BS.pack [_period] = False
|
| fn == BS.pack [_period] = False
|
||||||
| otherwise = BS.pack [extSeparator]
|
| otherwise = BS.pack [extSeparator]
|
||||||
`BS.isPrefixOf` fp
|
`BS.isPrefixOf` fn
|
||||||
|
where
|
||||||
|
fn = takeFileName fp
|
||||||
|
|
||||||
------------------------
|
|
||||||
-- conversion
|
|
||||||
|
|
||||||
-- |Uses UTF-8 decoding to convert the bytestring into a String.
|
|
||||||
fpToString :: ByteString -> String
|
|
||||||
fpToString = toString
|
|
||||||
|
|
||||||
|
|
||||||
-- |Uses UTF-8 encoding to convert a user provided String into
|
|
||||||
-- a ByteString, which represents a filepath.
|
|
||||||
userStringToFP :: String -> ByteString
|
|
||||||
userStringToFP = fromString
|
|
||||||
|
|
||||||
|
|
||||||
------------------------
|
------------------------
|
||||||
@@ -638,7 +812,7 @@ userStringToFP = fromString
|
|||||||
-- Just split the input FileName without adding/normalizing or changing
|
-- Just split the input FileName without adding/normalizing or changing
|
||||||
-- anything.
|
-- anything.
|
||||||
splitFileNameRaw :: RawFilePath -> (RawFilePath, RawFilePath)
|
splitFileNameRaw :: RawFilePath -> (RawFilePath, RawFilePath)
|
||||||
splitFileNameRaw x = BS.breakEnd isPathSeparator x
|
splitFileNameRaw = BS.breakEnd isPathSeparator
|
||||||
|
|
||||||
-- | Combine two paths, assuming rhs is NOT absolute.
|
-- | Combine two paths, assuming rhs is NOT absolute.
|
||||||
combineRaw :: RawFilePath -> RawFilePath -> RawFilePath
|
combineRaw :: RawFilePath -> RawFilePath -> RawFilePath
|
||||||
|
|||||||
@@ -17,50 +17,59 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
|
||||||
ba = BS.append
|
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/canonicalizePathSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
setupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
setupFiles = do
|
||||||
|
createRegularFile' "file"
|
||||||
|
createDir' "dir"
|
||||||
|
createSymlink' "dirSym" "dir/"
|
||||||
|
createSymlink' "brokenSym" "nothing"
|
||||||
|
createSymlink' "fileSym" "file"
|
||||||
|
|
||||||
|
cleanupFiles :: IO ()
|
||||||
|
cleanupFiles = do
|
||||||
|
deleteFile' "file"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteFile' "dirSym"
|
||||||
|
deleteFile' "brokenSym"
|
||||||
|
deleteFile' "fileSym"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.canonicalizePath" $ do
|
describe "HPath.IO.canonicalizePath" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "canonicalizePath, all fine" $ do
|
it "canonicalizePath, all fine" $ do
|
||||||
path <- withPwd (specDir `ba` "file") return
|
path <- withTmpDir "file" return
|
||||||
canonicalizePath' (specDir `ba` "file")
|
canonicalizePath' "file"
|
||||||
`shouldReturn` path
|
`shouldReturn` path
|
||||||
|
|
||||||
it "canonicalizePath, all fine" $ do
|
it "canonicalizePath, all fine" $ do
|
||||||
path <- withPwd (specDir `ba` "dir") return
|
path <- withTmpDir "dir" return
|
||||||
canonicalizePath' (specDir `ba` "dir")
|
canonicalizePath' "dir"
|
||||||
`shouldReturn` path
|
`shouldReturn` path
|
||||||
|
|
||||||
it "canonicalizePath, all fine" $ do
|
it "canonicalizePath, all fine" $ do
|
||||||
path <- withPwd (specDir `ba` "file") return
|
path <- withTmpDir "file" return
|
||||||
canonicalizePath' (specDir `ba` "fileSym")
|
canonicalizePath' "fileSym"
|
||||||
`shouldReturn` path
|
`shouldReturn` path
|
||||||
|
|
||||||
it "canonicalizePath, all fine" $ do
|
it "canonicalizePath, all fine" $ do
|
||||||
path <- withPwd (specDir `ba` "dir") return
|
path <- withTmpDir "dir" return
|
||||||
canonicalizePath' (specDir `ba` "dirSym")
|
canonicalizePath' "dirSym"
|
||||||
`shouldReturn` path
|
`shouldReturn` path
|
||||||
|
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "canonicalizePath, broken symlink" $
|
it "canonicalizePath, broken symlink" $
|
||||||
canonicalizePath' (specDir `ba` "brokenSym")
|
canonicalizePath' "brokenSym"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "canonicalizePath, file does not exist" $
|
it "canonicalizePath, file does not exist" $
|
||||||
canonicalizePath' (specDir `ba` "nothingBlah")
|
canonicalizePath' "nothingBlah"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
|
||||||
module HPath.IO.CopyDirRecursiveOverwriteSpec where
|
module HPath.IO.CopyDirRecursiveOverwriteSpec where
|
||||||
|
|
||||||
|
|
||||||
@@ -20,91 +21,149 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
|
||||||
ba = BS.append
|
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
setupFiles :: IO ()
|
||||||
specDir = "test/HPath/IO/copyDirRecursiveOverwriteSpec/"
|
setupFiles = do
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createRegularFile' "wrongInput"
|
||||||
|
createSymlink' "wrongInputSymL" "inputDir/"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerm"
|
||||||
|
|
||||||
|
createDir' "inputDir"
|
||||||
|
createDir' "inputDir/bar"
|
||||||
|
createDir' "inputDir/foo"
|
||||||
|
createRegularFile' "inputDir/foo/inputFile1"
|
||||||
|
createRegularFile' "inputDir/inputFile2"
|
||||||
|
createRegularFile' "inputDir/bar/inputFile3"
|
||||||
|
writeFile' "inputDir/foo/inputFile1" "SDAADSdsada"
|
||||||
|
writeFile' "inputDir/inputFile2" "Blahfaselgagaga"
|
||||||
|
writeFile' "inputDir/bar/inputFile3"
|
||||||
|
"fdfdssdffsd3223sasdasdasdadasasddasdasdasasd4"
|
||||||
|
|
||||||
|
createDir' "alreadyExistsD"
|
||||||
|
createDir' "alreadyExistsD/bar"
|
||||||
|
createDir' "alreadyExistsD/foo"
|
||||||
|
createRegularFile' "alreadyExistsD/foo/inputFile1"
|
||||||
|
createRegularFile' "alreadyExistsD/inputFile2"
|
||||||
|
createRegularFile' "alreadyExistsD/bar/inputFile3"
|
||||||
|
writeFile' "alreadyExistsD/foo/inputFile1" "DAAsada"
|
||||||
|
writeFile' "alreadyExistsD/inputFile2" "ahfaagaga"
|
||||||
|
writeFile' "alreadyExistsD/bar/inputFile3"
|
||||||
|
"f3223sasdasdaasdasdasasd4"
|
||||||
|
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerm"
|
||||||
|
|
||||||
|
|
||||||
|
cleanupFiles :: IO ()
|
||||||
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerm"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteFile' "wrongInput"
|
||||||
|
deleteFile' "wrongInputSymL"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerm"
|
||||||
|
deleteFile' "inputDir/foo/inputFile1"
|
||||||
|
deleteFile' "inputDir/inputFile2"
|
||||||
|
deleteFile' "inputDir/bar/inputFile3"
|
||||||
|
deleteDir' "inputDir/foo"
|
||||||
|
deleteDir' "inputDir/bar"
|
||||||
|
deleteDir' "inputDir"
|
||||||
|
deleteFile' "alreadyExistsD/foo/inputFile1"
|
||||||
|
deleteFile' "alreadyExistsD/inputFile2"
|
||||||
|
deleteFile' "alreadyExistsD/bar/inputFile3"
|
||||||
|
deleteDir' "alreadyExistsD/foo"
|
||||||
|
deleteDir' "alreadyExistsD/bar"
|
||||||
|
deleteDir' "alreadyExistsD"
|
||||||
|
|
||||||
specDir' :: String
|
|
||||||
specDir' = toString specDir
|
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.copyDirRecursiveOverwrite" $ do
|
describe "HPath.IO.copyDirRecursiveOverwrite" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "copyDirRecursiveOverwrite, all fine" $ do
|
it "copyDirRecursiveOverwrite, all fine" $ do
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "inputDir")
|
copyDirRecursiveOverwrite' "inputDir"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
removeDirIfExists $ specDir `ba` "outputDir"
|
removeDirIfExists "outputDir"
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, all fine and compare" $ do
|
it "copyDirRecursiveOverwrite, all fine and compare" $ do
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "inputDir")
|
copyDirRecursiveOverwrite' "inputDir"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
(system $ "diff -r --no-dereference "
|
(system $ "diff -r --no-dereference "
|
||||||
++ specDir' ++ "inputDir" ++ " "
|
++ toString tmpDir ++ "inputDir" ++ " "
|
||||||
++ specDir' ++ "outputDir")
|
++ toString tmpDir ++ "outputDir")
|
||||||
`shouldReturn` ExitSuccess
|
`shouldReturn` ExitSuccess
|
||||||
removeDirIfExists $ specDir `ba` "outputDir"
|
removeDirIfExists "outputDir"
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, destination dir already exists" $
|
it "copyDirRecursiveOverwrite, destination dir already exists" $ do
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "inputDir")
|
(system $ "diff -r --no-dereference "
|
||||||
(specDir `ba` "alreadyExistsD")
|
++ toString tmpDir ++ "inputDir" ++ " "
|
||||||
|
++ toString tmpDir ++ "alreadyExistsD")
|
||||||
|
`shouldReturn` (ExitFailure 1)
|
||||||
|
copyDirRecursiveOverwrite' "inputDir"
|
||||||
|
"alreadyExistsD"
|
||||||
|
(system $ "diff -r --no-dereference "
|
||||||
|
++ toString tmpDir ++ "inputDir" ++ " "
|
||||||
|
++ toString tmpDir ++ "alreadyExistsD")
|
||||||
|
`shouldReturn` ExitSuccess
|
||||||
|
removeDirIfExists "outputDir"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "copyDirRecursiveOverwrite, source directory does not exist" $
|
it "copyDirRecursiveOverwrite, source directory does not exist" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "doesNotExist")
|
copyDirRecursiveOverwrite' "doesNotExist"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, no write permission on output dir" $
|
it "copyDirRecursiveOverwrite, no write permission on output dir" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "inputDir")
|
copyDirRecursiveOverwrite' "inputDir"
|
||||||
(specDir `ba` "noWritePerm/foo")
|
"noWritePerm/foo"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, cannot open output dir" $
|
it "copyDirRecursiveOverwrite, cannot open output dir" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "inputDir")
|
copyDirRecursiveOverwrite' "inputDir"
|
||||||
(specDir `ba` "noPerms/foo")
|
"noPerms/foo"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, cannot open source dir" $
|
it "copyDirRecursiveOverwrite, cannot open source dir" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "noPerms/inputDir")
|
copyDirRecursiveOverwrite' "noPerms/inputDir"
|
||||||
(specDir `ba` "foo")
|
"foo"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, destination already exists and is a file" $
|
it "copyDirRecursiveOverwrite, destination already exists and is a file" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "inputDir")
|
copyDirRecursiveOverwrite' "inputDir"
|
||||||
(specDir `ba` "alreadyExists")
|
"alreadyExists"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, wrong input (regular file)" $
|
it "copyDirRecursiveOverwrite, wrong input (regular file)" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "wrongInput")
|
copyDirRecursiveOverwrite' "wrongInput"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, wrong input (symlink to directory)" $
|
it "copyDirRecursiveOverwrite, wrong input (symlink to directory)" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "wrongInputSymL")
|
copyDirRecursiveOverwrite' "wrongInputSymL"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InvalidArgument)
|
(\e -> ioeGetErrorType e == InvalidArgument)
|
||||||
|
|
||||||
-- custom failures
|
-- custom failures
|
||||||
it "copyDirRecursiveOverwrite, destination in source" $
|
it "copyDirRecursiveOverwrite, destination in source" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "inputDir")
|
copyDirRecursiveOverwrite' "inputDir"
|
||||||
(specDir `ba` "inputDir/foo")
|
"inputDir/foo"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isDestinationInSource
|
isDestinationInSource
|
||||||
|
|
||||||
it "copyDirRecursiveOverwrite, destination and source same directory" $
|
it "copyDirRecursiveOverwrite, destination and source same directory" $
|
||||||
copyDirRecursiveOverwrite' (specDir `ba` "inputDir")
|
copyDirRecursiveOverwrite' "inputDir"
|
||||||
(specDir `ba` "inputDir")
|
"inputDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isSameFile
|
isSameFile
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
|
||||||
module HPath.IO.CopyDirRecursiveSpec where
|
module HPath.IO.CopyDirRecursiveSpec where
|
||||||
|
|
||||||
|
|
||||||
@@ -20,93 +21,130 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
|
||||||
ba = BS.append
|
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/copyDirRecursiveSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
|
||||||
specDir' = toString specDir
|
setupFiles :: IO ()
|
||||||
|
setupFiles = do
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createRegularFile' "wrongInput"
|
||||||
|
createSymlink' "wrongInputSymL" "inputDir/"
|
||||||
|
createDir' "alreadyExistsD"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerm"
|
||||||
|
|
||||||
|
createDir' "inputDir"
|
||||||
|
createDir' "inputDir/bar"
|
||||||
|
createDir' "inputDir/foo"
|
||||||
|
createRegularFile' "inputDir/foo/inputFile1"
|
||||||
|
createRegularFile' "inputDir/inputFile2"
|
||||||
|
createRegularFile' "inputDir/bar/inputFile3"
|
||||||
|
writeFile' "inputDir/foo/inputFile1" "SDAADSdsada"
|
||||||
|
writeFile' "inputDir/inputFile2" "Blahfaselgagaga"
|
||||||
|
writeFile' "inputDir/bar/inputFile3"
|
||||||
|
"fdfdssdffsd3223sasdasdasdadasasddasdasdasasd4"
|
||||||
|
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerm"
|
||||||
|
|
||||||
|
|
||||||
|
cleanupFiles :: IO ()
|
||||||
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerm"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteFile' "wrongInput"
|
||||||
|
deleteFile' "wrongInputSymL"
|
||||||
|
deleteDir' "alreadyExistsD"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerm"
|
||||||
|
deleteFile' "inputDir/foo/inputFile1"
|
||||||
|
deleteFile' "inputDir/inputFile2"
|
||||||
|
deleteFile' "inputDir/bar/inputFile3"
|
||||||
|
deleteDir' "inputDir/foo"
|
||||||
|
deleteDir' "inputDir/bar"
|
||||||
|
deleteDir' "inputDir"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.copyDirRecursive" $ do
|
describe "HPath.IO.copyDirRecursive" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "copyDirRecursive, all fine" $ do
|
it "copyDirRecursive, all fine" $ do
|
||||||
copyDirRecursive' (specDir `ba` "inputDir")
|
copyDirRecursive' "inputDir"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
removeDirIfExists (specDir `ba` "outputDir")
|
removeDirIfExists "outputDir"
|
||||||
|
|
||||||
it "copyDirRecursive, all fine and compare" $ do
|
it "copyDirRecursive, all fine and compare" $ do
|
||||||
copyDirRecursive' (specDir `ba` "inputDir")
|
copyDirRecursive' "inputDir"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
(system $ "diff -r --no-dereference "
|
(system $ "diff -r --no-dereference "
|
||||||
++ specDir' ++ "inputDir" ++ " "
|
++ toString tmpDir ++ "inputDir" ++ " "
|
||||||
++ specDir' ++ "outputDir")
|
++ toString tmpDir ++ "outputDir")
|
||||||
`shouldReturn` ExitSuccess
|
`shouldReturn` ExitSuccess
|
||||||
removeDirIfExists (specDir `ba` "outputDir")
|
removeDirIfExists "outputDir"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "copyDirRecursive, source directory does not exist" $
|
it "copyDirRecursive, source directory does not exist" $
|
||||||
copyDirRecursive' (specDir `ba` "doesNotExist")
|
copyDirRecursive' "doesNotExist"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "copyDirRecursive, no write permission on output dir" $
|
it "copyDirRecursive, no write permission on output dir" $
|
||||||
copyDirRecursive' (specDir `ba` "inputDir")
|
copyDirRecursive' "inputDir"
|
||||||
(specDir `ba` "noWritePerm/foo")
|
"noWritePerm/foo"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyDirRecursive, cannot open output dir" $
|
it "copyDirRecursive, cannot open output dir" $
|
||||||
copyDirRecursive' (specDir `ba` "inputDir")
|
copyDirRecursive' "inputDir"
|
||||||
(specDir `ba` "noPerms/foo")
|
"noPerms/foo"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyDirRecursive, cannot open source dir" $
|
it "copyDirRecursive, cannot open source dir" $
|
||||||
copyDirRecursive' (specDir `ba` "noPerms/inputDir")
|
copyDirRecursive' "noPerms/inputDir"
|
||||||
(specDir `ba` "foo")
|
"foo"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyDirRecursive, destination dir already exists" $
|
it "copyDirRecursive, destination dir already exists" $
|
||||||
copyDirRecursive' (specDir `ba` "inputDir")
|
copyDirRecursive' "inputDir"
|
||||||
(specDir `ba` "alreadyExistsD")
|
"alreadyExistsD"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == AlreadyExists)
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
it "copyDirRecursive, destination already exists and is a file" $
|
it "copyDirRecursive, destination already exists and is a file" $
|
||||||
copyDirRecursive' (specDir `ba` "inputDir")
|
copyDirRecursive' "inputDir"
|
||||||
(specDir `ba` "alreadyExists")
|
"alreadyExists"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == AlreadyExists)
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
it "copyDirRecursive, wrong input (regular file)" $
|
it "copyDirRecursive, wrong input (regular file)" $
|
||||||
copyDirRecursive' (specDir `ba` "wrongInput")
|
copyDirRecursive' "wrongInput"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "copyDirRecursive, wrong input (symlink to directory)" $
|
it "copyDirRecursive, wrong input (symlink to directory)" $
|
||||||
copyDirRecursive' (specDir `ba` "wrongInputSymL")
|
copyDirRecursive' "wrongInputSymL"
|
||||||
(specDir `ba` "outputDir")
|
"outputDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InvalidArgument)
|
(\e -> ioeGetErrorType e == InvalidArgument)
|
||||||
|
|
||||||
-- custom failures
|
-- custom failures
|
||||||
it "copyDirRecursive, destination in source" $
|
it "copyDirRecursive, destination in source" $
|
||||||
copyDirRecursive' (specDir `ba` "inputDir")
|
copyDirRecursive' "inputDir"
|
||||||
(specDir `ba` "inputDir/foo")
|
"inputDir/foo"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isDestinationInSource
|
isDestinationInSource
|
||||||
|
|
||||||
it "copyDirRecursive, destination and source same directory" $
|
it "copyDirRecursive, destination and source same directory" $
|
||||||
copyDirRecursive' (specDir `ba` "inputDir")
|
copyDirRecursive' "inputDir"
|
||||||
(specDir `ba` "inputDir")
|
"inputDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isSameFile
|
isSameFile
|
||||||
|
|||||||
@@ -20,90 +20,110 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "inputFile"
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createSymlink' "inputFileSymL" "inputFile"
|
||||||
|
createDir' "alreadyExistsD"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createRegularFile' "noPerms/inputFile"
|
||||||
|
createDir' "outputDirNoWrite"
|
||||||
|
createDir' "wrongInput"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "outputDirNoWrite"
|
||||||
|
writeFile' "inputFile" "Blahfaselgagaga"
|
||||||
|
writeFile' "alreadyExists" "dsaldsalkaklsdlkasksdadasl"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/copyFileOverwriteSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "outputDirNoWrite"
|
||||||
|
deleteFile' "noPerms/inputFile"
|
||||||
|
deleteFile' "inputFile"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteFile' "inputFileSymL"
|
||||||
|
deleteDir' "alreadyExistsD"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "outputDirNoWrite"
|
||||||
|
deleteDir' "wrongInput"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.copyFileOverwrite" $ do
|
describe "HPath.IO.copyFileOverwrite" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "copyFileOverwrite, everything clear" $ do
|
it "copyFileOverwrite, everything clear" $ do
|
||||||
copyFileOverwrite' (specDir `ba` "inputFile")
|
copyFileOverwrite' "inputFile"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
removeFileIfExists (specDir `ba` "outputFile")
|
removeFileIfExists "outputFile"
|
||||||
|
|
||||||
it "copyFileOverwrite, output file already exists, all clear" $ do
|
it "copyFileOverwrite, output file already exists, all clear" $ do
|
||||||
copyFile' (specDir `ba` "alreadyExists") (specDir `ba` "alreadyExists.bak")
|
copyFile' "alreadyExists" "alreadyExists.bak"
|
||||||
copyFileOverwrite' (specDir `ba` "inputFile")
|
copyFileOverwrite' "inputFile"
|
||||||
(specDir `ba` "alreadyExists")
|
"alreadyExists"
|
||||||
(system $ "cmp -s " ++ specDir' ++ "inputFile" ++ " "
|
(system $ "cmp -s " ++ toString tmpDir ++ "inputFile" ++ " "
|
||||||
++ specDir' ++ "alreadyExists")
|
++ toString tmpDir ++ "alreadyExists")
|
||||||
`shouldReturn` ExitSuccess
|
`shouldReturn` ExitSuccess
|
||||||
removeFileIfExists (specDir `ba` "alreadyExists")
|
removeFileIfExists "alreadyExists"
|
||||||
copyFile' (specDir `ba` "alreadyExists.bak") (specDir `ba` "alreadyExists")
|
copyFile' "alreadyExists.bak" "alreadyExists"
|
||||||
removeFileIfExists (specDir `ba` "alreadyExists.bak")
|
removeFileIfExists "alreadyExists.bak"
|
||||||
|
|
||||||
it "copyFileOverwrite, and compare" $ do
|
it "copyFileOverwrite, and compare" $ do
|
||||||
copyFileOverwrite' (specDir `ba` "inputFile")
|
copyFileOverwrite' "inputFile"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
(system $ "cmp -s " ++ specDir' ++ "inputFile" ++ " "
|
(system $ "cmp -s " ++ toString tmpDir ++ "inputFile" ++ " "
|
||||||
++ specDir' ++ "outputFile")
|
++ toString tmpDir ++ "outputFile")
|
||||||
`shouldReturn` ExitSuccess
|
`shouldReturn` ExitSuccess
|
||||||
removeFileIfExists (specDir `ba` "outputFile")
|
removeFileIfExists "outputFile"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "copyFileOverwrite, input file does not exist" $
|
it "copyFileOverwrite, input file does not exist" $
|
||||||
copyFileOverwrite' (specDir `ba` "noSuchFile")
|
copyFileOverwrite' "noSuchFile"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "copyFileOverwrite, no permission to write to output directory" $
|
it "copyFileOverwrite, no permission to write to output directory" $
|
||||||
copyFileOverwrite' (specDir `ba` "inputFile")
|
copyFileOverwrite' "inputFile"
|
||||||
(specDir `ba` "outputDirNoWrite/outputFile")
|
"outputDirNoWrite/outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyFileOverwrite, cannot open output directory" $
|
it "copyFileOverwrite, cannot open output directory" $
|
||||||
copyFileOverwrite' (specDir `ba` "inputFile")
|
copyFileOverwrite' "inputFile"
|
||||||
(specDir `ba` "noPerms/outputFile")
|
"noPerms/outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyFileOverwrite, cannot open source directory" $
|
it "copyFileOverwrite, cannot open source directory" $
|
||||||
copyFileOverwrite' (specDir `ba` "noPerms/inputFile")
|
copyFileOverwrite' "noPerms/inputFile"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyFileOverwrite, wrong input type (symlink)" $
|
it "copyFileOverwrite, wrong input type (symlink)" $
|
||||||
copyFileOverwrite' (specDir `ba` "inputFileSymL")
|
copyFileOverwrite' "inputFileSymL"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InvalidArgument)
|
(\e -> ioeGetErrorType e == InvalidArgument)
|
||||||
|
|
||||||
it "copyFileOverwrite, wrong input type (directory)" $
|
it "copyFileOverwrite, wrong input type (directory)" $
|
||||||
copyFileOverwrite' (specDir `ba` "wrongInput")
|
copyFileOverwrite' "wrongInput"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "copyFileOverwrite, output file already exists and is a dir" $
|
it "copyFileOverwrite, output file already exists and is a dir" $
|
||||||
copyFileOverwrite' (specDir `ba` "inputFile")
|
copyFileOverwrite' "inputFile"
|
||||||
(specDir `ba` "alreadyExistsD")
|
"alreadyExistsD"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
-- custom failures --
|
-- custom failures --
|
||||||
it "copyFileOverwrite, output and input are same file" $
|
it "copyFileOverwrite, output and input are same file" $
|
||||||
copyFileOverwrite' (specDir `ba` "inputFile")
|
copyFileOverwrite' "inputFile"
|
||||||
(specDir `ba` "inputFile")
|
"inputFile"
|
||||||
`shouldThrow` isSameFile
|
`shouldThrow` isSameFile
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
|
||||||
module HPath.IO.CopyFileSpec where
|
module HPath.IO.CopyFileSpec where
|
||||||
|
|
||||||
|
|
||||||
@@ -20,86 +21,105 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "inputFile"
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createSymlink' "inputFileSymL" "inputFile"
|
||||||
|
createDir' "alreadyExistsD"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createRegularFile' "noPerms/inputFile"
|
||||||
|
createDir' "outputDirNoWrite"
|
||||||
|
createDir' "wrongInput"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "outputDirNoWrite"
|
||||||
|
writeFile' "inputFile" "Blahfaselgagaga"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/copyFileSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "outputDirNoWrite"
|
||||||
|
deleteFile' "noPerms/inputFile"
|
||||||
|
deleteFile' "inputFile"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteFile' "inputFileSymL"
|
||||||
|
deleteDir' "alreadyExistsD"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "outputDirNoWrite"
|
||||||
|
deleteDir' "wrongInput"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.copyFile" $ do
|
describe "HPath.IO.copyFile" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "copyFile, everything clear" $ do
|
it "copyFile, everything clear" $ do
|
||||||
copyFile' (specDir `ba` "inputFile")
|
copyFile' "inputFile"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
removeFileIfExists (specDir `ba` "outputFile")
|
removeFileIfExists "outputFile"
|
||||||
|
|
||||||
it "copyFile, and compare" $ do
|
it "copyFile, and compare" $ do
|
||||||
copyFile' (specDir `ba` "inputFile")
|
copyFile' "inputFile"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
(system $ "cmp -s " ++ specDir' ++ "inputFile" ++ " "
|
(system $ "cmp -s " ++ toString tmpDir ++ "inputFile" ++ " "
|
||||||
++ specDir' ++ "outputFile")
|
++ toString tmpDir ++ "outputFile")
|
||||||
`shouldReturn` ExitSuccess
|
`shouldReturn` ExitSuccess
|
||||||
removeFileIfExists (specDir `ba` "outputFile")
|
removeFileIfExists "outputFile"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "copyFile, input file does not exist" $
|
it "copyFile, input file does not exist" $
|
||||||
copyFile' (specDir `ba` "noSuchFile")
|
copyFile' "noSuchFile"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "copyFile, no permission to write to output directory" $
|
it "copyFile, no permission to write to output directory" $
|
||||||
copyFile' (specDir `ba` "inputFile")
|
copyFile' "inputFile"
|
||||||
(specDir `ba` "outputDirNoWrite/outputFile")
|
"outputDirNoWrite/outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyFile, cannot open output directory" $
|
it "copyFile, cannot open output directory" $
|
||||||
copyFile' (specDir `ba` "inputFile")
|
copyFile' "inputFile"
|
||||||
(specDir `ba` "noPerms/outputFile")
|
"noPerms/outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyFile, cannot open source directory" $
|
it "copyFile, cannot open source directory" $
|
||||||
copyFile' (specDir `ba` "noPerms/inputFile")
|
copyFile' "noPerms/inputFile"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "copyFile, wrong input type (symlink)" $
|
it "copyFile, wrong input type (symlink)" $
|
||||||
copyFile' (specDir `ba` "inputFileSymL")
|
copyFile' "inputFileSymL"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InvalidArgument)
|
(\e -> ioeGetErrorType e == InvalidArgument)
|
||||||
|
|
||||||
it "copyFile, wrong input type (directory)" $
|
it "copyFile, wrong input type (directory)" $
|
||||||
copyFile' (specDir `ba` "wrongInput")
|
copyFile' "wrongInput"
|
||||||
(specDir `ba` "outputFile")
|
"outputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "copyFile, output file already exists" $
|
it "copyFile, output file already exists" $
|
||||||
copyFile' (specDir `ba` "inputFile")
|
copyFile' "inputFile"
|
||||||
(specDir `ba` "alreadyExists")
|
"alreadyExists"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == AlreadyExists)
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
it "copyFile, output file already exists and is a dir" $
|
it "copyFile, output file already exists and is a dir" $
|
||||||
copyFile' (specDir `ba` "inputFile")
|
copyFile' "inputFile"
|
||||||
(specDir `ba` "alreadyExistsD")
|
"alreadyExistsD"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == AlreadyExists)
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
-- custom failures --
|
-- custom failures --
|
||||||
it "copyFile, output and input are same file" $
|
it "copyFile, output and input are same file" $
|
||||||
copyFile' (specDir `ba` "inputFile")
|
copyFile' "inputFile"
|
||||||
(specDir `ba` "inputFile")
|
"inputFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isSameFile
|
isSameFile
|
||||||
|
|||||||
@@ -17,38 +17,47 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createDir' "alreadyExists"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerms"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerms"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/createDirSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
|
||||||
specDir' = toString specDir
|
cleanupFiles :: IO ()
|
||||||
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerms"
|
||||||
|
deleteDir' "alreadyExists"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerms"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.createDir" $ do
|
describe "HPath.IO.createDir" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "createDir, all fine" $ do
|
it "createDir, all fine" $ do
|
||||||
createDir' (specDir `ba` "newDir")
|
createDir' "newDir"
|
||||||
removeDirIfExists (specDir `ba` "newDir")
|
removeDirIfExists "newDir"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "createDir, can't write to output directory" $
|
it "createDir, can't write to output directory" $
|
||||||
createDir' (specDir `ba` "noWritePerms/newDir")
|
createDir' "noWritePerms/newDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "createDir, can't open output directory" $
|
it "createDir, can't open output directory" $
|
||||||
createDir' (specDir `ba` "noPerms/newDir")
|
createDir' "noPerms/newDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "createDir, destination directory already exists" $
|
it "createDir, destination directory already exists" $
|
||||||
createDir' (specDir `ba` "alreadyExists")
|
createDir' "alreadyExists"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == AlreadyExists)
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
|
|||||||
@@ -17,38 +17,47 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerms"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerms"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/createRegularFileSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
|
||||||
specDir' = toString specDir
|
cleanupFiles :: IO ()
|
||||||
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerms"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerms"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.createRegularFile" $ do
|
describe "HPath.IO.createRegularFile" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "createRegularFile, all fine" $ do
|
it "createRegularFile, all fine" $ do
|
||||||
createRegularFile' (specDir `ba` "newDir")
|
createRegularFile' "newDir"
|
||||||
removeFileIfExists (specDir `ba` "newDir")
|
removeFileIfExists "newDir"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "createRegularFile, can't write to destination directory" $
|
it "createRegularFile, can't write to destination directory" $
|
||||||
createRegularFile' (specDir `ba` "noWritePerms/newDir")
|
createRegularFile' "noWritePerms/newDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "createRegularFile, can't write to destination directory" $
|
it "createRegularFile, can't write to destination directory" $
|
||||||
createRegularFile' (specDir `ba` "noPerms/newDir")
|
createRegularFile' "noPerms/newDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "createRegularFile, destination file already exists" $
|
it "createRegularFile, destination file already exists" $
|
||||||
createRegularFile' (specDir `ba` "alreadyExists")
|
createRegularFile' "alreadyExists"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == AlreadyExists)
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
|
|||||||
63
test/HPath/IO/CreateSymlinkSpec.hs
Normal file
63
test/HPath/IO/CreateSymlinkSpec.hs
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
module HPath.IO.CreateSymlinkSpec where
|
||||||
|
|
||||||
|
|
||||||
|
import Test.Hspec
|
||||||
|
import HPath.IO.Errors
|
||||||
|
import System.IO.Error
|
||||||
|
(
|
||||||
|
ioeGetErrorType
|
||||||
|
)
|
||||||
|
import GHC.IO.Exception
|
||||||
|
(
|
||||||
|
IOErrorType(..)
|
||||||
|
)
|
||||||
|
import Utils
|
||||||
|
import qualified Data.ByteString as BS
|
||||||
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
|
setupFiles :: IO ()
|
||||||
|
setupFiles = do
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerms"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerms"
|
||||||
|
|
||||||
|
|
||||||
|
cleanupFiles :: IO ()
|
||||||
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerms"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerms"
|
||||||
|
|
||||||
|
|
||||||
|
spec :: Spec
|
||||||
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
|
describe "HPath.IO.createSymlink" $ do
|
||||||
|
|
||||||
|
-- successes --
|
||||||
|
it "createSymlink, all fine" $ do
|
||||||
|
createSymlink' "newSymL" "alreadyExists/"
|
||||||
|
removeFileIfExists "newSymL"
|
||||||
|
|
||||||
|
-- posix failures --
|
||||||
|
it "createSymlink, can't write to destination directory" $
|
||||||
|
createSymlink' "noWritePerms/newDir" "lala"
|
||||||
|
`shouldThrow`
|
||||||
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
|
it "createSymlink, can't write to destination directory" $
|
||||||
|
createSymlink' "noPerms/newDir" "lala"
|
||||||
|
`shouldThrow`
|
||||||
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
|
it "createSymlink, destination file already exists" $
|
||||||
|
createSymlink' "alreadyExists" "lala"
|
||||||
|
`shouldThrow`
|
||||||
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
@@ -21,76 +21,90 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "file"
|
||||||
|
createDir' "dir"
|
||||||
|
createRegularFile' "dir/.keep"
|
||||||
|
createSymlink' "dirSym" "dir/"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createRegularFile' "noPerms/.keep"
|
||||||
|
createDir' "noWritable"
|
||||||
|
createRegularFile' "noWritable/.keep"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/deleteDirRecursiveSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
deleteFile' "file"
|
||||||
|
deleteFile' "dir/.keep"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteFile' "dirSym"
|
||||||
|
deleteFile' "noPerms/.keep"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteFile' "noWritable/.keep"
|
||||||
|
deleteDir' "noWritable"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.deleteDirRecursive" $ do
|
describe "HPath.IO.deleteDirRecursive" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "deleteDirRecursive, empty directory, all fine" $ do
|
it "deleteDirRecursive, empty directory, all fine" $ do
|
||||||
createDir' (specDir `ba` "testDir")
|
createDir' "testDir"
|
||||||
deleteDirRecursive' (specDir `ba` "testDir")
|
deleteDirRecursive' "testDir"
|
||||||
getSymbolicLinkStatus (specDir `ba` "testDir")
|
getSymbolicLinkStatus "testDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "deleteDirRecursive, empty directory with null permissions, all fine" $ do
|
it "deleteDirRecursive, empty directory with null permissions, all fine" $ do
|
||||||
createDir' (specDir `ba` "noPerms/testDir")
|
createDir' "noPerms/testDir"
|
||||||
noPerms (specDir `ba` "noPerms/testDir")
|
noPerms "noPerms/testDir"
|
||||||
deleteDirRecursive' (specDir `ba` "noPerms/testDir")
|
deleteDirRecursive' "noPerms/testDir"
|
||||||
|
|
||||||
it "deleteDirRecursive, non-empty directory, all fine" $ do
|
it "deleteDirRecursive, non-empty directory, all fine" $ do
|
||||||
createDir' (specDir `ba` "nonEmpty")
|
createDir' "nonEmpty"
|
||||||
createDir' (specDir `ba` "nonEmpty/dir1")
|
createDir' "nonEmpty/dir1"
|
||||||
createDir' (specDir `ba` "nonEmpty/dir2")
|
createDir' "nonEmpty/dir2"
|
||||||
createDir' (specDir `ba` "nonEmpty/dir2/dir3")
|
createDir' "nonEmpty/dir2/dir3"
|
||||||
createRegularFile' (specDir `ba` "nonEmpty/file1")
|
createRegularFile' "nonEmpty/file1"
|
||||||
createRegularFile' (specDir `ba` "nonEmpty/dir1/file2")
|
createRegularFile' "nonEmpty/dir1/file2"
|
||||||
deleteDirRecursive' (specDir `ba` "nonEmpty")
|
deleteDirRecursive' "nonEmpty"
|
||||||
getSymbolicLinkStatus (specDir `ba` "nonEmpty")
|
getSymbolicLinkStatus "nonEmpty"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "deleteDirRecursive, can't open parent directory" $ do
|
it "deleteDirRecursive, can't open parent directory" $ do
|
||||||
createDir' (specDir `ba` "noPerms/foo")
|
createDir' "noPerms/foo"
|
||||||
noPerms (specDir `ba` "noPerms")
|
noPerms "noPerms"
|
||||||
(deleteDirRecursive' (specDir `ba` "noPerms/foo")
|
(deleteDirRecursive' "noPerms/foo")
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied))
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
>> normalDirPerms (specDir `ba` "noPerms")
|
normalDirPerms "noPerms"
|
||||||
>> deleteDir' (specDir `ba` "noPerms/foo")
|
deleteDir' "noPerms/foo"
|
||||||
|
|
||||||
it "deleteDirRecursive, can't write to parent directory" $ do
|
it "deleteDirRecursive, can't write to parent directory" $ do
|
||||||
createDir' (specDir `ba` "noWritable/foo")
|
createDir' "noWritable/foo"
|
||||||
noWritableDirPerms (specDir `ba` "noWritable")
|
noWritableDirPerms "noWritable"
|
||||||
(deleteDirRecursive' (specDir `ba` "noWritable/foo")
|
(deleteDirRecursive' "noWritable/foo")
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied))
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
normalDirPerms (specDir `ba` "noWritable")
|
normalDirPerms "noWritable"
|
||||||
deleteDir' (specDir `ba` "noWritable/foo")
|
deleteDir' "noWritable/foo"
|
||||||
|
|
||||||
it "deleteDirRecursive, wrong file type (symlink to directory)" $
|
it "deleteDirRecursive, wrong file type (symlink to directory)" $
|
||||||
deleteDirRecursive' (specDir `ba` "dirSym")
|
deleteDirRecursive' "dirSym"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "deleteDirRecursive, wrong file type (regular file)" $
|
it "deleteDirRecursive, wrong file type (regular file)" $
|
||||||
deleteDirRecursive' (specDir `ba` "file")
|
deleteDirRecursive' "file"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "deleteDirRecursive, directory does not exist" $
|
it "deleteDirRecursive, directory does not exist" $
|
||||||
deleteDirRecursive' (specDir `ba` "doesNotExist")
|
deleteDirRecursive' "doesNotExist"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
|
|||||||
@@ -21,74 +21,88 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "file"
|
||||||
|
createDir' "dir"
|
||||||
|
createRegularFile' "dir/.keep"
|
||||||
|
createSymlink' "dirSym" "dir/"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createRegularFile' "noPerms/.keep"
|
||||||
|
createDir' "noWritable"
|
||||||
|
createRegularFile' "noWritable/.keep"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/deleteDirSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
deleteFile' "file"
|
||||||
|
deleteFile' "dir/.keep"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteFile' "dirSym"
|
||||||
|
deleteFile' "noPerms/.keep"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteFile' "noWritable/.keep"
|
||||||
|
deleteDir' "noWritable"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.deleteDir" $ do
|
describe "HPath.IO.deleteDir" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "deleteDir, empty directory, all fine" $ do
|
it "deleteDir, empty directory, all fine" $ do
|
||||||
createDir' (specDir `ba` "testDir")
|
createDir' "testDir"
|
||||||
deleteDir' (specDir `ba` "testDir")
|
deleteDir' "testDir"
|
||||||
getSymbolicLinkStatus (specDir `ba` "testDir")
|
getSymbolicLinkStatus "testDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "deleteDir, directory with null permissions, all fine" $ do
|
it "deleteDir, directory with null permissions, all fine" $ do
|
||||||
createDir' (specDir `ba` "noPerms/testDir")
|
createDir' "noPerms/testDir"
|
||||||
noPerms (specDir `ba` "noPerms/testDir")
|
noPerms "noPerms/testDir"
|
||||||
deleteDir' (specDir `ba` "noPerms/testDir")
|
deleteDir' "noPerms/testDir"
|
||||||
getSymbolicLinkStatus (specDir `ba` "testDir")
|
getSymbolicLinkStatus "testDir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "deleteDir, wrong file type (symlink to directory)" $
|
it "deleteDir, wrong file type (symlink to directory)" $
|
||||||
deleteDir' (specDir `ba` "dirSym")
|
deleteDir' "dirSym"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "deleteDir, wrong file type (regular file)" $
|
it "deleteDir, wrong file type (regular file)" $
|
||||||
deleteDir' (specDir `ba` "file")
|
deleteDir' "file"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "deleteDir, directory does not exist" $
|
it "deleteDir, directory does not exist" $
|
||||||
deleteDir' (specDir `ba` "doesNotExist")
|
deleteDir' "doesNotExist"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "deleteDir, directory not empty" $
|
it "deleteDir, directory not empty" $
|
||||||
deleteDir' (specDir `ba` "dir")
|
deleteDir' "dir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == UnsatisfiedConstraints)
|
(\e -> ioeGetErrorType e == UnsatisfiedConstraints)
|
||||||
|
|
||||||
it "deleteDir, can't open parent directory" $ do
|
it "deleteDir, can't open parent directory" $ do
|
||||||
createDir' (specDir `ba` "noPerms/foo")
|
createDir' "noPerms/foo"
|
||||||
noPerms (specDir `ba` "noPerms")
|
noPerms "noPerms"
|
||||||
(deleteDir' (specDir `ba` "noPerms/foo")
|
(deleteDir' "noPerms/foo")
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied))
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
>> normalDirPerms (specDir `ba` "noPerms")
|
normalDirPerms "noPerms"
|
||||||
>> deleteDir' (specDir `ba` "noPerms/foo")
|
deleteDir' "noPerms/foo"
|
||||||
|
|
||||||
it "deleteDir, can't write to parent directory, still fine" $ do
|
it "deleteDir, can't write to parent directory, still fine" $ do
|
||||||
createDir' (specDir `ba` "noWritable/foo")
|
createDir' "noWritable/foo"
|
||||||
noWritableDirPerms (specDir `ba` "noWritable")
|
noWritableDirPerms "noWritable"
|
||||||
(deleteDir' (specDir `ba` "noWritable/foo")
|
(deleteDir' "noWritable/foo")
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied))
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
normalDirPerms (specDir `ba` "noWritable")
|
normalDirPerms "noWritable"
|
||||||
deleteDir' (specDir `ba` "noWritable/foo")
|
deleteDir' "noWritable/foo"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -21,49 +21,58 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "foo"
|
||||||
|
createSymlink' "syml" "foo"
|
||||||
|
createDir' "dir"
|
||||||
|
createDir' "noPerms"
|
||||||
|
noPerms "noPerms"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/deleteFileSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
deleteFile' "foo"
|
||||||
|
deleteFile' "syml"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.deleteFile" $ do
|
describe "HPath.IO.deleteFile" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "deleteFile, regular file, all fine" $ do
|
it "deleteFile, regular file, all fine" $ do
|
||||||
createRegularFile' (specDir `ba` "testFile")
|
createRegularFile' "testFile"
|
||||||
deleteFile' (specDir `ba` "testFile")
|
deleteFile' "testFile"
|
||||||
getSymbolicLinkStatus (specDir `ba` "testFile")
|
getSymbolicLinkStatus "testFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "deleteFile, symlink, all fine" $ do
|
it "deleteFile, symlink, all fine" $ do
|
||||||
recreateSymlink' (specDir `ba` "syml")
|
recreateSymlink' "syml"
|
||||||
(specDir `ba` "testFile")
|
"testFile"
|
||||||
deleteFile' (specDir `ba` "testFile")
|
deleteFile' "testFile"
|
||||||
getSymbolicLinkStatus (specDir `ba` "testFile")
|
getSymbolicLinkStatus "testFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "deleteFile, wrong file type (directory)" $
|
it "deleteFile, wrong file type (directory)" $
|
||||||
deleteFile' (specDir `ba` "dir")
|
deleteFile' "dir"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "deleteFile, file does not exist" $
|
it "deleteFile, file does not exist" $
|
||||||
deleteFile' (specDir `ba` "doesNotExist")
|
deleteFile' "doesNotExist"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "deleteFile, can't read directory" $
|
it "deleteFile, can't read directory" $
|
||||||
deleteFile' (specDir `ba` "noPerms/blah")
|
deleteFile' "noPerms/blah"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ import Data.Maybe
|
|||||||
fromJust
|
fromJust
|
||||||
)
|
)
|
||||||
import qualified HPath as P
|
import qualified HPath as P
|
||||||
|
import HPath.IO
|
||||||
import Test.Hspec
|
import Test.Hspec
|
||||||
import System.IO.Error
|
import System.IO.Error
|
||||||
(
|
(
|
||||||
@@ -34,56 +35,71 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "file"
|
||||||
|
createRegularFile' "Lala"
|
||||||
|
createRegularFile' ".hidden"
|
||||||
|
createSymlink' "syml" "Lala"
|
||||||
|
createDir' "dir"
|
||||||
|
createSymlink' "dirsym" "dir"
|
||||||
|
createDir' "noPerms"
|
||||||
|
noPerms "noPerms"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/getDirsFilesSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
deleteFile' "file"
|
||||||
|
deleteFile' "Lala"
|
||||||
|
deleteFile' ".hidden"
|
||||||
|
deleteFile' "syml"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteFile' "dirsym"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.getDirsFiles" $ do
|
describe "HPath.IO.getDirsFiles" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "getDirsFiles, all fine" $ do
|
it "getDirsFiles, all fine" $
|
||||||
pwd <- fromJust <$> getEnv "PWD" >>= P.parseAbs
|
withRawTmpDir $ \p -> do
|
||||||
expectedFiles <- mapM P.parseRel [(specDir `ba ` ".hidden")
|
expectedFiles <- mapM P.parseRel [".hidden"
|
||||||
,(specDir `ba ` "Lala")
|
,"Lala"
|
||||||
,(specDir `ba ` "dir")
|
,"dir"
|
||||||
,(specDir `ba ` "dirsym")
|
,"dirsym"
|
||||||
,(specDir `ba ` "file")
|
,"file"
|
||||||
,(specDir `ba ` "noPerms")
|
,"noPerms"
|
||||||
,(specDir `ba ` "syml")]
|
,"syml"]
|
||||||
(fmap sort $ getDirsFiles' specDir)
|
(fmap sort $ getDirsFiles p)
|
||||||
`shouldReturn` fmap (pwd P.</>) expectedFiles
|
`shouldReturn` fmap (p P.</>) expectedFiles
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "getDirsFiles, nonexistent directory" $
|
it "getDirsFiles, nonexistent directory" $
|
||||||
getDirsFiles' (specDir `ba ` "nothingHere")
|
getDirsFiles' "nothingHere"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "getDirsFiles, wrong file type (file)" $
|
it "getDirsFiles, wrong file type (file)" $
|
||||||
getDirsFiles' (specDir `ba ` "file")
|
getDirsFiles' "file"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InappropriateType)
|
(\e -> ioeGetErrorType e == InappropriateType)
|
||||||
|
|
||||||
it "getDirsFiles, wrong file type (symlink to file)" $
|
it "getDirsFiles, wrong file type (symlink to file)" $
|
||||||
getDirsFiles' (specDir `ba ` "syml")
|
getDirsFiles' "syml"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InvalidArgument)
|
(\e -> ioeGetErrorType e == InvalidArgument)
|
||||||
|
|
||||||
it "getDirsFiles, wrong file type (symlink to dir)" $
|
it "getDirsFiles, wrong file type (symlink to dir)" $
|
||||||
getDirsFiles' (specDir `ba ` "dirsym")
|
getDirsFiles' "dirsym"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InvalidArgument)
|
(\e -> ioeGetErrorType e == InvalidArgument)
|
||||||
|
|
||||||
it "getDirsFiles, can't open directory" $
|
it "getDirsFiles, can't open directory" $
|
||||||
getDirsFiles' (specDir `ba ` "noPerms")
|
getDirsFiles' "noPerms"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
|
|||||||
@@ -18,53 +18,66 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "regularfile"
|
||||||
|
createSymlink' "symlink" "regularfile"
|
||||||
|
createSymlink' "brokenSymlink" "broken"
|
||||||
|
createDir' "directory"
|
||||||
|
createSymlink' "symlinkD" "directory"
|
||||||
|
createDir' "noPerms"
|
||||||
|
noPerms "noPerms"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/getFileTypeSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
deleteFile' "regularfile"
|
||||||
|
deleteFile' "symlink"
|
||||||
|
deleteFile' "brokenSymlink"
|
||||||
|
deleteDir' "directory"
|
||||||
|
deleteFile' "symlinkD"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.getFileType" $ do
|
describe "HPath.IO.getFileType" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "getFileType, regular file" $
|
it "getFileType, regular file" $
|
||||||
getFileType' (specDir `ba` "regularfile")
|
getFileType' "regularfile"
|
||||||
`shouldReturn` RegularFile
|
`shouldReturn` RegularFile
|
||||||
|
|
||||||
it "getFileType, directory" $
|
it "getFileType, directory" $
|
||||||
getFileType' (specDir `ba` "directory")
|
getFileType' "directory"
|
||||||
`shouldReturn` Directory
|
`shouldReturn` Directory
|
||||||
|
|
||||||
it "getFileType, directory with null permissions" $
|
it "getFileType, directory with null permissions" $
|
||||||
getFileType' (specDir `ba` "noPerms")
|
getFileType' "noPerms"
|
||||||
`shouldReturn` Directory
|
`shouldReturn` Directory
|
||||||
|
|
||||||
it "getFileType, symlink to file" $
|
it "getFileType, symlink to file" $
|
||||||
getFileType' (specDir `ba` "symlink")
|
getFileType' "symlink"
|
||||||
`shouldReturn` SymbolicLink
|
`shouldReturn` SymbolicLink
|
||||||
|
|
||||||
it "getFileType, symlink to directory" $
|
it "getFileType, symlink to directory" $
|
||||||
getFileType' (specDir `ba` "symlinkD")
|
getFileType' "symlinkD"
|
||||||
`shouldReturn` SymbolicLink
|
`shouldReturn` SymbolicLink
|
||||||
|
|
||||||
it "getFileType, broken symlink" $
|
it "getFileType, broken symlink" $
|
||||||
getFileType' (specDir `ba` "brokenSymlink")
|
getFileType' "brokenSymlink"
|
||||||
`shouldReturn` SymbolicLink
|
`shouldReturn` SymbolicLink
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "getFileType, file does not exist" $
|
it "getFileType, file does not exist" $
|
||||||
getFileType' (specDir `ba` "nothingHere")
|
getFileType' "nothingHere"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "getFileType, can't open directory" $
|
it "getFileType, can't open directory" $
|
||||||
getFileType' (specDir `ba` "noPerms/forz")
|
getFileType' "noPerms/forz"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
|
|||||||
@@ -18,76 +18,92 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "myFile"
|
||||||
|
createSymlink' "myFileL" "myFile"
|
||||||
|
createDir' "alreadyExistsD"
|
||||||
|
createDir' "dir"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerm"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerm"
|
||||||
|
writeFile' "myFile" "Blahfaselgagaga"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/moveFileOverwriteSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerm"
|
||||||
|
deleteFile' "myFile"
|
||||||
|
deleteFile' "myFileL"
|
||||||
|
deleteDir' "alreadyExistsD"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerm"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.moveFileOverwrite" $ do
|
describe "HPath.IO.moveFileOverwrite" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "moveFileOverwrite, all fine" $
|
it "moveFileOverwrite, all fine" $
|
||||||
moveFileOverwrite' (specDir `ba` "myFile")
|
moveFileOverwrite' "myFile"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
|
|
||||||
it "moveFileOverwrite, all fine" $
|
it "moveFileOverwrite, all fine" $
|
||||||
moveFileOverwrite' (specDir `ba` "myFile")
|
moveFileOverwrite' "myFile"
|
||||||
(specDir `ba` "dir/movedFile")
|
"dir/movedFile"
|
||||||
|
|
||||||
it "moveFileOverwrite, all fine on symlink" $
|
it "moveFileOverwrite, all fine on symlink" $
|
||||||
moveFileOverwrite' (specDir `ba` "myFileL")
|
moveFileOverwrite' "myFileL"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
|
|
||||||
it "moveFileOverwrite, all fine on directory" $
|
it "moveFileOverwrite, all fine on directory" $
|
||||||
moveFileOverwrite' (specDir `ba` "dir")
|
moveFileOverwrite' "dir"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
|
|
||||||
it "moveFileOverwrite, destination file already exists" $
|
it "moveFileOverwrite, destination file already exists" $
|
||||||
moveFileOverwrite' (specDir `ba` "myFile")
|
moveFileOverwrite' "myFile"
|
||||||
(specDir `ba` "alreadyExists")
|
"alreadyExists"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "moveFileOverwrite, source file does not exist" $
|
it "moveFileOverwrite, source file does not exist" $
|
||||||
moveFileOverwrite' (specDir `ba` "fileDoesNotExist")
|
moveFileOverwrite' "fileDoesNotExist"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "moveFileOverwrite, can't write to destination directory" $
|
it "moveFileOverwrite, can't write to destination directory" $
|
||||||
moveFileOverwrite' (specDir `ba` "myFile")
|
moveFileOverwrite' "myFile"
|
||||||
(specDir `ba` "noWritePerm/movedFile")
|
"noWritePerm/movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "moveFileOverwrite, can't open destination directory" $
|
it "moveFileOverwrite, can't open destination directory" $
|
||||||
moveFileOverwrite' (specDir `ba` "myFile")
|
moveFileOverwrite' "myFile"
|
||||||
(specDir `ba` "noPerms/movedFile")
|
"noPerms/movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "moveFileOverwrite, can't open source directory" $
|
it "moveFileOverwrite, can't open source directory" $
|
||||||
moveFileOverwrite' (specDir `ba` "noPerms/myFile")
|
moveFileOverwrite' "noPerms/myFile"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
-- custom failures --
|
-- custom failures --
|
||||||
it "moveFileOverwrite, move from file to dir" $
|
it "moveFileOverwrite, move from file to dir" $
|
||||||
moveFileOverwrite' (specDir `ba` "myFile")
|
moveFileOverwrite' "myFile"
|
||||||
(specDir `ba` "alreadyExistsD")
|
"alreadyExistsD"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isDirDoesExist
|
isDirDoesExist
|
||||||
|
|
||||||
it "moveFileOverwrite, source and dest are same file" $
|
it "moveFileOverwrite, source and dest are same file" $
|
||||||
moveFileOverwrite' (specDir `ba` "myFile")
|
moveFileOverwrite' "myFile"
|
||||||
(specDir `ba` "myFile")
|
"myFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isSameFile
|
isSameFile
|
||||||
|
|
||||||
|
|||||||
@@ -18,78 +18,96 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "myFile"
|
||||||
|
createSymlink' "myFileL" "myFile"
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createDir' "alreadyExistsD"
|
||||||
|
createDir' "dir"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerm"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerm"
|
||||||
|
writeFile' "myFile" "Blahfaselgagaga"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/moveFileSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerm"
|
||||||
|
deleteFile' "myFile"
|
||||||
|
deleteFile' "myFileL"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteDir' "alreadyExistsD"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerm"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.moveFile" $ do
|
describe "HPath.IO.moveFile" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "moveFile, all fine" $
|
it "moveFile, all fine" $
|
||||||
moveFile' (specDir `ba` "myFile")
|
moveFile' "myFile"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
|
|
||||||
it "moveFile, all fine" $
|
it "moveFile, all fine" $
|
||||||
moveFile' (specDir `ba` "myFile")
|
moveFile' "myFile"
|
||||||
(specDir `ba` "dir/movedFile")
|
"dir/movedFile"
|
||||||
|
|
||||||
it "moveFile, all fine on symlink" $
|
it "moveFile, all fine on symlink" $
|
||||||
moveFile' (specDir `ba` "myFileL")
|
moveFile' "myFileL"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
|
|
||||||
it "moveFile, all fine on directory" $
|
it "moveFile, all fine on directory" $
|
||||||
moveFile' (specDir `ba` "dir")
|
moveFile' "dir"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "moveFile, source file does not exist" $
|
it "moveFile, source file does not exist" $
|
||||||
moveFile' (specDir `ba` "fileDoesNotExist")
|
moveFile' "fileDoesNotExist"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "moveFile, can't write to destination directory" $
|
it "moveFile, can't write to destination directory" $
|
||||||
moveFile' (specDir `ba` "myFile")
|
moveFile' "myFile"
|
||||||
(specDir `ba` "noWritePerm/movedFile")
|
"noWritePerm/movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "moveFile, can't open destination directory" $
|
it "moveFile, can't open destination directory" $
|
||||||
moveFile' (specDir `ba` "myFile")
|
moveFile' "myFile"
|
||||||
(specDir `ba` "noPerms/movedFile")
|
"noPerms/movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "moveFile, can't open source directory" $
|
it "moveFile, can't open source directory" $
|
||||||
moveFile' (specDir `ba` "noPerms/myFile")
|
moveFile' "noPerms/myFile"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
-- custom failures --
|
-- custom failures --
|
||||||
it "moveFile, destination file already exists" $
|
it "moveFile, destination file already exists" $
|
||||||
moveFile' (specDir `ba` "myFile")
|
moveFile' "myFile"
|
||||||
(specDir `ba` "alreadyExists")
|
"alreadyExists"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isFileDoesExist
|
isFileDoesExist
|
||||||
|
|
||||||
it "moveFile, move from file to dir" $
|
it "moveFile, move from file to dir" $
|
||||||
moveFile' (specDir `ba` "myFile")
|
moveFile' "myFile"
|
||||||
(specDir `ba` "alreadyExistsD")
|
"alreadyExistsD"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isDirDoesExist
|
isDirDoesExist
|
||||||
|
|
||||||
it "moveFile, source and dest are same file" $
|
it "moveFile, source and dest are same file" $
|
||||||
moveFile' (specDir `ba` "myFile")
|
moveFile' "myFile"
|
||||||
(specDir `ba` "myFile")
|
"myFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isSameFile
|
isSameFile
|
||||||
|
|
||||||
|
|||||||
@@ -18,78 +18,95 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "myFile"
|
||||||
|
createSymlink' "myFileL" "myFile"
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createDir' "alreadyExistsD"
|
||||||
|
createDir' "dir"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerm"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerm"
|
||||||
|
writeFile' "myFile" "Blahfaselgagaga"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/recreateSymlinkSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerm"
|
||||||
|
deleteFile' "myFile"
|
||||||
|
deleteFile' "myFileL"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteDir' "alreadyExistsD"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerm"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.recreateSymlink" $ do
|
describe "HPath.IO.recreateSymlink" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "recreateSymLink, all fine" $ do
|
it "recreateSymLink, all fine" $ do
|
||||||
recreateSymlink' (specDir `ba` "myFileL")
|
recreateSymlink' "myFileL"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
removeFileIfExists (specDir `ba` "movedFile")
|
removeFileIfExists "movedFile"
|
||||||
|
|
||||||
it "recreateSymLink, all fine" $ do
|
it "recreateSymLink, all fine" $ do
|
||||||
recreateSymlink' (specDir `ba` "myFileL")
|
recreateSymlink' "myFileL"
|
||||||
(specDir `ba` "dir/movedFile")
|
"dir/movedFile"
|
||||||
removeFileIfExists (specDir `ba` "dir/movedFile")
|
removeFileIfExists "dir/movedFile"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "recreateSymLink, wrong input type (file)" $
|
it "recreateSymLink, wrong input type (file)" $
|
||||||
recreateSymlink' (specDir `ba` "myFile")
|
recreateSymlink' "myFile"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InvalidArgument)
|
(\e -> ioeGetErrorType e == InvalidArgument)
|
||||||
|
|
||||||
it "recreateSymLink, wrong input type (directory)" $
|
it "recreateSymLink, wrong input type (directory)" $
|
||||||
recreateSymlink' (specDir `ba` "dir")
|
recreateSymlink' "dir"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == InvalidArgument)
|
(\e -> ioeGetErrorType e == InvalidArgument)
|
||||||
|
|
||||||
it "recreateSymLink, can't write to destination directory" $
|
it "recreateSymLink, can't write to destination directory" $
|
||||||
recreateSymlink' (specDir `ba` "myFileL")
|
recreateSymlink' "myFileL"
|
||||||
(specDir `ba` "noWritePerm/movedFile")
|
"noWritePerm/movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "recreateSymLink, can't open destination directory" $
|
it "recreateSymLink, can't open destination directory" $
|
||||||
recreateSymlink' (specDir `ba` "myFileL")
|
recreateSymlink' "myFileL"
|
||||||
(specDir `ba` "noPerms/movedFile")
|
"noPerms/movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "recreateSymLink, can't open source directory" $
|
it "recreateSymLink, can't open source directory" $
|
||||||
recreateSymlink' (specDir `ba` "noPerms/myFileL")
|
recreateSymlink' "noPerms/myFileL"
|
||||||
(specDir `ba` "movedFile")
|
"movedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "recreateSymLink, destination file already exists" $
|
it "recreateSymLink, destination file already exists" $
|
||||||
recreateSymlink' (specDir `ba` "myFileL")
|
recreateSymlink' "myFileL"
|
||||||
(specDir `ba` "alreadyExists")
|
"alreadyExists"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == AlreadyExists)
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
it "recreateSymLink, destination already exists and is a dir" $
|
it "recreateSymLink, destination already exists and is a dir" $
|
||||||
recreateSymlink' (specDir `ba` "myFileL")
|
recreateSymlink' "myFileL"
|
||||||
(specDir `ba` "alreadyExistsD")
|
"alreadyExistsD"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == AlreadyExists)
|
(\e -> ioeGetErrorType e == AlreadyExists)
|
||||||
|
|
||||||
-- custom failures --
|
-- custom failures --
|
||||||
it "recreateSymLink, source and destination are the same file" $
|
it "recreateSymLink, source and destination are the same file" $
|
||||||
recreateSymlink' (specDir `ba` "myFileL")
|
recreateSymlink' "myFileL"
|
||||||
(specDir `ba` "myFileL")
|
"myFileL"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isSameFile
|
isSameFile
|
||||||
|
|
||||||
|
|||||||
@@ -18,78 +18,95 @@ import qualified Data.ByteString as BS
|
|||||||
import Data.ByteString.UTF8 (toString)
|
import Data.ByteString.UTF8 (toString)
|
||||||
|
|
||||||
|
|
||||||
ba :: BS.ByteString -> BS.ByteString -> BS.ByteString
|
setupFiles :: IO ()
|
||||||
ba = BS.append
|
setupFiles = do
|
||||||
|
createRegularFile' "myFile"
|
||||||
|
createSymlink' "myFileL" "myFile"
|
||||||
|
createRegularFile' "alreadyExists"
|
||||||
|
createDir' "alreadyExistsD"
|
||||||
|
createDir' "dir"
|
||||||
|
createDir' "noPerms"
|
||||||
|
createDir' "noWritePerm"
|
||||||
|
noPerms "noPerms"
|
||||||
|
noWritableDirPerms "noWritePerm"
|
||||||
|
writeFile' "myFile" "Blahfaselgagaga"
|
||||||
|
|
||||||
specDir :: BS.ByteString
|
|
||||||
specDir = "test/HPath/IO/renameFileSpec/"
|
|
||||||
|
|
||||||
specDir' :: String
|
cleanupFiles :: IO ()
|
||||||
specDir' = toString specDir
|
cleanupFiles = do
|
||||||
|
normalDirPerms "noPerms"
|
||||||
|
normalDirPerms "noWritePerm"
|
||||||
|
deleteFile' "myFile"
|
||||||
|
deleteFile' "myFileL"
|
||||||
|
deleteFile' "alreadyExists"
|
||||||
|
deleteDir' "alreadyExistsD"
|
||||||
|
deleteDir' "dir"
|
||||||
|
deleteDir' "noPerms"
|
||||||
|
deleteDir' "noWritePerm"
|
||||||
|
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec =
|
spec = before_ setupFiles $ after_ cleanupFiles $
|
||||||
describe "HPath.IO.renameFile" $ do
|
describe "HPath.IO.renameFile" $ do
|
||||||
|
|
||||||
-- successes --
|
-- successes --
|
||||||
it "renameFile, all fine" $
|
it "renameFile, all fine" $
|
||||||
renameFile' (specDir `ba` "myFile")
|
renameFile' "myFile"
|
||||||
(specDir `ba` "renamedFile")
|
"renamedFile"
|
||||||
|
|
||||||
it "renameFile, all fine" $
|
it "renameFile, all fine" $
|
||||||
renameFile' (specDir `ba` "myFile")
|
renameFile' "myFile"
|
||||||
(specDir `ba` "dir/renamedFile")
|
"dir/renamedFile"
|
||||||
|
|
||||||
it "renameFile, all fine on symlink" $
|
it "renameFile, all fine on symlink" $
|
||||||
renameFile' (specDir `ba` "myFileL")
|
renameFile' "myFileL"
|
||||||
(specDir `ba` "renamedFile")
|
"renamedFile"
|
||||||
|
|
||||||
it "renameFile, all fine on directory" $
|
it "renameFile, all fine on directory" $
|
||||||
renameFile' (specDir `ba` "dir")
|
renameFile' "dir"
|
||||||
(specDir `ba` "renamedFile")
|
"renamedFile"
|
||||||
|
|
||||||
-- posix failures --
|
-- posix failures --
|
||||||
it "renameFile, source file does not exist" $
|
it "renameFile, source file does not exist" $
|
||||||
renameFile' (specDir `ba` "fileDoesNotExist")
|
renameFile' "fileDoesNotExist"
|
||||||
(specDir `ba` "renamedFile")
|
"renamedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == NoSuchThing)
|
(\e -> ioeGetErrorType e == NoSuchThing)
|
||||||
|
|
||||||
it "renameFile, can't write to output directory" $
|
it "renameFile, can't write to output directory" $
|
||||||
renameFile' (specDir `ba` "myFile")
|
renameFile' "myFile"
|
||||||
(specDir `ba` "noWritePerm/renamedFile")
|
"noWritePerm/renamedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "renameFile, can't open output directory" $
|
it "renameFile, can't open output directory" $
|
||||||
renameFile' (specDir `ba` "myFile")
|
renameFile' "myFile"
|
||||||
(specDir `ba` "noPerms/renamedFile")
|
"noPerms/renamedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
it "renameFile, can't open source directory" $
|
it "renameFile, can't open source directory" $
|
||||||
renameFile' (specDir `ba` "noPerms/myFile")
|
renameFile' "noPerms/myFile"
|
||||||
(specDir `ba` "renamedFile")
|
"renamedFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
(\e -> ioeGetErrorType e == PermissionDenied)
|
(\e -> ioeGetErrorType e == PermissionDenied)
|
||||||
|
|
||||||
-- custom failures --
|
-- custom failures --
|
||||||
it "renameFile, destination file already exists" $
|
it "renameFile, destination file already exists" $
|
||||||
renameFile' (specDir `ba` "myFile")
|
renameFile' "myFile"
|
||||||
(specDir `ba` "alreadyExists")
|
"alreadyExists"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isFileDoesExist
|
isFileDoesExist
|
||||||
|
|
||||||
it "renameFile, move from file to dir" $
|
it "renameFile, move from file to dir" $
|
||||||
renameFile' (specDir `ba` "myFile")
|
renameFile' "myFile"
|
||||||
(specDir `ba` "alreadyExistsD")
|
"alreadyExistsD"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isDirDoesExist
|
isDirDoesExist
|
||||||
|
|
||||||
it "renameFile, source and dest are same file" $
|
it "renameFile, source and dest are same file" $
|
||||||
renameFile' (specDir `ba` "myFile")
|
renameFile' "myFile"
|
||||||
(specDir `ba` "myFile")
|
"myFile"
|
||||||
`shouldThrow`
|
`shouldThrow`
|
||||||
isSameFile
|
isSameFile
|
||||||
|
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
nothing
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dir
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
file
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
sda
|
|
||||||
|
|
||||||
!!1
|
|
||||||
sda
|
|
||||||
|
|
||||||
|
|
||||||
11
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
das
|
|
||||||
sda
|
|
||||||
sda
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
sda
|
|
||||||
|
|
||||||
!!1
|
|
||||||
sda
|
|
||||||
|
|
||||||
|
|
||||||
11
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
das
|
|
||||||
sda
|
|
||||||
sda
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
sda
|
|
||||||
|
|
||||||
!!1
|
|
||||||
sda
|
|
||||||
|
|
||||||
|
|
||||||
11
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
das
|
|
||||||
sda
|
|
||||||
sda
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
inputDir/
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
sda
|
|
||||||
|
|
||||||
!!1
|
|
||||||
sda
|
|
||||||
|
|
||||||
|
|
||||||
11
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
das
|
|
||||||
sda
|
|
||||||
sda
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
sda
|
|
||||||
|
|
||||||
!!1
|
|
||||||
sda
|
|
||||||
|
|
||||||
|
|
||||||
11
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
dadasasddas
|
|
||||||
das
|
|
||||||
sda
|
|
||||||
sda
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
inputDir/
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
adaöölsdaöl
|
|
||||||
dsalö
|
|
||||||
ölsda
|
|
||||||
ääödsf
|
|
||||||
äsdfä
|
|
||||||
öä453
|
|
||||||
öä
|
|
||||||
435
|
|
||||||
ä45343
|
|
||||||
5
|
|
||||||
453
|
|
||||||
453453453
|
|
||||||
das
|
|
||||||
asd
|
|
||||||
das
|
|
||||||
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
abc
|
|
||||||
def
|
|
||||||
|
|
||||||
dsadasdsa
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
inputFile
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
abc
|
|
||||||
def
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
inputFile
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dir
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dir
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
foo
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
dir
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user