summaryrefslogtreecommitdiff
path: root/hsm-command/Hsm/Command
diff options
context:
space:
mode:
authorPaul Oliver <contact@pauloliver.dev>2024-08-24 11:57:18 -0700
committerPaul Oliver <contact@pauloliver.dev>2024-12-01 07:01:30 -0800
commitf0854265f7a1b59078308965d33fe2583a5c0f9c (patch)
treed8b06110d84fce783f1cc91aa37155351c655b2c /hsm-command/Hsm/Command
Initial commitHEADmaster
Diffstat (limited to 'hsm-command/Hsm/Command')
-rw-r--r--hsm-command/Hsm/Command/Command.hs67
-rw-r--r--hsm-command/Hsm/Command/Readline.hs58
2 files changed, 125 insertions, 0 deletions
diff --git a/hsm-command/Hsm/Command/Command.hs b/hsm-command/Hsm/Command/Command.hs
new file mode 100644
index 0000000..2080143
--- /dev/null
+++ b/hsm-command/Hsm/Command/Command.hs
@@ -0,0 +1,67 @@
+{-# LANGUAGE DeriveAnyClass #-}
+{-# LANGUAGE ImportQualifiedPost #-}
+
+module Hsm.Command.Command
+ ( Direction (Forward, Backward, Left, Right)
+ , Angle (CW, CCW)
+ , Speed (Slow, Mid, Fast)
+ , Command (Move, Rotate)
+ , commandStream
+ )
+where
+
+import Data.Binary (Binary)
+import Data.Function ((&))
+import Data.Maybe (fromJust, isJust)
+import Data.Text (pack)
+import Effectful (Eff, (:>))
+import Effectful.Log (Log, logAttention_)
+import GHC.Generics (Generic)
+import Hsm.Command.Readline (Readline, readline)
+import Streamly.Data.Stream (Stream, mapMaybeM, repeatM, takeWhile)
+import Text.Read (readEither)
+import Prelude hiding (Left, Right, takeWhile)
+import Prelude qualified as P
+
+data Direction
+ = Forward
+ | Backward
+ | Left
+ | Right
+ deriving (Binary, Generic, Read, Show)
+
+data Angle
+ = CW
+ | CCW
+ deriving (Binary, Generic, Read, Show)
+
+data Speed
+ = Slow
+ | Mid
+ | Fast
+ deriving (Binary, Generic, Read, Show)
+
+data Command
+ = Move Direction Speed Int
+ | Rotate Angle Speed Int
+ deriving (Binary, Generic, Read, Show)
+
+commandStream
+ :: forall es
+ . ( Log :> es
+ , Readline :> es
+ )
+ => Stream (Eff es) Command
+commandStream =
+ repeatM readline
+ & takeWhile isJust
+ & mapMaybeM (parse . fromJust)
+ where
+ parse
+ :: Log :> es
+ => String
+ -> Eff es (Maybe Command)
+ parse string =
+ case readEither string of
+ P.Left err -> logAttention_ (pack err) >> return Nothing
+ P.Right command -> return $ Just command
diff --git a/hsm-command/Hsm/Command/Readline.hs b/hsm-command/Hsm/Command/Readline.hs
new file mode 100644
index 0000000..66246b5
--- /dev/null
+++ b/hsm-command/Hsm/Command/Readline.hs
@@ -0,0 +1,58 @@
+{-# LANGUAGE ImportQualifiedPost #-}
+{-# LANGUAGE TypeFamilies #-}
+
+module Hsm.Command.Readline
+ ( Readline
+ , readline
+ , runReadline
+ )
+where
+
+import Data.Function ((&))
+import Effectful (Dispatch (Static), DispatchOf, Eff, Effect, IOE, (:>))
+import Effectful.Dispatch.Static qualified as S
+import Effectful.Log (Log, getLoggerEnv, leLogger, waitForLogger)
+import Effectful.Resource (Resource, allocate)
+import System.Console.Haskeline qualified as H
+import System.Console.Haskeline.IO qualified as H
+import Prelude hiding (takeWhile)
+
+data Readline :: Effect
+
+type instance DispatchOf Readline = Static S.WithSideEffects
+
+newtype instance S.StaticRep Readline = Readline H.InputState
+
+readline
+ :: forall es
+ . ( Log :> es
+ , Readline :> es
+ )
+ => Eff es (Maybe String)
+readline = do
+ flushLogger
+ Readline handle <- S.getStaticRep
+ H.getInputLine "% "
+ & H.withInterrupt
+ & H.handleInterrupt (return Nothing)
+ & H.queryInput handle
+ & S.unsafeEff_
+ where
+ flushLogger :: Eff es ()
+ flushLogger = getLoggerEnv >>= S.unsafeEff_ . waitForLogger . leLogger
+
+runReadline
+ :: ( IOE :> es
+ , Resource :> es
+ )
+ => Eff (Readline : es) a
+ -> Eff es a
+runReadline action = do
+ handle <- snd <$> allocate state H.cancelInput
+ S.evalStaticRep (Readline handle) action
+ where
+ settings :: H.Settings IO
+ settings = H.defaultSettings {H.historyFile = Just "/tmp/hsm_command_history"}
+
+ state :: IO H.InputState
+ state = H.initializeInput settings