{-# LANGUAGE LambdaCase #-} {-# LANGUAGE RecordWildCards #-} {-# LANGUAGE TypeFamilies #-} module Control.Monad.Interactive where import Control.Applicative import qualified Control.Monad.Fail as Fail import Control.Monad.Primitive import Control.Monad.Reader import qualified Data.List as L import System.Environment import System.IO import System.Process class (Monad m) => MonadInteractive m where {-# MINIMAL sendStr, recvLine #-} sendStr :: String -> m () recvLine :: m String send :: (Show a) => a -> m () send = String -> m () forall (m :: * -> *). MonadInteractive m => String -> m () sendStrLn (String -> m ()) -> (a -> String) -> a -> m () forall b c a. (b -> c) -> (a -> b) -> a -> c . a -> String forall a. Show a => a -> String show sendStrLn :: String -> m () sendStrLn String cs = String -> m () forall (m :: * -> *). MonadInteractive m => String -> m () sendStr String cs m () -> m () -> m () forall a b. m a -> m b -> m b forall (m :: * -> *) a b. Monad m => m a -> m b -> m b >> String -> m () forall (m :: * -> *). MonadInteractive m => String -> m () sendStr String "\n" recvLn :: (Read a) => m a recvLn = String -> a forall a. Read a => String -> a read (String -> a) -> m String -> m a forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> m String forall (m :: * -> *). MonadInteractive m => m String recvLine data InteractiveHandle = InteractiveHandle { InteractiveHandle -> Handle hin :: Handle , InteractiveHandle -> Handle hout :: Handle , InteractiveHandle -> Maybe Handle mherr :: Maybe Handle } createInteractiveHandle :: IO InteractiveHandle createInteractiveHandle :: IO InteractiveHandle createInteractiveHandle = IO [String] getArgs IO [String] -> ([String] -> IO InteractiveHandle) -> IO InteractiveHandle forall a b. IO a -> (a -> IO b) -> IO b forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b >>= \case [] -> InteractiveHandle -> IO InteractiveHandle forall a. a -> IO a forall (m :: * -> *) a. Monad m => a -> m a return (InteractiveHandle -> IO InteractiveHandle) -> InteractiveHandle -> IO InteractiveHandle forall a b. (a -> b) -> a -> b $ Handle -> Handle -> Maybe Handle -> InteractiveHandle InteractiveHandle Handle stdin Handle stdout Maybe Handle forall a. Maybe a Nothing [String] args -> do let cmds :: [String] cmds = (String -> Bool) -> [String] -> [String] forall a. (a -> Bool) -> [a] -> [a] filter (Bool -> Bool not (Bool -> Bool) -> (String -> Bool) -> String -> Bool forall b c a. (b -> c) -> (a -> b) -> a -> c . String -> String -> Bool forall a. Eq a => [a] -> [a] -> Bool L.isPrefixOf String "-") [String] args (Just hout, Just hin, _, _) <- CreateProcess -> IO (Maybe Handle, Maybe Handle, Maybe Handle, ProcessHandle) createProcess (String -> CreateProcess shell (String -> CreateProcess) -> ([String] -> String) -> [String] -> CreateProcess forall b c a. (b -> c) -> (a -> b) -> a -> c . [String] -> String unwords ([String] -> CreateProcess) -> [String] -> CreateProcess forall a b. (a -> b) -> a -> b $ String "python3 local_testing_tool.py" String -> [String] -> [String] forall a. a -> [a] -> [a] : [String] cmds) { std_in = CreatePipe , std_out = CreatePipe } let mherr | String "--verbose" String -> [String] -> Bool forall a. Eq a => a -> [a] -> Bool forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool `elem` [String] args = Handle -> Maybe Handle forall a. a -> Maybe a Just Handle stderr | Bool otherwise = Maybe Handle forall a. Maybe a Nothing return $ InteractiveHandle{..} withInteractiveHandle :: ReaderT InteractiveHandle IO a -> IO a withInteractiveHandle :: forall a. ReaderT InteractiveHandle IO a -> IO a withInteractiveHandle ReaderT InteractiveHandle IO a f = IO InteractiveHandle createInteractiveHandle IO InteractiveHandle -> (InteractiveHandle -> IO a) -> IO a forall a b. IO a -> (a -> IO b) -> IO b forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b >>= ReaderT InteractiveHandle IO a -> InteractiveHandle -> IO a forall r (m :: * -> *) a. ReaderT r m a -> r -> m a runReaderT ReaderT InteractiveHandle IO a f sendDebugFormat :: String -> String sendDebugFormat :: String -> String sendDebugFormat = (String "> " String -> String -> String forall a. [a] -> [a] -> [a] ++) recvDebugFormat :: String -> String recvDebugFormat :: String -> String recvDebugFormat = (String "< " String -> String -> String forall a. [a] -> [a] -> [a] ++) instance (MonadIO m) => MonadInteractive (ReaderT InteractiveHandle m) where sendStr :: String -> ReaderT InteractiveHandle m () sendStr String cs = (InteractiveHandle -> m ()) -> ReaderT InteractiveHandle m () forall r (m :: * -> *) a. (r -> m a) -> ReaderT r m a ReaderT ((InteractiveHandle -> m ()) -> ReaderT InteractiveHandle m ()) -> (InteractiveHandle -> m ()) -> ReaderT InteractiveHandle m () forall a b. (a -> b) -> a -> b $ \InteractiveHandle{Maybe Handle Handle hin :: InteractiveHandle -> Handle hout :: InteractiveHandle -> Handle mherr :: InteractiveHandle -> Maybe Handle hin :: Handle hout :: Handle mherr :: Maybe Handle ..} -> 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 (Handle -> IO ()) -> Maybe Handle -> IO () forall (t :: * -> *) (m :: * -> *) a b. (Foldable t, Monad m) => (a -> m b) -> t a -> m () mapM_ (\Handle herr -> Handle -> String -> IO () hPutStr Handle herr (String -> IO ()) -> String -> IO () forall a b. (a -> b) -> a -> b $ String -> String sendDebugFormat String cs) Maybe Handle mherr Handle -> String -> IO () hPutStr Handle hout String cs Handle -> IO () hFlush Handle hout sendStrLn :: String -> ReaderT InteractiveHandle m () sendStrLn String cs = (InteractiveHandle -> m ()) -> ReaderT InteractiveHandle m () forall r (m :: * -> *) a. (r -> m a) -> ReaderT r m a ReaderT ((InteractiveHandle -> m ()) -> ReaderT InteractiveHandle m ()) -> (InteractiveHandle -> m ()) -> ReaderT InteractiveHandle m () forall a b. (a -> b) -> a -> b $ \InteractiveHandle{Maybe Handle Handle hin :: InteractiveHandle -> Handle hout :: InteractiveHandle -> Handle mherr :: InteractiveHandle -> Maybe Handle hin :: Handle hout :: Handle mherr :: Maybe Handle ..} -> 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 (Handle -> IO ()) -> Maybe Handle -> IO () forall (t :: * -> *) (m :: * -> *) a b. (Foldable t, Monad m) => (a -> m b) -> t a -> m () mapM_ (\Handle herr -> Handle -> String -> IO () hPutStrLn Handle herr (String -> IO ()) -> String -> IO () forall a b. (a -> b) -> a -> b $ String -> String sendDebugFormat String cs) Maybe Handle mherr Handle -> String -> IO () hPutStrLn Handle hout String cs Handle -> IO () hFlush Handle hout recvLine :: ReaderT InteractiveHandle m String recvLine = (InteractiveHandle -> m String) -> ReaderT InteractiveHandle m String forall r (m :: * -> *) a. (r -> m a) -> ReaderT r m a ReaderT ((InteractiveHandle -> m String) -> ReaderT InteractiveHandle m String) -> (InteractiveHandle -> m String) -> ReaderT InteractiveHandle m String forall a b. (a -> b) -> a -> b $ \InteractiveHandle{Maybe Handle Handle hin :: InteractiveHandle -> Handle hout :: InteractiveHandle -> Handle mherr :: InteractiveHandle -> Maybe Handle hin :: Handle hout :: Handle mherr :: Maybe Handle ..} -> IO String -> m String forall a. IO a -> m a forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO String -> m String) -> IO String -> m String forall a b. (a -> b) -> a -> b $ do res <- Handle -> IO String hGetLine Handle hin mapM_ (\Handle herr -> Handle -> String -> IO () hPutStrLn Handle herr (String -> IO ()) -> String -> IO () forall a b. (a -> b) -> a -> b $ String -> String recvDebugFormat String res) mherr return res type Result m r = m r type Accepted m r = Result m r type Failed m r = Result m r type Running a m r = a -> Result m r type JudgeInternal m a r = Accepted m r -> Failed m r -> Running a m r -> Result m r newtype Judge m a = Judge {forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge :: forall r. JudgeInternal m a r} instance Functor (Judge m) where fmap :: forall a b. (a -> b) -> Judge m a -> Judge m b fmap a -> b f Judge m a m = (forall (r :: k). JudgeInternal m b r) -> Judge m b forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge ((forall (r :: k). JudgeInternal m b r) -> Judge m b) -> (forall (r :: k). JudgeInternal m b r) -> Judge m b forall a b. (a -> b) -> a -> b $ \Accepted m r ac Accepted m r wa Running b m r k -> Judge m a -> forall (r :: k). JudgeInternal m a r forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge Judge m a m Accepted m r ac Accepted m r wa (Running b m r k Running b m r -> (a -> b) -> a -> Accepted m r forall b c a. (b -> c) -> (a -> b) -> a -> c . a -> b f) {-# INLINE fmap #-} instance Applicative (Judge m) where pure :: forall a. a -> Judge m a pure a x = (forall (r :: k). JudgeInternal m a r) -> Judge m a forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge ((forall (r :: k). JudgeInternal m a r) -> Judge m a) -> (forall (r :: k). JudgeInternal m a r) -> Judge m a forall a b. (a -> b) -> a -> b $ \Accepted m r _ Accepted m r _ Running a m r k -> Running a m r k a x {-# INLINE pure #-} Judge m (a -> b) mf <*> :: forall a b. Judge m (a -> b) -> Judge m a -> Judge m b <*> Judge m a mx = (forall (r :: k). JudgeInternal m b r) -> Judge m b forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge ((forall (r :: k). JudgeInternal m b r) -> Judge m b) -> (forall (r :: k). JudgeInternal m b r) -> Judge m b forall a b. (a -> b) -> a -> b $ \Accepted m r ac Accepted m r wa Running b m r k -> Judge m (a -> b) -> forall (r :: k). JudgeInternal m (a -> b) r forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge Judge m (a -> b) mf Accepted m r ac Accepted m r wa (Running (a -> b) m r -> Accepted m r) -> Running (a -> b) m r -> Accepted m r forall a b. (a -> b) -> a -> b $ \a -> b f -> Judge m a -> forall (r :: k). JudgeInternal m a r forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge Judge m a mx Accepted m r ac Accepted m r wa (Running b m r k Running b m r -> (a -> b) -> a -> Accepted m r forall b c a. (b -> c) -> (a -> b) -> a -> c . a -> b f) {-# INLINE (<*>) #-} instance Alternative (Judge m) where empty :: forall a. Judge m a empty = (forall (r :: k). JudgeInternal m a r) -> Judge m a forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge ((forall (r :: k). JudgeInternal m a r) -> Judge m a) -> (forall (r :: k). JudgeInternal m a r) -> Judge m a forall a b. (a -> b) -> a -> b $ \Accepted m r ac Accepted m r _ Running a m r _ -> Accepted m r ac {-# INLINE empty #-} Judge m a mx <|> :: forall a. Judge m a -> Judge m a -> Judge m a <|> Judge m a my = (forall (r :: k). JudgeInternal m a r) -> Judge m a forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge ((forall (r :: k). JudgeInternal m a r) -> Judge m a) -> (forall (r :: k). JudgeInternal m a r) -> Judge m a forall a b. (a -> b) -> a -> b $ \Accepted m r ac Accepted m r wa Running a m r k -> Judge m a -> forall (r :: k). JudgeInternal m a r forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge Judge m a mx (Judge m a -> forall (r :: k). JudgeInternal m a r forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge Judge m a my Accepted m r ac Accepted m r wa Running a m r k) Accepted m r wa Running a m r k {-# INLINE (<|>) #-} instance Monad (Judge m) where Judge m a mx >>= :: forall a b. Judge m a -> (a -> Judge m b) -> Judge m b >>= a -> Judge m b f = (forall (r :: k). JudgeInternal m b r) -> Judge m b forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge ((forall (r :: k). JudgeInternal m b r) -> Judge m b) -> (forall (r :: k). JudgeInternal m b r) -> Judge m b forall a b. (a -> b) -> a -> b $ \Accepted m r ac Accepted m r wa Running b m r k -> Judge m a -> forall (r :: k). JudgeInternal m a r forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge Judge m a mx Accepted m r ac Accepted m r wa (Running a m r -> Accepted m r) -> Running a m r -> Accepted m r forall a b. (a -> b) -> a -> b $ \a x -> Judge m b -> forall (r :: k). JudgeInternal m b r forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge (a -> Judge m b f a x) Accepted m r ac Accepted m r wa Running b m r k {-# INLINE (>>=) #-} instance Fail.MonadFail (Judge m) where fail :: forall a. String -> Judge m a fail String _ = (forall (r :: k). JudgeInternal m a r) -> Judge m a forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge ((forall (r :: k). JudgeInternal m a r) -> Judge m a) -> (forall (r :: k). JudgeInternal m a r) -> Judge m a forall a b. (a -> b) -> a -> b $ \Accepted m r _ Accepted m r wa Running a m r _ -> Accepted m r wa {-# INLINE fail #-} instance MonadTrans Judge where lift :: forall (m :: * -> *) a. Monad m => m a -> Judge m a lift m a m = (forall r. JudgeInternal m a r) -> Judge m a forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge ((forall r. JudgeInternal m a r) -> Judge m a) -> (forall r. JudgeInternal m a r) -> Judge m a forall a b. (a -> b) -> a -> b $ \Accepted m r _ Accepted m r _ Running a m r k -> m a m m a -> Running a m r -> Accepted m r forall a b. m a -> (a -> m b) -> m b forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b >>= Running a m r k {-# INLINE lift #-} instance (PrimMonad m) => PrimMonad (Judge m) where type PrimState (Judge m) = PrimState m primitive :: forall a. (State# (PrimState (Judge m)) -> (# State# (PrimState (Judge m)), a #)) -> Judge m a primitive = m a -> Judge m a forall (m :: * -> *) a. Monad m => m a -> Judge m a forall (t :: (* -> *) -> * -> *) (m :: * -> *) a. (MonadTrans t, Monad m) => m a -> t m a lift (m a -> Judge m a) -> ((State# (PrimState m) -> (# State# (PrimState m), a #)) -> m a) -> (State# (PrimState m) -> (# State# (PrimState m), a #)) -> Judge m a forall b c a. (b -> c) -> (a -> b) -> a -> c . (State# (PrimState m) -> (# State# (PrimState m), a #)) -> m a forall a. (State# (PrimState m) -> (# State# (PrimState m), a #)) -> m a forall (m :: * -> *) a. PrimMonad m => (State# (PrimState m) -> (# State# (PrimState m), a #)) -> m a primitive {-# INLINE primitive #-} type Interactive a = Judge (ReaderT InteractiveHandle IO) a interactive :: (forall r. JudgeInternal (ReaderT InteractiveHandle IO) a r) -> Interactive a interactive :: forall a. (forall r. JudgeInternal (ReaderT InteractiveHandle IO) a r) -> Interactive a interactive = (forall r. JudgeInternal (ReaderT InteractiveHandle IO) a r) -> Judge (ReaderT InteractiveHandle IO) a forall {k} (m :: k -> *) a. (forall (r :: k). JudgeInternal m a r) -> Judge m a Judge runInteractive_ :: Interactive a -> ReaderT InteractiveHandle IO () runInteractive_ :: forall a. Interactive a -> ReaderT InteractiveHandle IO () runInteractive_ Interactive a m = Interactive a -> forall r. JudgeInternal (ReaderT InteractiveHandle IO) a r forall {k} (m :: k -> *) a. Judge m a -> forall (r :: k). JudgeInternal m a r unJudge Interactive a m (() -> ReaderT InteractiveHandle IO () forall a. a -> ReaderT InteractiveHandle IO a forall (m :: * -> *) a. Monad m => a -> m a return ()) (() -> ReaderT InteractiveHandle IO () forall a. a -> ReaderT InteractiveHandle IO a forall (m :: * -> *) a. Monad m => a -> m a return ()) (ReaderT InteractiveHandle IO () -> a -> ReaderT InteractiveHandle IO () forall a b. a -> b -> a const (ReaderT InteractiveHandle IO () -> a -> ReaderT InteractiveHandle IO ()) -> ReaderT InteractiveHandle IO () -> a -> ReaderT InteractiveHandle IO () forall a b. (a -> b) -> a -> b $ () -> ReaderT InteractiveHandle IO () forall a. a -> ReaderT InteractiveHandle IO a forall (m :: * -> *) a. Monad m => a -> m a return ()) withInteractive :: Interactive a -> IO () withInteractive :: forall a. Interactive a -> IO () withInteractive = ReaderT InteractiveHandle IO () -> IO () forall a. ReaderT InteractiveHandle IO a -> IO a withInteractiveHandle (ReaderT InteractiveHandle IO () -> IO ()) -> (Interactive a -> ReaderT InteractiveHandle IO ()) -> Interactive a -> IO () forall b c a. (b -> c) -> (a -> b) -> a -> c . Interactive a -> ReaderT InteractiveHandle IO () forall a. Interactive a -> ReaderT InteractiveHandle IO () runInteractive_