{-# LANGUAGE CPP #-}
{-# LANGUAGE ForeignFunctionInterface #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ViewPatterns #-}
module UnliftIO.IO.File.Posix
( withBinaryFileDurable
, withBinaryFileDurableAtomic
, withBinaryFileAtomic
, ensureFileDurable
)
where
#if __GLASGOW_HASKELL__ < 710
import Control.Applicative
#endif
import Control.Monad (forM_, guard, unless, void, when)
import Control.Monad.IO.Unlift
import Data.Bits (Bits, (.|.))
import Data.ByteString (ByteString)
import Data.Maybe (fromMaybe)
import Data.Typeable (cast)
import Foreign (allocaBytes)
import Foreign.C (CInt(..), throwErrnoIfMinus1, throwErrnoIfMinus1Retry,
throwErrnoIfMinus1Retry_)
import GHC.IO.Device (IODeviceType(RegularFile))
import qualified GHC.IO.Device as Device
import GHC.IO.Exception (IOErrorType(UnsupportedOperation))
import qualified GHC.IO.FD as FD
import qualified GHC.IO.Handle.FD as HandleFD
import qualified GHC.IO.Handle.Types as HandleFD (Handle(..), Handle__(..))
import System.Directory (removeFile)
import System.FilePath (takeDirectory, takeFileName)
import System.IO (Handle, IOMode(..), SeekMode(..), hGetBuf, hPutBuf,
openBinaryTempFile)
import System.IO.Error (ioeGetErrorType, isAlreadyExistsError,
isDoesNotExistError)
import qualified System.Posix.Files as Posix
import System.Posix.Internals (CFilePath, c_close, c_safe_open, withFilePath)
import System.Posix.Types (CMode(..), Fd(..), FileMode)
import UnliftIO.Exception
import UnliftIO.IO
import UnliftIO.MVar
newtype CFlag =
CFlag CInt
deriving (CFlag -> CFlag -> Bool
(CFlag -> CFlag -> Bool) -> (CFlag -> CFlag -> Bool) -> Eq CFlag
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: CFlag -> CFlag -> Bool
== :: CFlag -> CFlag -> Bool
$c/= :: CFlag -> CFlag -> Bool
/= :: CFlag -> CFlag -> Bool
Eq, Int -> CFlag -> ShowS
[CFlag] -> ShowS
CFlag -> [Char]
(Int -> CFlag -> ShowS)
-> (CFlag -> [Char]) -> ([CFlag] -> ShowS) -> Show CFlag
forall a.
(Int -> a -> ShowS) -> (a -> [Char]) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> CFlag -> ShowS
showsPrec :: Int -> CFlag -> ShowS
$cshow :: CFlag -> [Char]
show :: CFlag -> [Char]
$cshowList :: [CFlag] -> ShowS
showList :: [CFlag] -> ShowS
Show, Eq CFlag
CFlag
Eq CFlag =>
(CFlag -> CFlag -> CFlag)
-> (CFlag -> CFlag -> CFlag)
-> (CFlag -> CFlag -> CFlag)
-> (CFlag -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> CFlag
-> (Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> Bool)
-> (CFlag -> Maybe Int)
-> (CFlag -> Int)
-> (CFlag -> Bool)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int -> CFlag)
-> (CFlag -> Int)
-> Bits CFlag
Int -> CFlag
CFlag -> Bool
CFlag -> Int
CFlag -> Maybe Int
CFlag -> CFlag
CFlag -> Int -> Bool
CFlag -> Int -> CFlag
CFlag -> CFlag -> CFlag
forall a.
Eq a =>
(a -> a -> a)
-> (a -> a -> a)
-> (a -> a -> a)
-> (a -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> a
-> (Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> Bool)
-> (a -> Maybe Int)
-> (a -> Int)
-> (a -> Bool)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int)
-> Bits a
$c.&. :: CFlag -> CFlag -> CFlag
.&. :: CFlag -> CFlag -> CFlag
$c.|. :: CFlag -> CFlag -> CFlag
.|. :: CFlag -> CFlag -> CFlag
$cxor :: CFlag -> CFlag -> CFlag
xor :: CFlag -> CFlag -> CFlag
$ccomplement :: CFlag -> CFlag
complement :: CFlag -> CFlag
$cshift :: CFlag -> Int -> CFlag
shift :: CFlag -> Int -> CFlag
$crotate :: CFlag -> Int -> CFlag
rotate :: CFlag -> Int -> CFlag
$czeroBits :: CFlag
zeroBits :: CFlag
$cbit :: Int -> CFlag
bit :: Int -> CFlag
$csetBit :: CFlag -> Int -> CFlag
setBit :: CFlag -> Int -> CFlag
$cclearBit :: CFlag -> Int -> CFlag
clearBit :: CFlag -> Int -> CFlag
$ccomplementBit :: CFlag -> Int -> CFlag
complementBit :: CFlag -> Int -> CFlag
$ctestBit :: CFlag -> Int -> Bool
testBit :: CFlag -> Int -> Bool
$cbitSizeMaybe :: CFlag -> Maybe Int
bitSizeMaybe :: CFlag -> Maybe Int
$cbitSize :: CFlag -> Int
bitSize :: CFlag -> Int
$cisSigned :: CFlag -> Bool
isSigned :: CFlag -> Bool
$cshiftL :: CFlag -> Int -> CFlag
shiftL :: CFlag -> Int -> CFlag
$cunsafeShiftL :: CFlag -> Int -> CFlag
unsafeShiftL :: CFlag -> Int -> CFlag
$cshiftR :: CFlag -> Int -> CFlag
shiftR :: CFlag -> Int -> CFlag
$cunsafeShiftR :: CFlag -> Int -> CFlag
unsafeShiftR :: CFlag -> Int -> CFlag
$crotateL :: CFlag -> Int -> CFlag
rotateL :: CFlag -> Int -> CFlag
$crotateR :: CFlag -> Int -> CFlag
rotateR :: CFlag -> Int -> CFlag
$cpopCount :: CFlag -> Int
popCount :: CFlag -> Int
Bits)
foreign import ccall unsafe "HsBase.h __hscore_o_rdonly" o_RDONLY :: CFlag
foreign import ccall unsafe "HsBase.h __hscore_o_wronly" o_WRONLY :: CFlag
foreign import ccall unsafe "HsBase.h __hscore_o_rdwr" o_RDWR :: CFlag
foreign import ccall unsafe "HsBase.h __hscore_o_append" o_APPEND :: CFlag
foreign import ccall unsafe "HsBase.h __hscore_o_creat" o_CREAT :: CFlag
foreign import ccall unsafe "HsBase.h __hscore_o_noctty" o_NOCTTY :: CFlag
foreign import ccall unsafe "file-posix.c unliftio_o_tmpfile" o_TMPFILE :: CFlag
o_TMPFILE_not_supported :: CFlag
o_TMPFILE_not_supported :: CFlag
o_TMPFILE_not_supported = CInt -> CFlag
CFlag CInt
0
newtype CAt = CAt
{ CAt -> CInt
unCAt :: CInt
} deriving (CAt -> CAt -> Bool
(CAt -> CAt -> Bool) -> (CAt -> CAt -> Bool) -> Eq CAt
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: CAt -> CAt -> Bool
== :: CAt -> CAt -> Bool
$c/= :: CAt -> CAt -> Bool
/= :: CAt -> CAt -> Bool
Eq, Int -> CAt -> ShowS
[CAt] -> ShowS
CAt -> [Char]
(Int -> CAt -> ShowS)
-> (CAt -> [Char]) -> ([CAt] -> ShowS) -> Show CAt
forall a.
(Int -> a -> ShowS) -> (a -> [Char]) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> CAt -> ShowS
showsPrec :: Int -> CAt -> ShowS
$cshow :: CAt -> [Char]
show :: CAt -> [Char]
$cshowList :: [CAt] -> ShowS
showList :: [CAt] -> ShowS
Show, Eq CAt
CAt
Eq CAt =>
(CAt -> CAt -> CAt)
-> (CAt -> CAt -> CAt)
-> (CAt -> CAt -> CAt)
-> (CAt -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> CAt)
-> CAt
-> (Int -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> Bool)
-> (CAt -> Maybe Int)
-> (CAt -> Int)
-> (CAt -> Bool)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int -> CAt)
-> (CAt -> Int)
-> Bits CAt
Int -> CAt
CAt -> Bool
CAt -> Int
CAt -> Maybe Int
CAt -> CAt
CAt -> Int -> Bool
CAt -> Int -> CAt
CAt -> CAt -> CAt
forall a.
Eq a =>
(a -> a -> a)
-> (a -> a -> a)
-> (a -> a -> a)
-> (a -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> a
-> (Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> Bool)
-> (a -> Maybe Int)
-> (a -> Int)
-> (a -> Bool)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int -> a)
-> (a -> Int)
-> Bits a
$c.&. :: CAt -> CAt -> CAt
.&. :: CAt -> CAt -> CAt
$c.|. :: CAt -> CAt -> CAt
.|. :: CAt -> CAt -> CAt
$cxor :: CAt -> CAt -> CAt
xor :: CAt -> CAt -> CAt
$ccomplement :: CAt -> CAt
complement :: CAt -> CAt
$cshift :: CAt -> Int -> CAt
shift :: CAt -> Int -> CAt
$crotate :: CAt -> Int -> CAt
rotate :: CAt -> Int -> CAt
$czeroBits :: CAt
zeroBits :: CAt
$cbit :: Int -> CAt
bit :: Int -> CAt
$csetBit :: CAt -> Int -> CAt
setBit :: CAt -> Int -> CAt
$cclearBit :: CAt -> Int -> CAt
clearBit :: CAt -> Int -> CAt
$ccomplementBit :: CAt -> Int -> CAt
complementBit :: CAt -> Int -> CAt
$ctestBit :: CAt -> Int -> Bool
testBit :: CAt -> Int -> Bool
$cbitSizeMaybe :: CAt -> Maybe Int
bitSizeMaybe :: CAt -> Maybe Int
$cbitSize :: CAt -> Int
bitSize :: CAt -> Int
$cisSigned :: CAt -> Bool
isSigned :: CAt -> Bool
$cshiftL :: CAt -> Int -> CAt
shiftL :: CAt -> Int -> CAt
$cunsafeShiftL :: CAt -> Int -> CAt
unsafeShiftL :: CAt -> Int -> CAt
$cshiftR :: CAt -> Int -> CAt
shiftR :: CAt -> Int -> CAt
$cunsafeShiftR :: CAt -> Int -> CAt
unsafeShiftR :: CAt -> Int -> CAt
$crotateL :: CAt -> Int -> CAt
rotateL :: CAt -> Int -> CAt
$crotateR :: CAt -> Int -> CAt
rotateR :: CAt -> Int -> CAt
$cpopCount :: CAt -> Int
popCount :: CAt -> Int
Bits)
foreign import ccall unsafe "file-posix.c unliftio_at_fdcwd" at_FDCWD :: CAt
foreign import ccall unsafe "file-posix.c unliftio_at_symlink_follow" at_SYMLINK_FOLLOW :: CAt
foreign import ccall unsafe "file-posix.c unliftio_s_irusr" s_IRUSR :: CMode
foreign import ccall unsafe "file-posix.c unliftio_s_iwusr" s_IWUSR :: CMode
c_open :: CFilePath -> CFlag -> CMode -> IO CInt
c_open :: CFilePath -> CFlag -> CMode -> IO CInt
c_open CFilePath
fp (CFlag CInt
flags) = CFilePath -> CInt -> CMode -> IO CInt
c_safe_open CFilePath
fp CInt
flags
foreign import ccall safe "fcntl.h openat"
c_safe_openat :: CInt -> CFilePath -> CInt -> CMode -> IO CInt
c_openat :: DirFd -> CFilePath -> CFlag -> CMode -> IO CInt
c_openat :: DirFd -> CFilePath -> CFlag -> CMode -> IO CInt
c_openat (DirFd (Fd CInt
fd)) CFilePath
fp (CFlag CInt
flags) = CInt -> CFilePath -> CInt -> CMode -> IO CInt
c_safe_openat CInt
fd CFilePath
fp CInt
flags
foreign import ccall safe "fcntl.h renameat"
c_safe_renameat :: CInt -> CFilePath -> CInt -> CFilePath -> IO CInt
c_renameat :: DirFd -> CFilePath -> DirFd -> CFilePath -> IO CInt
c_renameat :: DirFd -> CFilePath -> DirFd -> CFilePath -> IO CInt
c_renameat (DirFd (Fd CInt
fdFrom)) CFilePath
cFpFrom (DirFd (Fd CInt
fdTo)) CFilePath
cFpTo =
CInt -> CFilePath -> CInt -> CFilePath -> IO CInt
c_safe_renameat CInt
fdFrom CFilePath
cFpFrom CInt
fdTo CFilePath
cFpTo
foreign import ccall safe "unistd.h fsync"
c_safe_fsync :: CInt -> IO CInt
c_fsync :: Fd -> IO CInt
c_fsync :: Fd -> IO CInt
c_fsync (Fd CInt
fd) = CInt -> IO CInt
c_safe_fsync CInt
fd
foreign import ccall safe "unistd.h linkat"
c_safe_linkat :: CInt -> CFilePath -> CInt -> CFilePath -> CInt -> IO CInt
c_linkat :: CAt -> CFilePath -> Either DirFd CAt -> CFilePath -> CAt -> IO CInt
c_linkat :: CAt -> CFilePath -> Either DirFd CAt -> CFilePath -> CAt -> IO CInt
c_linkat CAt
cat CFilePath
oldPath Either DirFd CAt
eNewDir CFilePath
newPath (CAt CInt
flags) =
CInt -> CFilePath -> CInt -> CFilePath -> CInt -> IO CInt
c_safe_linkat (CAt -> CInt
unCAt CAt
cat) CFilePath
oldPath CInt
newDir CFilePath
newPath CInt
flags
where
unFd :: Fd -> CInt
unFd (Fd CInt
fd) = CInt
fd
newDir :: CInt
newDir = (DirFd -> CInt) -> (CAt -> CInt) -> Either DirFd CAt -> CInt
forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either (Fd -> CInt
unFd (Fd -> CInt) -> (DirFd -> Fd) -> DirFd -> CInt
forall b c a. (b -> c) -> (a -> b) -> a -> c
. DirFd -> Fd
unDirFd) CAt -> CInt
unCAt Either DirFd CAt
eNewDir
std_flags, output_flags, read_flags, write_flags, rw_flags,
append_flags :: CFlag
std_flags :: CFlag
std_flags = CFlag
o_NOCTTY
output_flags :: CFlag
output_flags = CFlag
std_flags CFlag -> CFlag -> CFlag
forall a. Bits a => a -> a -> a
.|. CFlag
o_CREAT
read_flags :: CFlag
read_flags = CFlag
std_flags CFlag -> CFlag -> CFlag
forall a. Bits a => a -> a -> a
.|. CFlag
o_RDONLY
write_flags :: CFlag
write_flags = CFlag
output_flags CFlag -> CFlag -> CFlag
forall a. Bits a => a -> a -> a
.|. CFlag
o_WRONLY
rw_flags :: CFlag
rw_flags = CFlag
output_flags CFlag -> CFlag -> CFlag
forall a. Bits a => a -> a -> a
.|. CFlag
o_RDWR
append_flags :: CFlag
append_flags = CFlag
write_flags CFlag -> CFlag -> CFlag
forall a. Bits a => a -> a -> a
.|. CFlag
o_APPEND
ioModeToFlags :: IOMode -> CFlag
ioModeToFlags :: IOMode -> CFlag
ioModeToFlags IOMode
iomode =
case IOMode
iomode of
IOMode
ReadMode -> CFlag
read_flags
IOMode
WriteMode -> CFlag
write_flags
IOMode
ReadWriteMode -> CFlag
rw_flags
IOMode
AppendMode -> CFlag
append_flags
newtype DirFd = DirFd
{ DirFd -> Fd
unDirFd :: Fd
}
openDir :: MonadIO m => FilePath -> m Fd
openDir :: forall (m :: * -> *). MonadIO m => [Char] -> m Fd
openDir [Char]
fp
=
IO Fd -> m Fd
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO Fd -> m Fd) -> IO Fd -> m Fd
forall a b. (a -> b) -> a -> b
$
[Char] -> (CFilePath -> IO Fd) -> IO Fd
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath [Char]
fp ((CFilePath -> IO Fd) -> IO Fd) -> (CFilePath -> IO Fd) -> IO Fd
forall a b. (a -> b) -> a -> b
$ \CFilePath
cFp ->
CInt -> Fd
Fd (CInt -> Fd) -> IO CInt -> IO Fd
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$>
[Char] -> IO CInt -> IO CInt
forall a. (Eq a, Num a) => [Char] -> IO a -> IO a
throwErrnoIfMinus1Retry
[Char]
"openDir"
(CFilePath -> CFlag -> CMode -> IO CInt
c_open CFilePath
cFp (IOMode -> CFlag
ioModeToFlags IOMode
ReadMode) CMode
0o660)
closeDirectory :: MonadIO m => DirFd -> m ()
closeDirectory :: forall (m :: * -> *). MonadIO m => DirFd -> m ()
closeDirectory (DirFd (Fd CInt
dirFd)) =
IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$
[Char] -> IO CInt -> IO ()
forall a. (Eq a, Num a) => [Char] -> IO a -> IO ()
throwErrnoIfMinus1Retry_ [Char]
"closeDirectory" (IO CInt -> IO ()) -> IO CInt -> IO ()
forall a b. (a -> b) -> a -> b
$ CInt -> IO CInt
c_close CInt
dirFd
fsyncFileDescriptor
:: MonadIO m
=> String
-> Fd
-> m ()
fsyncFileDescriptor :: forall (m :: * -> *). MonadIO m => [Char] -> Fd -> m ()
fsyncFileDescriptor [Char]
name Fd
fd =
IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$ IO CInt -> IO ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (IO CInt -> IO ()) -> IO CInt -> IO ()
forall a b. (a -> b) -> a -> b
$ [Char] -> IO CInt -> IO CInt
forall a. (Eq a, Num a) => [Char] -> IO a -> IO a
throwErrnoIfMinus1 ([Char]
"fsync - " [Char] -> ShowS
forall a. [a] -> [a] -> [a]
++ [Char]
name) (IO CInt -> IO CInt) -> IO CInt -> IO CInt
forall a b. (a -> b) -> a -> b
$ Fd -> IO CInt
c_fsync Fd
fd
fsyncFileHandle :: String -> Handle -> IO ()
fsyncFileHandle :: [Char] -> Handle -> IO ()
fsyncFileHandle [Char]
fname Handle
hdl = Handle -> (Fd -> IO ()) -> IO ()
forall a. Handle -> (Fd -> IO a) -> IO a
withHandleFd Handle
hdl ([Char] -> Fd -> IO ()
forall (m :: * -> *). MonadIO m => [Char] -> Fd -> m ()
fsyncFileDescriptor ([Char]
fname [Char] -> ShowS
forall a. [a] -> [a] -> [a]
++ [Char]
"/File"))
fsyncDirectoryFd :: String -> DirFd -> IO ()
fsyncDirectoryFd :: [Char] -> DirFd -> IO ()
fsyncDirectoryFd [Char]
fname = [Char] -> Fd -> IO ()
forall (m :: * -> *). MonadIO m => [Char] -> Fd -> m ()
fsyncFileDescriptor ([Char]
fname [Char] -> ShowS
forall a. [a] -> [a] -> [a]
++ [Char]
"/Directory") (Fd -> IO ()) -> (DirFd -> Fd) -> DirFd -> IO ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. DirFd -> Fd
unDirFd
openFileFromDir :: MonadIO m => DirFd -> FilePath -> IOMode -> m Handle
openFileFromDir :: forall (m :: * -> *).
MonadIO m =>
DirFd -> [Char] -> IOMode -> m Handle
openFileFromDir DirFd
dirFd filePath :: [Char]
filePath@(ShowS
takeFileName -> [Char]
fileName) IOMode
iomode =
IO Handle -> m Handle
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO Handle -> m Handle) -> IO Handle -> m Handle
forall a b. (a -> b) -> a -> b
$
[Char] -> (CFilePath -> IO Handle) -> IO Handle
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath [Char]
fileName ((CFilePath -> IO Handle) -> IO Handle)
-> (CFilePath -> IO Handle) -> IO Handle
forall a b. (a -> b) -> a -> b
$ \CFilePath
cFileName ->
IO (FD, IODeviceType)
-> ((FD, IODeviceType) -> IO ())
-> ((FD, IODeviceType) -> IO Handle)
-> IO Handle
forall (m :: * -> *) a b c.
MonadUnliftIO m =>
m a -> (a -> m b) -> (a -> m c) -> m c
bracketOnError
(do CInt
fileFd <-
[Char] -> IO CInt -> IO CInt
forall a. (Eq a, Num a) => [Char] -> IO a -> IO a
throwErrnoIfMinus1Retry [Char]
"openFileFromDir" (IO CInt -> IO CInt) -> IO CInt -> IO CInt
forall a b. (a -> b) -> a -> b
$
DirFd -> CFilePath -> CFlag -> CMode -> IO CInt
c_openat DirFd
dirFd CFilePath
cFileName (IOMode -> CFlag
ioModeToFlags IOMode
iomode) CMode
0o666
CInt
-> IOMode
-> Maybe (IODeviceType, CDev, CIno)
-> Bool
-> Bool
-> IO (FD, IODeviceType)
FD.mkFD
CInt
fileFd
IOMode
iomode
Maybe (IODeviceType, CDev, CIno)
forall a. Maybe a
Nothing
Bool
False
Bool
False
IO (FD, IODeviceType) -> IO CInt -> IO (FD, IODeviceType)
forall (m :: * -> *) a b. MonadUnliftIO m => m a -> m b -> m a
`onException`
CInt -> IO CInt
c_close CInt
fileFd)
(IO () -> IO ()
forall a. IO a -> IO a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> IO ())
-> ((FD, IODeviceType) -> IO ()) -> (FD, IODeviceType) -> IO ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. FD -> IO ()
forall a. IODevice a => a -> IO ()
Device.close (FD -> IO ())
-> ((FD, IODeviceType) -> FD) -> (FD, IODeviceType) -> IO ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (FD, IODeviceType) -> FD
forall a b. (a, b) -> a
fst)
(\(FD
fD, IODeviceType
fd_type)
-> do
Bool -> IO () -> IO ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (IOMode
iomode IOMode -> IOMode -> Bool
forall a. Eq a => a -> a -> Bool
== IOMode
WriteMode Bool -> Bool -> Bool
&& IODeviceType
fd_type IODeviceType -> IODeviceType -> Bool
forall a. Eq a => a -> a -> Bool
== IODeviceType
RegularFile) (IO () -> IO ()) -> IO () -> IO ()
forall a b. (a -> b) -> a -> b
$
FD -> Integer -> IO ()
forall a. IODevice a => a -> Integer -> IO ()
Device.setSize FD
fD Integer
0
FD
-> IODeviceType
-> [Char]
-> IOMode
-> Bool
-> Maybe TextEncoding
-> IO Handle
HandleFD.mkHandleFromFD FD
fD IODeviceType
fd_type [Char]
filePath IOMode
iomode Bool
False Maybe TextEncoding
forall a. Maybe a
Nothing)
openAnonymousTempFileFromDir ::
MonadIO m =>
Maybe DirFd
-> FilePath
-> IOMode
-> m Handle
openAnonymousTempFileFromDir :: forall (m :: * -> *).
MonadIO m =>
Maybe DirFd -> [Char] -> IOMode -> m Handle
openAnonymousTempFileFromDir Maybe DirFd
mDirFd [Char]
filePath IOMode
iomode =
IO Handle -> m Handle
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO Handle -> m Handle) -> IO Handle -> m Handle
forall a b. (a -> b) -> a -> b
$
case Maybe DirFd
mDirFd of
Just DirFd
dirFd -> [Char] -> (CFilePath -> IO Handle) -> IO Handle
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath [Char]
"." ((CFlag -> CMode -> IO CInt) -> IO Handle
openAnonymousWith ((CFlag -> CMode -> IO CInt) -> IO Handle)
-> (CFilePath -> CFlag -> CMode -> IO CInt)
-> CFilePath
-> IO Handle
forall b c a. (b -> c) -> (a -> b) -> a -> c
. DirFd -> CFilePath -> CFlag -> CMode -> IO CInt
c_openat DirFd
dirFd)
Maybe DirFd
Nothing ->
[Char] -> (CFilePath -> IO Handle) -> IO Handle
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath (ShowS
takeDirectory [Char]
filePath) ((CFlag -> CMode -> IO CInt) -> IO Handle
openAnonymousWith ((CFlag -> CMode -> IO CInt) -> IO Handle)
-> (CFilePath -> CFlag -> CMode -> IO CInt)
-> CFilePath
-> IO Handle
forall b c a. (b -> c) -> (a -> b) -> a -> c
. CFilePath -> CFlag -> CMode -> IO CInt
c_open)
where
fdName :: [Char]
fdName = [Char]
"openAnonymousTempFileFromDir - " [Char] -> ShowS
forall a. [a] -> [a] -> [a]
++ [Char]
filePath
ioModeToTmpFlags :: IOMode -> CFlag
ioModeToTmpFlags :: IOMode -> CFlag
ioModeToTmpFlags =
\case
IOMode
ReadMode -> CFlag
o_RDWR
IOMode
ReadWriteMode -> CFlag
o_RDWR
IOMode
_ -> CFlag
o_WRONLY
openAnonymousWith :: (CFlag -> CMode -> IO CInt) -> IO Handle
openAnonymousWith CFlag -> CMode -> IO CInt
fopen =
IO (FD, IODeviceType)
-> ((FD, IODeviceType) -> IO ())
-> ((FD, IODeviceType) -> IO Handle)
-> IO Handle
forall (m :: * -> *) a b c.
MonadUnliftIO m =>
m a -> (a -> m b) -> (a -> m c) -> m c
bracketOnError
(do CInt
fileFd <-
[Char] -> IO CInt -> IO CInt
forall a. (Eq a, Num a) => [Char] -> IO a -> IO a
throwErrnoIfMinus1Retry [Char]
"openAnonymousTempFileFromDir" (IO CInt -> IO CInt) -> IO CInt -> IO CInt
forall a b. (a -> b) -> a -> b
$
CFlag -> CMode -> IO CInt
fopen (CFlag
o_TMPFILE CFlag -> CFlag -> CFlag
forall a. Bits a => a -> a -> a
.|. IOMode -> CFlag
ioModeToTmpFlags IOMode
iomode) (CMode
s_IRUSR CMode -> CMode -> CMode
forall a. Bits a => a -> a -> a
.|. CMode
s_IWUSR)
CInt
-> IOMode
-> Maybe (IODeviceType, CDev, CIno)
-> Bool
-> Bool
-> IO (FD, IODeviceType)
FD.mkFD
CInt
fileFd
IOMode
iomode
Maybe (IODeviceType, CDev, CIno)
forall a. Maybe a
Nothing
Bool
False
Bool
False
IO (FD, IODeviceType) -> IO CInt -> IO (FD, IODeviceType)
forall (m :: * -> *) a b. MonadUnliftIO m => m a -> m b -> m a
`onException`
CInt -> IO CInt
c_close CInt
fileFd)
(IO () -> IO ()
forall a. IO a -> IO a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> IO ())
-> ((FD, IODeviceType) -> IO ()) -> (FD, IODeviceType) -> IO ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. FD -> IO ()
forall a. IODevice a => a -> IO ()
Device.close (FD -> IO ())
-> ((FD, IODeviceType) -> FD) -> (FD, IODeviceType) -> IO ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (FD, IODeviceType) -> FD
forall a b. (a, b) -> a
fst)
(\(FD
fD, IODeviceType
fd_type) ->
FD
-> IODeviceType
-> [Char]
-> IOMode
-> Bool
-> Maybe TextEncoding
-> IO Handle
HandleFD.mkHandleFromFD FD
fD IODeviceType
fd_type [Char]
fdName IOMode
iomode Bool
False Maybe TextEncoding
forall a. Maybe a
Nothing)
atomicDurableTempFileRename ::
DirFd -> Maybe FileMode -> Handle -> Maybe FilePath -> FilePath -> IO ()
atomicDurableTempFileRename :: DirFd -> Maybe CMode -> Handle -> Maybe [Char] -> [Char] -> IO ()
atomicDurableTempFileRename DirFd
dirFd Maybe CMode
mFileMode Handle
tmpFileHandle Maybe [Char]
mTmpFilePath [Char]
filePath = do
[Char] -> Handle -> IO ()
fsyncFileHandle [Char]
"atomicDurableTempFileCreate" Handle
tmpFileHandle
let eTmpFile :: Either Handle [Char]
eTmpFile = Either Handle [Char]
-> ([Char] -> Either Handle [Char])
-> Maybe [Char]
-> Either Handle [Char]
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (Handle -> Either Handle [Char]
forall a b. a -> Either a b
Left Handle
tmpFileHandle) [Char] -> Either Handle [Char]
forall a b. b -> Either a b
Right Maybe [Char]
mTmpFilePath
Maybe DirFd
-> Maybe CMode -> Either Handle [Char] -> [Char] -> IO ()
atomicTempFileRename (DirFd -> Maybe DirFd
forall a. a -> Maybe a
Just DirFd
dirFd) Maybe CMode
mFileMode Either Handle [Char]
eTmpFile [Char]
filePath
Handle -> IO ()
forall (m :: * -> *). MonadIO m => Handle -> m ()
hClose Handle
tmpFileHandle
[Char] -> DirFd -> IO ()
fsyncDirectoryFd [Char]
"atomicDurableTempFileCreate" DirFd
dirFd
atomicTempFileCreate ::
Maybe DirFd
-> Maybe FileMode
-> Handle
-> FilePath
-> IO ()
atomicTempFileCreate :: Maybe DirFd -> Maybe CMode -> Handle -> [Char] -> IO ()
atomicTempFileCreate Maybe DirFd
mDirFd Maybe CMode
mFileMode Handle
tmpFileHandle [Char]
filePath =
Handle -> (Fd -> IO ()) -> IO ()
forall a. Handle -> (Fd -> IO a) -> IO a
withHandleFd Handle
tmpFileHandle ((Fd -> IO ()) -> IO ()) -> (Fd -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \fd :: Fd
fd@(Fd CInt
cFd) ->
[Char] -> (CFilePath -> IO ()) -> IO ()
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath ([Char]
"/proc/self/fd/" [Char] -> ShowS
forall a. [a] -> [a] -> [a]
++ CInt -> [Char]
forall a. Show a => a -> [Char]
show CInt
cFd) ((CFilePath -> IO ()) -> IO ()) -> (CFilePath -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \CFilePath
cFromFilePath ->
[Char] -> (CFilePath -> IO ()) -> IO ()
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath [Char]
filePathName ((CFilePath -> IO ()) -> IO ()) -> (CFilePath -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \CFilePath
cToFilePath -> do
let fileMode :: CMode
fileMode = CMode -> Maybe CMode -> CMode
forall a. a -> Maybe a -> a
fromMaybe CMode
Posix.stdFileMode Maybe CMode
mFileMode
Fd -> CMode -> IO ()
Posix.setFdMode Fd
fd CMode
fileMode
let safeLink :: [Char] -> CFilePath -> IO ()
safeLink [Char]
which CFilePath
to =
[Char] -> IO CInt -> IO ()
forall a. (Eq a, Num a) => [Char] -> IO a -> IO ()
throwErrnoIfMinus1Retry_
([Char]
"atomicFileCreate - c_safe_linkat - " [Char] -> ShowS
forall a. [a] -> [a] -> [a]
++ [Char]
which) (IO CInt -> IO ()) -> IO CInt -> IO ()
forall a b. (a -> b) -> a -> b
$
CAt -> CFilePath -> Either DirFd CAt -> CFilePath -> CAt -> IO CInt
c_linkat CAt
at_FDCWD CFilePath
cFromFilePath Either DirFd CAt
cDirFd CFilePath
to CAt
at_SYMLINK_FOLLOW
Either () ()
eExc <-
(IOError -> Maybe ()) -> IO () -> IO (Either () ())
forall (m :: * -> *) e b a.
(MonadUnliftIO m, Exception e) =>
(e -> Maybe b) -> m a -> m (Either b a)
tryJust (Bool -> Maybe ()
forall (f :: * -> *). Alternative f => Bool -> f ()
guard (Bool -> Maybe ()) -> (IOError -> Bool) -> IOError -> Maybe ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. IOError -> Bool
isAlreadyExistsError) (IO () -> IO (Either () ())) -> IO () -> IO (Either () ())
forall a b. (a -> b) -> a -> b
$
[Char] -> CFilePath -> IO ()
safeLink [Char]
"anonymous" CFilePath
cToFilePath
case Either () ()
eExc of
Right () -> () -> IO ()
forall a. a -> IO a
forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
Left () ->
[Char] -> ([Char] -> Handle -> IO ()) -> IO ()
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> ([Char] -> Handle -> m a) -> m a
withBinaryTempFileFor [Char]
filePath (([Char] -> Handle -> IO ()) -> IO ())
-> ([Char] -> Handle -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \[Char]
visTmpFileName Handle
visTmpFileHandle -> do
Handle -> IO ()
forall (m :: * -> *). MonadIO m => Handle -> m ()
hClose Handle
visTmpFileHandle
[Char] -> IO ()
removeFile [Char]
visTmpFileName
case Maybe DirFd
mDirFd of
Maybe DirFd
Nothing -> do
[Char] -> (CFilePath -> IO ()) -> IO ()
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath [Char]
visTmpFileName ([Char] -> CFilePath -> IO ()
safeLink [Char]
"visible")
[Char] -> [Char] -> IO ()
Posix.rename [Char]
visTmpFileName [Char]
filePath
Just DirFd
dirFd ->
[Char] -> (CFilePath -> IO ()) -> IO ()
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath (ShowS
takeFileName [Char]
visTmpFileName) ((CFilePath -> IO ()) -> IO ()) -> (CFilePath -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \CFilePath
cVisTmpFile -> do
[Char] -> CFilePath -> IO ()
safeLink [Char]
"visible" CFilePath
cVisTmpFile
[Char] -> IO CInt -> IO ()
forall a. (Eq a, Num a) => [Char] -> IO a -> IO ()
throwErrnoIfMinus1Retry_
[Char]
"atomicFileCreate - c_safe_renameat" (IO CInt -> IO ()) -> IO CInt -> IO ()
forall a b. (a -> b) -> a -> b
$
DirFd -> CFilePath -> DirFd -> CFilePath -> IO CInt
c_renameat DirFd
dirFd CFilePath
cVisTmpFile DirFd
dirFd CFilePath
cToFilePath
where
(Either DirFd CAt
cDirFd, [Char]
filePathName) =
case Maybe DirFd
mDirFd of
Maybe DirFd
Nothing -> (CAt -> Either DirFd CAt
forall a b. b -> Either a b
Right CAt
at_FDCWD, [Char]
filePath)
Just DirFd
dirFd -> (DirFd -> Either DirFd CAt
forall a b. a -> Either a b
Left DirFd
dirFd, ShowS
takeFileName [Char]
filePath)
atomicTempFileRename ::
Maybe DirFd
-> Maybe FileMode
-> Either Handle FilePath
-> FilePath
-> IO ()
atomicTempFileRename :: Maybe DirFd
-> Maybe CMode -> Either Handle [Char] -> [Char] -> IO ()
atomicTempFileRename Maybe DirFd
mDirFd Maybe CMode
mFileMode Either Handle [Char]
eTmpFile [Char]
filePath =
case Either Handle [Char]
eTmpFile of
Left Handle
tmpFileHandle ->
Maybe DirFd -> Maybe CMode -> Handle -> [Char] -> IO ()
atomicTempFileCreate Maybe DirFd
mDirFd Maybe CMode
mFileMode Handle
tmpFileHandle [Char]
filePath
Right [Char]
tmpFilePath -> do
Maybe CMode -> (CMode -> IO ()) -> IO ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
t a -> (a -> m b) -> m ()
forM_ Maybe CMode
mFileMode ((CMode -> IO ()) -> IO ()) -> (CMode -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \CMode
fileMode -> [Char] -> CMode -> IO ()
Posix.setFileMode [Char]
tmpFilePath CMode
fileMode
case Maybe DirFd
mDirFd of
Maybe DirFd
Nothing -> [Char] -> [Char] -> IO ()
Posix.rename [Char]
tmpFilePath [Char]
filePath
Just DirFd
dirFd ->
[Char] -> (CFilePath -> IO ()) -> IO ()
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath (ShowS
takeFileName [Char]
filePath) ((CFilePath -> IO ()) -> IO ()) -> (CFilePath -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \CFilePath
cToFilePath ->
[Char] -> (CFilePath -> IO ()) -> IO ()
forall a. [Char] -> (CFilePath -> IO a) -> IO a
withFilePath (ShowS
takeFileName [Char]
tmpFilePath) ((CFilePath -> IO ()) -> IO ()) -> (CFilePath -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \CFilePath
cTmpFilePath ->
[Char] -> IO CInt -> IO ()
forall a. (Eq a, Num a) => [Char] -> IO a -> IO ()
throwErrnoIfMinus1Retry_ [Char]
"atomicFileCreate - c_safe_renameat" (IO CInt -> IO ()) -> IO CInt -> IO ()
forall a b. (a -> b) -> a -> b
$
DirFd -> CFilePath -> DirFd -> CFilePath -> IO CInt
c_renameat DirFd
dirFd CFilePath
cTmpFilePath DirFd
dirFd CFilePath
cToFilePath
withDirectory :: MonadUnliftIO m => FilePath -> (DirFd -> m a) -> m a
withDirectory :: forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> (DirFd -> m a) -> m a
withDirectory [Char]
dirPath = m DirFd -> (DirFd -> m ()) -> (DirFd -> m a) -> m a
forall (m :: * -> *) a b c.
MonadUnliftIO m =>
m a -> (a -> m b) -> (a -> m c) -> m c
bracket (Fd -> DirFd
DirFd (Fd -> DirFd) -> m Fd -> m DirFd
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [Char] -> m Fd
forall (m :: * -> *). MonadIO m => [Char] -> m Fd
openDir [Char]
dirPath) DirFd -> m ()
forall (m :: * -> *). MonadIO m => DirFd -> m ()
closeDirectory
withFileInDirectory ::
MonadUnliftIO m => DirFd -> FilePath -> IOMode -> (Handle -> m a) -> m a
withFileInDirectory :: forall (m :: * -> *) a.
MonadUnliftIO m =>
DirFd -> [Char] -> IOMode -> (Handle -> m a) -> m a
withFileInDirectory DirFd
dirFd [Char]
filePath IOMode
iomode =
m Handle -> (Handle -> m ()) -> (Handle -> m a) -> m a
forall (m :: * -> *) a b c.
MonadUnliftIO m =>
m a -> (a -> m b) -> (a -> m c) -> m c
bracket (DirFd -> [Char] -> IOMode -> m Handle
forall (m :: * -> *).
MonadIO m =>
DirFd -> [Char] -> IOMode -> m Handle
openFileFromDir DirFd
dirFd [Char]
filePath IOMode
iomode) Handle -> m ()
forall (m :: * -> *). MonadIO m => Handle -> m ()
hClose
withBinaryTempFileFor ::
MonadUnliftIO m
=> FilePath
-> (FilePath -> Handle -> m a)
-> m a
withBinaryTempFileFor :: forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> ([Char] -> Handle -> m a) -> m a
withBinaryTempFileFor [Char]
filePath [Char] -> Handle -> m a
action =
m ([Char], Handle)
-> (([Char], Handle) -> m (Either IOError ()))
-> (([Char], Handle) -> m a)
-> m a
forall (m :: * -> *) a b c.
MonadUnliftIO m =>
m a -> (a -> m b) -> (a -> m c) -> m c
bracketOnError
(IO ([Char], Handle) -> m ([Char], Handle)
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO ([Char] -> [Char] -> IO ([Char], Handle)
openBinaryTempFile [Char]
dirPath [Char]
tmpFileName))
(\([Char]
tmpFilePath, Handle
tmpFileHandle) ->
Handle -> m ()
forall (m :: * -> *). MonadIO m => Handle -> m ()
hClose Handle
tmpFileHandle m () -> m (Either IOError ()) -> m (Either IOError ())
forall a b. m a -> m b -> m b
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> IO (Either IOError ()) -> m (Either IOError ())
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> IO (Either IOError ())
forall (m :: * -> *) a.
MonadUnliftIO m =>
m a -> m (Either IOError a)
tryIO ([Char] -> IO ()
removeFile [Char]
tmpFilePath)))
(([Char] -> Handle -> m a) -> ([Char], Handle) -> m a
forall a b c. (a -> b -> c) -> (a, b) -> c
uncurry [Char] -> Handle -> m a
action)
where
dirPath :: [Char]
dirPath = ShowS
takeDirectory [Char]
filePath
fileName :: [Char]
fileName = ShowS
takeFileName [Char]
filePath
tmpFileName :: [Char]
tmpFileName = [Char]
"." [Char] -> ShowS
forall a. [a] -> [a] -> [a]
++ [Char]
fileName [Char] -> ShowS
forall a. [a] -> [a] -> [a]
++ [Char]
".tmp"
withAnonymousBinaryTempFileFor ::
MonadUnliftIO m
=> Maybe DirFd
-> FilePath
-> IOMode
-> (Handle -> m a)
-> m (Maybe a)
withAnonymousBinaryTempFileFor :: forall (m :: * -> *) a.
MonadUnliftIO m =>
Maybe DirFd -> [Char] -> IOMode -> (Handle -> m a) -> m (Maybe a)
withAnonymousBinaryTempFileFor Maybe DirFd
mDirFd [Char]
filePath IOMode
iomode Handle -> m a
action
| CFlag
o_TMPFILE CFlag -> CFlag -> Bool
forall a. Eq a => a -> a -> Bool
== CFlag
o_TMPFILE_not_supported = Maybe a -> m (Maybe a)
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure Maybe a
forall a. Maybe a
Nothing
| Bool
otherwise =
m a -> m (Maybe a)
forall {m :: * -> *} {a}. MonadUnliftIO m => m a -> m (Maybe a)
trySupported (m a -> m (Maybe a)) -> m a -> m (Maybe a)
forall a b. (a -> b) -> a -> b
$
m Handle -> (Handle -> m ()) -> (Handle -> m a) -> m a
forall (m :: * -> *) a b c.
MonadUnliftIO m =>
m a -> (a -> m b) -> (a -> m c) -> m c
bracket (Maybe DirFd -> [Char] -> IOMode -> m Handle
forall (m :: * -> *).
MonadIO m =>
Maybe DirFd -> [Char] -> IOMode -> m Handle
openAnonymousTempFileFromDir Maybe DirFd
mDirFd [Char]
filePath IOMode
iomode) Handle -> m ()
forall (m :: * -> *). MonadIO m => Handle -> m ()
hClose Handle -> m a
action
where
trySupported :: m a -> m (Maybe a)
trySupported m a
m =
m a -> m (Either IOError a)
forall (m :: * -> *) a.
MonadUnliftIO m =>
m a -> m (Either IOError a)
tryIO m a
m m (Either IOError a)
-> (Either IOError a -> m (Maybe a)) -> m (Maybe a)
forall a b. m a -> (a -> m b) -> m b
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
Right a
res -> Maybe a -> m (Maybe a)
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Maybe a -> m (Maybe a)) -> Maybe a -> m (Maybe a)
forall a b. (a -> b) -> a -> b
$ a -> Maybe a
forall a. a -> Maybe a
Just a
res
Left IOError
exc
| IOError -> IOErrorType
ioeGetErrorType IOError
exc IOErrorType -> IOErrorType -> Bool
forall a. Eq a => a -> a -> Bool
== IOErrorType
UnsupportedOperation -> Maybe a -> m (Maybe a)
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure Maybe a
forall a. Maybe a
Nothing
Left IOError
exc -> IOError -> m (Maybe a)
forall (m :: * -> *) e a. (MonadIO m, Exception e) => e -> m a
throwIO IOError
exc
withNonAnonymousBinaryTempFileFor ::
MonadUnliftIO m
=> Maybe DirFd
-> FilePath
-> IOMode
-> (FilePath -> Handle -> m a)
-> m a
withNonAnonymousBinaryTempFileFor :: forall (m :: * -> *) a.
MonadUnliftIO m =>
Maybe DirFd -> [Char] -> IOMode -> ([Char] -> Handle -> m a) -> m a
withNonAnonymousBinaryTempFileFor Maybe DirFd
mDirFd [Char]
filePath IOMode
iomode [Char] -> Handle -> m a
action =
[Char] -> ([Char] -> Handle -> m a) -> m a
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> ([Char] -> Handle -> m a) -> m a
withBinaryTempFileFor [Char]
filePath (([Char] -> Handle -> m a) -> m a)
-> ([Char] -> Handle -> m a) -> m a
forall a b. (a -> b) -> a -> b
$ \[Char]
tmpFilePath Handle
tmpFileHandle -> do
Handle -> m ()
forall (m :: * -> *). MonadIO m => Handle -> m ()
hClose Handle
tmpFileHandle
case Maybe DirFd
mDirFd of
Maybe DirFd
Nothing -> [Char] -> IOMode -> (Handle -> m a) -> m a
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> IOMode -> (Handle -> m a) -> m a
withBinaryFile [Char]
tmpFilePath IOMode
iomode ([Char] -> Handle -> m a
action [Char]
tmpFilePath)
Just DirFd
dirFd -> DirFd -> [Char] -> IOMode -> (Handle -> m a) -> m a
forall (m :: * -> *) a.
MonadUnliftIO m =>
DirFd -> [Char] -> IOMode -> (Handle -> m a) -> m a
withFileInDirectory DirFd
dirFd [Char]
tmpFilePath IOMode
iomode ([Char] -> Handle -> m a
action [Char]
tmpFilePath)
copyFileHandle ::
MonadUnliftIO f => IOMode -> FilePath -> Handle -> f (Maybe FileMode)
copyFileHandle :: forall (f :: * -> *).
MonadUnliftIO f =>
IOMode -> [Char] -> Handle -> f (Maybe CMode)
copyFileHandle IOMode
iomode [Char]
fromFilePath Handle
toHandle =
(() -> Maybe CMode)
-> (CMode -> Maybe CMode) -> Either () CMode -> Maybe CMode
forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either (Maybe CMode -> () -> Maybe CMode
forall a b. a -> b -> a
const Maybe CMode
forall a. Maybe a
Nothing) CMode -> Maybe CMode
forall a. a -> Maybe a
Just (Either () CMode -> Maybe CMode)
-> f (Either () CMode) -> f (Maybe CMode)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$>
(IOError -> Maybe ()) -> f CMode -> f (Either () CMode)
forall (m :: * -> *) e b a.
(MonadUnliftIO m, Exception e) =>
(e -> Maybe b) -> m a -> m (Either b a)
tryJust
(Bool -> Maybe ()
forall (f :: * -> *). Alternative f => Bool -> f ()
guard (Bool -> Maybe ()) -> (IOError -> Bool) -> IOError -> Maybe ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. IOError -> Bool
isDoesNotExistError)
(do FileStatus
fileStatus <- IO FileStatus -> f FileStatus
forall a. IO a -> f a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO FileStatus -> f FileStatus) -> IO FileStatus -> f FileStatus
forall a b. (a -> b) -> a -> b
$ [Char] -> IO FileStatus
Posix.getFileStatus [Char]
fromFilePath
Bool -> f () -> f ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (IOMode
iomode IOMode -> IOMode -> Bool
forall a. Eq a => a -> a -> Bool
== IOMode
WriteMode) (f () -> f ()) -> f () -> f ()
forall a b. (a -> b) -> a -> b
$ do
[Char] -> IOMode -> (Handle -> f ()) -> f ()
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> IOMode -> (Handle -> m a) -> m a
withBinaryFile [Char]
fromFilePath IOMode
ReadMode (Handle -> Handle -> f ()
forall (m :: * -> *). MonadIO m => Handle -> Handle -> m ()
`copyHandleData` Handle
toHandle)
Bool -> f () -> f ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (IOMode
iomode IOMode -> IOMode -> Bool
forall a. Eq a => a -> a -> Bool
== IOMode
AppendMode) (f () -> f ()) -> f () -> f ()
forall a b. (a -> b) -> a -> b
$ Handle -> SeekMode -> Integer -> f ()
forall (m :: * -> *).
MonadIO m =>
Handle -> SeekMode -> Integer -> m ()
hSeek Handle
toHandle SeekMode
AbsoluteSeek Integer
0
CMode -> f CMode
forall a. a -> f a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (CMode -> f CMode) -> CMode -> f CMode
forall a b. (a -> b) -> a -> b
$ FileStatus -> CMode
Posix.fileMode FileStatus
fileStatus)
copyHandleData :: MonadIO m => Handle -> Handle -> m ()
copyHandleData :: forall (m :: * -> *). MonadIO m => Handle -> Handle -> m ()
copyHandleData Handle
hFrom Handle
hTo = IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$ Int -> (Ptr Any -> IO ()) -> IO ()
forall a b. Int -> (Ptr a -> IO b) -> IO b
allocaBytes Int
bufferSize Ptr Any -> IO ()
forall {a}. Ptr a -> IO ()
go
where
bufferSize :: Int
bufferSize = Int
131072
go :: Ptr a -> IO ()
go Ptr a
buffer = do
Int
count <- Handle -> Ptr a -> Int -> IO Int
forall a. Handle -> Ptr a -> Int -> IO Int
hGetBuf Handle
hFrom Ptr a
buffer Int
bufferSize
Bool -> IO () -> IO ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Int
count Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
0) (IO () -> IO ()) -> IO () -> IO ()
forall a b. (a -> b) -> a -> b
$ do
Handle -> Ptr a -> Int -> IO ()
forall a. Handle -> Ptr a -> Int -> IO ()
hPutBuf Handle
hTo Ptr a
buffer Int
count
Ptr a -> IO ()
go Ptr a
buffer
withHandleFd :: Handle -> (Fd -> IO a) -> IO a
withHandleFd :: forall a. Handle -> (Fd -> IO a) -> IO a
withHandleFd Handle
h Fd -> IO a
cb =
case Handle
h of
HandleFD.FileHandle [Char]
_ MVar Handle__
mv ->
MVar Handle__ -> (Handle__ -> IO a) -> IO a
forall (m :: * -> *) a b.
MonadUnliftIO m =>
MVar a -> (a -> m b) -> m b
withMVar MVar Handle__
mv ((Handle__ -> IO a) -> IO a) -> (Handle__ -> IO a) -> IO a
forall a b. (a -> b) -> a -> b
$ \HandleFD.Handle__{haDevice :: ()
HandleFD.haDevice = dev
dev} ->
case dev -> Maybe FD
forall a b. (Typeable a, Typeable b) => a -> Maybe b
cast dev
dev of
Just FD
fd -> Fd -> IO a
cb (Fd -> IO a) -> Fd -> IO a
forall a b. (a -> b) -> a -> b
$ CInt -> Fd
Fd (CInt -> Fd) -> CInt -> Fd
forall a b. (a -> b) -> a -> b
$ FD -> CInt
FD.fdFD FD
fd
Maybe FD
Nothing -> [Char] -> IO a
forall a. HasCallStack => [Char] -> a
error [Char]
"withHandleFd: not a file handle"
HandleFD.DuplexHandle {} -> [Char] -> IO a
forall a. HasCallStack => [Char] -> a
error [Char]
"withHandleFd: not a file handle"
ensureFileDurable :: MonadIO m => FilePath -> m ()
ensureFileDurable :: forall (m :: * -> *). MonadIO m => [Char] -> m ()
ensureFileDurable [Char]
filePath =
IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$
[Char] -> (DirFd -> IO ()) -> IO ()
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> (DirFd -> m a) -> m a
withDirectory (ShowS
takeDirectory [Char]
filePath) ((DirFd -> IO ()) -> IO ()) -> (DirFd -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \DirFd
dirFd ->
DirFd -> [Char] -> IOMode -> (Handle -> IO ()) -> IO ()
forall (m :: * -> *) a.
MonadUnliftIO m =>
DirFd -> [Char] -> IOMode -> (Handle -> m a) -> m a
withFileInDirectory DirFd
dirFd [Char]
filePath IOMode
ReadMode ((Handle -> IO ()) -> IO ()) -> (Handle -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \Handle
fileHandle ->
IO () -> IO ()
forall a. IO a -> IO a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> IO ()) -> IO () -> IO ()
forall a b. (a -> b) -> a -> b
$ do
[Char] -> Handle -> IO ()
fsyncFileHandle [Char]
"ensureFileDurablePosix" Handle
fileHandle
[Char] -> DirFd -> IO ()
fsyncDirectoryFd [Char]
"ensureFileDurablePosix" DirFd
dirFd
withBinaryFileDurable ::
MonadUnliftIO m => FilePath -> IOMode -> (Handle -> m r) -> m r
withBinaryFileDurable :: forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> IOMode -> (Handle -> m a) -> m a
withBinaryFileDurable [Char]
filePath IOMode
iomode Handle -> m r
action =
case IOMode
iomode of
IOMode
ReadMode
-> [Char] -> IOMode -> (Handle -> m r) -> m r
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> IOMode -> (Handle -> m a) -> m a
withBinaryFile [Char]
filePath IOMode
iomode Handle -> m r
action
IOMode
_
->
[Char] -> (DirFd -> m r) -> m r
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> (DirFd -> m a) -> m a
withDirectory (ShowS
takeDirectory [Char]
filePath) ((DirFd -> m r) -> m r) -> (DirFd -> m r) -> m r
forall a b. (a -> b) -> a -> b
$ \DirFd
dirFd ->
DirFd -> [Char] -> IOMode -> (Handle -> m r) -> m r
forall (m :: * -> *) a.
MonadUnliftIO m =>
DirFd -> [Char] -> IOMode -> (Handle -> m a) -> m a
withFileInDirectory DirFd
dirFd [Char]
filePath IOMode
iomode ((Handle -> m r) -> m r) -> (Handle -> m r) -> m r
forall a b. (a -> b) -> a -> b
$ \Handle
tmpFileHandle -> do
r
res <- Handle -> m r
action Handle
tmpFileHandle
IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$ do
[Char] -> Handle -> IO ()
fsyncFileHandle [Char]
"withBinaryFileDurablePosix" Handle
tmpFileHandle
[Char] -> DirFd -> IO ()
fsyncDirectoryFd [Char]
"withBinaryFileDurablePosix" DirFd
dirFd
r -> m r
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure r
res
withBinaryFileDurableAtomic ::
MonadUnliftIO m => FilePath -> IOMode -> (Handle -> m r) -> m r
withBinaryFileDurableAtomic :: forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> IOMode -> (Handle -> m a) -> m a
withBinaryFileDurableAtomic [Char]
filePath IOMode
iomode Handle -> m r
action =
case IOMode
iomode of
IOMode
ReadMode
-> [Char] -> IOMode -> (Handle -> m r) -> m r
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> IOMode -> (Handle -> m a) -> m a
withBinaryFile [Char]
filePath IOMode
iomode Handle -> m r
action
IOMode
_
->
[Char] -> (DirFd -> m r) -> m r
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> (DirFd -> m a) -> m a
withDirectory (ShowS
takeDirectory [Char]
filePath) ((DirFd -> m r) -> m r) -> (DirFd -> m r) -> m r
forall a b. (a -> b) -> a -> b
$ \DirFd
dirFd -> do
Maybe r
mRes <- Maybe DirFd -> [Char] -> IOMode -> (Handle -> m r) -> m (Maybe r)
forall (m :: * -> *) a.
MonadUnliftIO m =>
Maybe DirFd -> [Char] -> IOMode -> (Handle -> m a) -> m (Maybe a)
withAnonymousBinaryTempFileFor (DirFd -> Maybe DirFd
forall a. a -> Maybe a
Just DirFd
dirFd) [Char]
filePath IOMode
iomode ((Handle -> m r) -> m (Maybe r)) -> (Handle -> m r) -> m (Maybe r)
forall a b. (a -> b) -> a -> b
$
DirFd -> Maybe [Char] -> Handle -> m r
durableAtomicAction DirFd
dirFd Maybe [Char]
forall a. Maybe a
Nothing
case Maybe r
mRes of
Just r
res -> r -> m r
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure r
res
Maybe r
Nothing ->
Maybe DirFd -> [Char] -> IOMode -> ([Char] -> Handle -> m r) -> m r
forall (m :: * -> *) a.
MonadUnliftIO m =>
Maybe DirFd -> [Char] -> IOMode -> ([Char] -> Handle -> m a) -> m a
withNonAnonymousBinaryTempFileFor (DirFd -> Maybe DirFd
forall a. a -> Maybe a
Just DirFd
dirFd) [Char]
filePath IOMode
iomode (([Char] -> Handle -> m r) -> m r)
-> ([Char] -> Handle -> m r) -> m r
forall a b. (a -> b) -> a -> b
$ \[Char]
tmpFilePath ->
DirFd -> Maybe [Char] -> Handle -> m r
durableAtomicAction DirFd
dirFd ([Char] -> Maybe [Char]
forall a. a -> Maybe a
Just [Char]
tmpFilePath)
where
durableAtomicAction :: DirFd -> Maybe [Char] -> Handle -> m r
durableAtomicAction DirFd
dirFd Maybe [Char]
mTmpFilePath Handle
tmpFileHandle = do
Maybe CMode
mFileMode <- IOMode -> [Char] -> Handle -> m (Maybe CMode)
forall (f :: * -> *).
MonadUnliftIO f =>
IOMode -> [Char] -> Handle -> f (Maybe CMode)
copyFileHandle IOMode
iomode [Char]
filePath Handle
tmpFileHandle
r
res <- Handle -> m r
action Handle
tmpFileHandle
IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$
DirFd -> Maybe CMode -> Handle -> Maybe [Char] -> [Char] -> IO ()
atomicDurableTempFileRename
DirFd
dirFd
Maybe CMode
mFileMode
Handle
tmpFileHandle
Maybe [Char]
mTmpFilePath
[Char]
filePath
r -> m r
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure r
res
withBinaryFileAtomic ::
MonadUnliftIO m => FilePath -> IOMode -> (Handle -> m r) -> m r
withBinaryFileAtomic :: forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> IOMode -> (Handle -> m a) -> m a
withBinaryFileAtomic [Char]
filePath IOMode
iomode Handle -> m r
action =
case IOMode
iomode of
IOMode
ReadMode
-> [Char] -> IOMode -> (Handle -> m r) -> m r
forall (m :: * -> *) a.
MonadUnliftIO m =>
[Char] -> IOMode -> (Handle -> m a) -> m a
withBinaryFile [Char]
filePath IOMode
iomode Handle -> m r
action
IOMode
_
-> do
Maybe r
mRes <-
Maybe DirFd -> [Char] -> IOMode -> (Handle -> m r) -> m (Maybe r)
forall (m :: * -> *) a.
MonadUnliftIO m =>
Maybe DirFd -> [Char] -> IOMode -> (Handle -> m a) -> m (Maybe a)
withAnonymousBinaryTempFileFor Maybe DirFd
forall a. Maybe a
Nothing [Char]
filePath IOMode
iomode ((Handle -> m r) -> m (Maybe r)) -> (Handle -> m r) -> m (Maybe r)
forall a b. (a -> b) -> a -> b
$
Maybe [Char] -> Handle -> m r
atomicAction Maybe [Char]
forall a. Maybe a
Nothing
case Maybe r
mRes of
Just r
res -> r -> m r
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure r
res
Maybe r
Nothing ->
Maybe DirFd -> [Char] -> IOMode -> ([Char] -> Handle -> m r) -> m r
forall (m :: * -> *) a.
MonadUnliftIO m =>
Maybe DirFd -> [Char] -> IOMode -> ([Char] -> Handle -> m a) -> m a
withNonAnonymousBinaryTempFileFor Maybe DirFd
forall a. Maybe a
Nothing [Char]
filePath IOMode
iomode (([Char] -> Handle -> m r) -> m r)
-> ([Char] -> Handle -> m r) -> m r
forall a b. (a -> b) -> a -> b
$ \[Char]
tmpFilePath ->
Maybe [Char] -> Handle -> m r
atomicAction ([Char] -> Maybe [Char]
forall a. a -> Maybe a
Just [Char]
tmpFilePath)
where
atomicAction :: Maybe [Char] -> Handle -> m r
atomicAction Maybe [Char]
mTmpFilePath Handle
tmpFileHandle = do
let eTmpFile :: Either Handle [Char]
eTmpFile = Either Handle [Char]
-> ([Char] -> Either Handle [Char])
-> Maybe [Char]
-> Either Handle [Char]
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (Handle -> Either Handle [Char]
forall a b. a -> Either a b
Left Handle
tmpFileHandle) [Char] -> Either Handle [Char]
forall a b. b -> Either a b
Right Maybe [Char]
mTmpFilePath
Maybe CMode
mFileMode <- IOMode -> [Char] -> Handle -> m (Maybe CMode)
forall (f :: * -> *).
MonadUnliftIO f =>
IOMode -> [Char] -> Handle -> f (Maybe CMode)
copyFileHandle IOMode
iomode [Char]
filePath Handle
tmpFileHandle
r
res <- Handle -> m r
action Handle
tmpFileHandle
IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$ Maybe DirFd
-> Maybe CMode -> Either Handle [Char] -> [Char] -> IO ()
atomicTempFileRename Maybe DirFd
forall a. Maybe a
Nothing Maybe CMode
mFileMode Either Handle [Char]
eTmpFile [Char]
filePath
r -> m r
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure r
res