module BuildBox.Command.System.Internals
( streamIn
, streamOuts)
where
import System.IO
import Control.Concurrent
import Control.Concurrent.STM.TChan
import Control.Monad.STM
import Data.ByteString.Char8 (ByteString)
import qualified Data.ByteString.Char8 as BS
streamIn :: Handle -> TChan (Maybe ByteString) -> IO ()
streamIn !hRead !chan
= do eof <- hIsEOF hRead
if eof
then do
atomically $ writeTChan chan Nothing
return ()
else do
str <- BS.hGetLine hRead
atomically $ writeTChan chan (Just str)
streamIn hRead chan
streamOuts :: [(TChan (Maybe ByteString), (Maybe Handle), QSem)] -> IO ()
streamOuts !chans
= streamOuts' False [] chans
where
streamOuts' _ [] []
= return ()
streamOuts' True prev []
= streamOuts' False [] prev
streamOuts' False prev []
= do threadDelay 100000
streamOuts' False [] prev
streamOuts' !active !prev (!x@(!chan, !mHandle, !qsem) : rest)
= do
mStr <- atomically
$ do isEmpty <- isEmptyTChan chan
if isEmpty
then return (False, Nothing)
else do mStr <- readTChan chan
return (True, mStr)
case mStr of
(False, _)
-> streamOuts' active (prev ++ [x]) rest
(True, Nothing)
-> do signalQSem qsem
streamOuts' active prev rest
(True, Just str)
| Just h <- mHandle
-> do BS.hPutStr h str
hPutChar h '\n'
streamOuts' True (prev ++ [x]) rest
| otherwise
-> streamOuts' True (prev ++ [x]) rest