diff --git a/.github/workflows/emulated.yml b/.github/workflows/emulated.yml index 28d1c12..b92120d 100644 --- a/.github/workflows/emulated.yml +++ b/.github/workflows/emulated.yml @@ -29,10 +29,52 @@ jobs: githubToken: ${{ github.token }} install: | apt-get update -y - apt-get install -y curl ghc libghc-tasty-quickcheck-dev libghc-file-embed-dev libghc-temporary-dev + apt-get install -y autoconf build-essential curl ghc libghc-tasty-quickcheck-dev libghc-file-embed-dev libghc-temporary-dev run: | - find Codec -iname '*.hs' -type f -exec sed -i.bck 's/"os-string"//g' {} \; - curl -s https://hackage.haskell.org/package/os-string-2.0.3/os-string-2.0.3.tar.gz | tar xz + curl -s https://hackage.haskell.org/package/os-string-2.0.6/os-string-2.0.6.tar.gz | tar xz + curl -s https://hackage.haskell.org/package/filepath-1.5.3.0/filepath-1.5.3.0.tar.gz | tar xz + curl -s https://hackage.haskell.org/package/file-io-0.1.4/file-io-0.1.4.tar.gz | tar xz + curl -s https://hackage.haskell.org/package/unix-2.8.5.1/unix-2.8.5.1.tar.gz | tar xz + curl -s https://hackage.haskell.org/package/directory-1.3.8.5/directory-1.3.8.5.tar.gz | tar xz + + cd unix-2.8.5.1 + chmod +x configure + ./configure + find /usr/lib/ghc -iname HsFFI.h + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/Directory.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/Directory/Common.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/Directory/PosixPath.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/Env/Internal.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/Files/Common.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/Files/PosixString.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/IO/Common.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/IO/PosixString.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/User/ByteString.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/User/Common.hsc + + sed -i -e 's/MIN_VERSION_base(4, 11, 0)/1/g' System/Posix/Files.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/Files.hsc + + sed -i -e 's/MIN_VERSION_filepath(1, 5, 0)/1/g' System/Posix/Env/PosixString.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/Env/PosixString.hsc + + sed -i -e 's/MIN_VERSION_filepath(1, 5, 0)/1/g' System/Posix/PosixPath/FilePath.hsc + sed -i -e 's/MIN_VERSION_base(4, 11, 0)/1/g' System/Posix/PosixPath/FilePath.hsc + hsc2hs -Iinclude -I$(dirname $(find /usr/lib/ghc -iname HsFFI.h)) System/Posix/PosixPath/FilePath.hsc + cd .. + + cd directory-1.3.8.5 + chmod +x configure + ./configure + hsc2hs -I. System/Directory/Internal/Posix.hsc + cd .. + + find . -iname '*.hs' -type f -exec sed -i.bck 's/import "filepath"/import/g' {} \; + find . -iname '*.hs' -type f -exec sed -i.bck 's/import "os-string"/import/g' {} \; + find . -iname '*.hs' -type f -exec sed -i.bck 's/import qualified "filepath"/import qualified/g' {} \; + find . -iname '*.hs' -type f -exec sed -i.bck 's/import qualified "os-string"/import qualified/g' {} \; + sed -i -e 's/MIN_VERSION_filepath(1, 5, 0)/1/g' file-io-0.1.4/posix/System/File/Platform.hs + sed -i -e 's/MIN_VERSION_filepath(1, 5, 0)/1/g' file-io-0.1.4/System/File/OsPath/Internal.hs ghc --version - ghc --make -itest:os-string-2.0.3 -o Main test/Properties.hs +RTS -s + ghc --make -fno-safe-haskell -itest:os-string-2.0.6:filepath-1.5.3.0:file-io-0.1.4:file-io-0.1.4/posix:unix-2.8.5.1:directory-1.3.8.5 -Iunix-2.8.5.1/include:directory-1.3.8.5 -o Main test/Properties.hs +RTS -s ./Main +RTS -s diff --git a/.github/workflows/haskell-ci.yml b/.github/workflows/haskell-ci.yml index 2390746..a8d06c6 100644 --- a/.github/workflows/haskell-ci.yml +++ b/.github/workflows/haskell-ci.yml @@ -197,7 +197,7 @@ jobs: echo " ghc-options: -Werror=missing-methods" >> cabal.project cat >> cabal.project <> cabal.project.local + $HCPKG list --simple-output --names-only | perl -ne 'for (split /\s+/) { print "constraints: any.$_ installed\n" unless /^(bytestring|directory|filepath|htar|tar|unix)$/; }' >> cabal.project.local cat cabal.project cat cabal.project.local - name: dump install plan diff --git a/Codec/Archive/Tar/PackAscii.hs b/Codec/Archive/Tar/PackAscii.hs index d1b810c..7d61fe1 100644 --- a/Codec/Archive/Tar/PackAscii.hs +++ b/Codec/Archive/Tar/PackAscii.hs @@ -7,6 +7,7 @@ module Codec.Archive.Tar.PackAscii , posixToByteString , byteToPosixString , packAscii + , filePathToOsPath ) where import Data.ByteString (ByteString) @@ -16,6 +17,7 @@ import Data.Char import GHC.Stack import System.IO.Unsafe (unsafePerformIO) import "os-string" System.OsString.Posix (PosixString) +import qualified "filepath" System.OsPath as OS import qualified "os-string" System.OsString.Posix as PS import qualified "os-string" System.OsString.Internal.Types as PS @@ -35,3 +37,6 @@ packAscii :: HasCallStack => FilePath -> BS.Char8.ByteString packAscii xs | all isAscii xs = BS.Char8.pack xs | otherwise = error $ "packAscii: only ASCII inputs are supported, but got " ++ xs + +filePathToOsPath :: FilePath -> OS.OsPath +filePathToOsPath = unsafePerformIO . OS.encodeFS diff --git a/Codec/Archive/Tar/Unpack.hs b/Codec/Archive/Tar/Unpack.hs index f1335ae..d757176 100644 --- a/Codec/Archive/Tar/Unpack.hs +++ b/Codec/Archive/Tar/Unpack.hs @@ -27,6 +27,7 @@ module Codec.Archive.Tar.Unpack ( import Codec.Archive.Tar.Types import Codec.Archive.Tar.Check import Codec.Archive.Tar.LongNames +import Codec.Archive.Tar.PackAscii (filePathToOsPath) import Data.Bits ( testBit ) @@ -34,11 +35,13 @@ import Data.List (partition, nub) import Data.Maybe ( fromMaybe ) import qualified Data.ByteString.Char8 as Char8 import qualified Data.ByteString.Lazy as BS -import System.FilePath - ( () ) -import qualified System.FilePath as FilePath.Native +import Prelude hiding (writeFile) +import System.File.OsPath +import System.OsPath + ( OsPath, () ) +import qualified System.OsPath as FilePath.Native ( takeDirectory ) -import System.Directory +import System.Directory.OsPath ( createDirectoryIfMissing, copyFile, setPermissions, @@ -110,7 +113,7 @@ unpackAndCheck -> Entries e -- ^ Entries to upack -> IO () -unpackAndCheck secCB baseDir entries = do +unpackAndCheck secCB (filePathToOsPath -> baseDir) entries = do let resolvedEntries = decodeLongNames entries uEntries <- unpackEntries [] resolvedEntries let (hardlinks, symlinks) = partition (\(_, _, x) -> x) uEntries @@ -123,11 +126,11 @@ unpackAndCheck secCB baseDir entries = do -- files all over the place. unpackEntries :: Exception e - => [(FilePath, FilePath, Bool)] + => [(OsPath, OsPath, Bool)] -- ^ links (path, link, isHardLink) -> GenEntries FilePath FilePath (Either e DecodeLongNamesError) -- ^ entries - -> IO [(FilePath, FilePath, Bool)] + -> IO [(OsPath, OsPath, Bool)] unpackEntries _ (Fail err) = either throwIO throwIO err unpackEntries links Done = return links unpackEntries links (Next entry es) = do @@ -154,31 +157,37 @@ unpackAndCheck secCB baseDir entries = do BlockDevice{} -> unpackEntries links es NamedPipe -> unpackEntries links es - extractFile permissions (fromFilePathToNative -> path) content mtime = do + extractFile :: Permissions -> FilePath -> BS.ByteString -> EpochTime -> IO () + extractFile permissions (filePathToNativeOsPath -> path) content mtime = do -- Note that tar archives do not make sure each directory is created -- before files they contain, indeed we may have to create several -- levels of directory. createDirectoryIfMissing True absDir - BS.writeFile absPath content + writeFile absPath content setOwnerPermissions absPath permissions setModTime absPath mtime where absDir = baseDir FilePath.Native.takeDirectory path absPath = baseDir path - extractDir (fromFilePathToNative -> path) mtime = do + extractDir :: FilePath -> EpochTime -> IO () + extractDir (filePathToNativeOsPath -> path) mtime = do createDirectoryIfMissing True absPath setModTime absPath mtime where absPath = baseDir path - saveLink isHardLink (fromFilePathToNative -> path) (fromFilePathToNative -> link) links - = seq (length path) - $ seq (length link) - $ (path, link, isHardLink):links - + saveLink + :: t + -> FilePath + -> FilePath + -> [(OsPath, OsPath, t)] + -> [(OsPath, OsPath, t)] + saveLink isHardLink (filePathToNativeOsPath -> path) (filePathToNativeOsPath -> link) = + path `seq` link `seq` ((path, link, isHardLink) :) -- for hardlinks, we just copy + handleHardLinks :: [(OsPath, OsPath, t)] -> IO () handleHardLinks = mapM_ $ \(relPath, relLinkTarget, _) -> let absPath = baseDir relPath -- hard links link targets are always "absolute" paths in @@ -197,6 +206,7 @@ unpackAndCheck secCB baseDir entries = do -- This error handling isn't too fine grained and maybe should be -- platform specific, but this way it might catch erros on unix even on -- FAT32 fuse mounted volumes. + handleSymlinks :: [(OsPath, OsPath, c)] -> IO () handleSymlinks = mapM_ $ \(relPath, relLinkTarget, _) -> let absPath = baseDir relPath -- hard links link targets are always "absolute" paths in @@ -220,10 +230,13 @@ unpackAndCheck secCB baseDir entries = do else throwIO e ) +filePathToNativeOsPath :: FilePath -> OsPath +filePathToNativeOsPath = filePathToOsPath . fromFilePathToNative + -- | Recursively copy the contents of one directory to another path. -- -- This is a rip-off of Cabal library. -copyDirectoryRecursive :: FilePath -> FilePath -> IO () +copyDirectoryRecursive :: OsPath -> OsPath -> IO () copyDirectoryRecursive srcDir destDir = do srcFiles <- getDirectoryContentsRecursive srcDir copyFilesWith copyFile destDir [ (srcDir, f) @@ -231,8 +244,8 @@ copyDirectoryRecursive srcDir destDir = do where -- | Common implementation of 'copyFiles', 'installOrdinaryFiles', -- 'installExecutableFiles' and 'installMaybeExecutableFiles'. - copyFilesWith :: (FilePath -> FilePath -> IO ()) - -> FilePath -> [(FilePath, FilePath)] -> IO () + copyFilesWith :: (OsPath -> OsPath -> IO ()) + -> OsPath -> [(OsPath, OsPath)] -> IO () copyFilesWith doCopy targetDir srcFiles = do -- Create parent directories for everything @@ -251,10 +264,10 @@ copyDirectoryRecursive srcDir destDir = do -- parent directories. The list is generated lazily so is not well defined if -- the source directory structure changes before the list is used. -- - getDirectoryContentsRecursive :: FilePath -> IO [FilePath] - getDirectoryContentsRecursive topdir = recurseDirectories [""] + getDirectoryContentsRecursive :: OsPath -> IO [OsPath] + getDirectoryContentsRecursive topdir = recurseDirectories [mempty] where - recurseDirectories :: [FilePath] -> IO [FilePath] + recurseDirectories :: [OsPath] -> IO [OsPath] recurseDirectories [] = return [] recurseDirectories (dir:dirs) = unsafeInterleaveIO $ do (files, dirs') <- collect [] [] =<< listDirectory (topdir dir) @@ -271,7 +284,7 @@ copyDirectoryRecursive srcDir destDir = do then collect files (dirEntry:dirs') entries else collect (dirEntry:files) dirs' entries -setModTime :: FilePath -> EpochTime -> IO () +setModTime :: OsPath -> EpochTime -> IO () setModTime path t = setModificationTime path (posixSecondsToUTCTime (fromIntegral t)) `Exception.catch` \e -> case ioeGetErrorType e of @@ -281,7 +294,7 @@ setModTime path t = InvalidArgument -> return () _ -> throwIO e -setOwnerPermissions :: FilePath -> Permissions -> IO () +setOwnerPermissions :: OsPath -> Permissions -> IO () setOwnerPermissions path permissions = setPermissions path ownerPermissions where @@ -291,5 +304,5 @@ setOwnerPermissions path permissions = setOwnerReadable (testBit permissions 8) $ setOwnerWritable (testBit permissions 7) $ setOwnerExecutable (testBit permissions 6) $ - setOwnerSearchable (testBit permissions 6) $ + setOwnerSearchable (testBit permissions 6) emptyPermissions diff --git a/cabal.haskell-ci b/cabal.haskell-ci index 5152f94..f5a9a56 100644 --- a/cabal.haskell-ci +++ b/cabal.haskell-ci @@ -1,2 +1,2 @@ -installed: -directory -unix -bytestring +installed: -directory -unix -bytestring -filepath haddock: >= 8.6 diff --git a/tar.cabal b/tar.cabal index ebaf4fe..a922f11 100644 --- a/tar.cabal +++ b/tar.cabal @@ -54,7 +54,8 @@ library tar-internal containers >= 0.2 && < 0.8, deepseq >= 1.1 && < 1.6, directory >= 1.3.1 && < 1.4, - filepath < 1.6, + file-io < 0.2, + filepath >= 1.4.100 && < 1.6, os-string >= 2.0 && < 2.1, time < 1.15, transformers < 0.7,