{-# LANGUAGE CPP #-} {-# LANGUAGE ExistentialQuantification #-} {-# LANGUAGE Safe #-} -- | A more type-safe version of file paths -- -- This module provides the basic 'Path' abstraction. See also -- "System.Path.IO" which extends this module by thin wrappers -- wrappers around common 'IO' operations. module System.Path ( -- * Paths Path(..) , castRoot -- * FilePath-like operations on paths with arbitrary roots , takeDirectory , takeFileName , (<.>) , splitExtension , takeExtension -- * Unrooted paths , Unrooted , (</>) , rootPath , unrootPath , toUnrootedFilePath , fromUnrootedFilePath , fragment , joinFragments , splitFragments -- , isPathPrefixOf -- * File-system paths , FsRoot(..) , FsPath(..) , Relative , Absolute , HomeDir -- ** Conversions , toFilePath , fromFilePath , makeAbsolute , fromAbsoluteFilePath {- -- * Wrappers around Codec.Archive.Tar , Tar , tarIndexLookup , tarAppend -- * Wrappers around Network.URI , Web , toURIPath , fromURIPath , uriPath , modifyUriPath -} ) where #if !MIN_VERSION_base(4,8,0) import Control.Applicative ((<$>)) #endif import Control.DeepSeq (NFData (rnf)) -- import Data.List (isPrefixOf) import qualified System.Directory as Dir import qualified System.FilePath as FP.Native import qualified System.FilePath.Posix as FP.Posix {------------------------------------------------------------------------------- Paths -------------------------------------------------------------------------------} -- | Paths -- -- A 'Path' is simply a 'FilePath' with a type-level tag indicating where this -- path is rooted (relative to the current directory, absolute path, relative to -- a web domain, whatever). Most operations on 'Path' are just lifted versions -- of the operations on the underlying 'FilePath'. The tag however allows us to -- give a lot of operations a more meaningful type. For instance, it does not -- make sense to append two absolute paths together; instead, we can only append -- an unrooted path to another path. It also means we avoid bugs where we use -- one kind of path where we expect another. newtype Path a = Path FilePath -- always a Posix style path internally deriving (Show, Eq, Ord) instance NFData (Path a) where rnf (Path p) = rnf p mkPathNative :: FilePath -> Path a mkPathNative = Path . FP.Posix.joinPath . FP.Native.splitDirectories unPathNative :: Path a -> FilePath unPathNative (Path fp) = FP.Native.joinPath . FP.Posix.splitDirectories $ fp mkPathPosix :: FilePath -> Path a mkPathPosix = Path unPathPosix :: Path a -> FilePath unPathPosix (Path fp) = fp -- | Reinterpret the root of a path -- -- This literally just changes the type-level tag; use with caution! castRoot :: Path root -> Path root' castRoot (Path fp) = Path fp {------------------------------------------------------------------------------- FilePath-like operations on paths with an arbitrary root -------------------------------------------------------------------------------} -- | Wrapped 'FP.Posix.takeDirectory' takeDirectory :: Path a -> Path a takeDirectory = liftFP FP.Posix.takeDirectory -- | Wrapped 'FP.Posix.takeFileName' takeFileName :: Path a -> String takeFileName = liftFromFP FP.Posix.takeFileName -- | Wrapped 'FP.Posix.<.>' (<.>) :: Path a -> String -> Path a fp <.> ext = liftFP (FP.Posix.<.> ext) fp -- | Wrapped 'FP.Posix.splitExtension' splitExtension :: Path a -> (Path a, String) splitExtension (Path fp) = (Path fp', ext) where (fp', ext) = FP.Posix.splitExtension fp -- | Wrapped 'FP.Posix.takeExtension' takeExtension :: Path a -> String takeExtension (Path fp) = FP.Posix.takeExtension fp {------------------------------------------------------------------------------- Unrooted paths -------------------------------------------------------------------------------} -- | Type-level tag for unrooted paths -- -- Unrooted paths need a root before they can be interpreted. data Unrooted -- instance Pretty (Path Unrooted) where -- pretty (Path fp) = fp -- | Wrapped 'FP.Posix.</>' (</>) :: Path a -> Path Unrooted -> Path a (</>) = liftFP2 (FP.Posix.</>) -- | Reinterpret an unrooted path -- -- This is an alias for 'castRoot'; see comments there. rootPath :: Path Unrooted -> Path root rootPath (Path fp) = Path fp -- | Forget a path's root -- -- This is an alias for 'castRoot'; see comments there. unrootPath :: Path root -> Path Unrooted unrootPath (Path fp) = Path fp -- | Convert a relative\/unrooted Path to a FilePath (using POSIX style -- directory separators). -- -- See also 'toAbsoluteFilePath' -- toUnrootedFilePath :: Path Unrooted -> FilePath toUnrootedFilePath = unPathPosix -- | Convert from a relative\/unrooted FilePath (using POSIX style directory -- separators). -- fromUnrootedFilePath :: FilePath -> Path Unrooted fromUnrootedFilePath = mkPathPosix -- | A path fragment (like a single directory or filename) fragment :: String -> Path Unrooted fragment = Path -- | Wrapped 'FP.Posix.joinPath' joinFragments :: [String] -> Path Unrooted joinFragments = liftToFP FP.Posix.joinPath -- | Wrapped 'FP.Posix.splitDirectories' splitFragments :: Path Unrooted -> [String] splitFragments (Path fp) = FP.Posix.splitDirectories fp -- FIXME -- isPathPrefixOf :: Path Unrooted -> Path Unrooted -> Bool -- isPathPrefixOf = liftFromFP2 isPrefixOf {------------------------------------------------------------------------------- File-system paths -------------------------------------------------------------------------------} data Relative data Absolute data HomeDir -- instance Pretty (Path Absolute) where -- pretty (Path fp) = fp -- instance Pretty (Path Relative) where -- pretty (Path fp) = "./" ++ fp -- instance Pretty (Path HomeDir) where -- pretty (Path fp) = "~/" ++ fp -- | A file system root can be interpreted as an (absolute) FilePath class FsRoot root where -- | Convert a Path to an absolute FilePath (using native style directory separators). -- toAbsoluteFilePath :: Path root -> IO FilePath instance FsRoot Relative where toAbsoluteFilePath p = go (unPathNative p) where go :: FilePath -> IO FilePath #if MIN_VERSION_directory(1,2,2) go = Dir.makeAbsolute #else -- copied implementation from the directory package go = (FP.Native.normalise <$>) . absolutize absolutize path -- avoid the call to `getCurrentDirectory` if we can | FP.Native.isRelative path = (FP.Native.</> path) . FP.Native.addTrailingPathSeparator <$> Dir.getCurrentDirectory | otherwise = return path #endif instance FsRoot Absolute where toAbsoluteFilePath = return . unPathNative instance FsRoot HomeDir where toAbsoluteFilePath p = do home <- Dir.getHomeDirectory return $ home FP.Native.</> unPathNative p -- | Abstract over a file system root -- -- see 'fromFilePath' data FsPath = forall root. FsRoot root => FsPath (Path root) instance NFData FsPath where rnf (FsPath a) = rnf a {------------------------------------------------------------------------------- Conversions -------------------------------------------------------------------------------} toFilePath :: Path Absolute -> FilePath toFilePath = unPathNative fromFilePath :: FilePath -> FsPath fromFilePath fp | FP.Native.isAbsolute fp = FsPath (mkPathNative fp :: Path Absolute) | Just fp' <- atHome fp = FsPath (mkPathNative fp' :: Path HomeDir) | otherwise = FsPath (mkPathNative fp :: Path Relative) where -- TODO: I don't know if there a standard way that Windows users refer to -- their home directory. For now, we'll only interpret '~'. Everybody else -- can specify an absolute path if this doesn't work. atHome :: FilePath -> Maybe FilePath atHome "~" = Just "" atHome ('~':sep:fp') | FP.Native.isPathSeparator sep = Just fp' atHome _otherwise = Nothing makeAbsolute :: FsPath -> IO (Path Absolute) makeAbsolute (FsPath p) = mkPathNative <$> toAbsoluteFilePath p fromAbsoluteFilePath :: FilePath -> Path Absolute fromAbsoluteFilePath fp | FP.Native.isAbsolute fp = mkPathNative fp | otherwise = error "fromAbsoluteFilePath: not an absolute path" {------------------------------------------------------------------------------- Wrappers around Codec.Archive.Tar.* ------------------------------------------------------------------------------- data Tar instance Pretty (Path Tar) where pretty (Path fp) = "<tarball>/" ++ fp tarIndexLookup :: TarIndex.TarIndex -> Path Tar -> Maybe TarIndex.TarIndexEntry tarIndexLookup index path = TarIndex.lookup index path' where path' :: FilePath path' = toUnrootedFilePath $ unrootPath path tarAppend :: (FsRoot root, FsRoot root') => Path root -- ^ Path of the @.tar@ file -> Path root' -- ^ Base directory -> [Path Tar] -- ^ Files to add, relative to the base dir -> IO () tarAppend tarFile baseDir contents = do tarFile' <- toAbsoluteFilePath tarFile baseDir' <- toAbsoluteFilePath baseDir Tar.append tarFile' baseDir' contents' where contents' :: [FilePath] contents' = map (unPathNative . unrootPath) contents ------------------------------------------------------------------------------- Wrappers around Network.URI ------------------------------------------------------------------------------- data Web toURIPath :: FilePath -> Path Web toURIPath = rootPath . fromUnrootedFilePath fromURIPath :: Path Web -> FilePath fromURIPath = toUnrootedFilePath . unrootPath uriPath :: URI.URI -> Path Web uriPath = toURIPath . URI.uriPath modifyUriPath :: URI.URI -> (Path Web -> Path Web) -> URI.URI modifyUriPath uri f = uri { URI.uriPath = f' (URI.uriPath uri) } where f' :: FilePath -> FilePath f' = fromURIPath . f . toURIPath -} {------------------------------------------------------------------------------- Auxiliary -------------------------------------------------------------------------------} liftFP :: (FilePath -> FilePath) -> Path a -> Path b liftFP f (Path fp) = Path (f fp) liftFP2 :: (FilePath -> FilePath -> FilePath) -> Path a -> Path b -> Path c liftFP2 f (Path fp) (Path fp') = Path (f fp fp') liftFromFP :: (FilePath -> x) -> Path a -> x liftFromFP f (Path fp) = f fp -- liftFromFP2 :: (FilePath -> FilePath -> x) -> Path a -> Path b -> x -- liftFromFP2 f (Path fp) (Path fp') = f fp fp' liftToFP :: (x -> FilePath) -> x -> Path a liftToFP f x = Path (f x)