{-# LANGUAGE OverloadedStrings, ExistentialQuantification #-} module MOO.Task ( -- * Monad Interface MOO , Environment(..) , initEnvironment , liftSTM -- * World Interface , World(..) , newWorld , getWorld , getWorld' , putWorld , modifyWorld , getDatabase , putDatabase -- * Task Interface , Task(..) , TaskStatus(..) , Wake(..) , TaskState(..) , CallStack(..) , DelayedIO(..) , TaskDisposition(..) , Resume(..) , Resource(..) , initState , newState , initTask , newTaskId , newTask , taskOwner , isQueued , queuedTasks , stepTask , runTask , forkTask , interrupt , requestIO , delayIO , unsafeIOtoMOO , getTask , putTask , purgeTask -- * Object Interface , getPlayer , getObject , getObjectName , getProperty , modifyProperty , modifyVerb , readProperty , writeProperty , setBuiltinProperty -- * Verb Execution Interface , getVerb , findVerb , callSystemVerb , callSystemVerb' , callCommandVerb , callVerb , callFromFunc , evalFromFunc , runVerb , runTick -- * Verb Frame Interface , StackFrame(..) , Continuation(..) , initFrame , formatFrames , pushFrame , popFrame , activeFrame , frame , caller , modifyFrame , setLineNumber , mkVariables , formatTraceback -- * Loop and Try/Finally Control Functions , pushTryFinallyContext , pushLoopContext , setLoopContinue , popContext , breakLoop , continueLoop -- * Exception Handling , Exception(..) , Code , Message , raiseException , raise , catchException , passException , handleDebug , timeoutException -- * Utility Check Functions , isWizard , checkFloat , checkProgrammer , checkWizard , checkPermission , checkValid , checkFertile , checkRecurrence -- * Miscellaneous , binaryString , random , newRandomGen , delay , shutdown , notyet ) where import Control.Arrow ((&&&)) import Control.Concurrent (MVar, ThreadId, myThreadId, forkIO, threadDelay, newEmptyMVar, putMVar, tryPutMVar, takeMVar) import Control.Concurrent.STM (STM, TVar, atomically, retry, throwSTM, newEmptyTMVar, putTMVar, takeTMVar, newTVarIO, readTVar, writeTVar, modifyTVar) import Control.Exception (SomeException, catch) import Control.Monad (when, unless, join, liftM, void, (>=>)) import Control.Monad.Cont (ContT, runContT, callCC) import Control.Monad.Reader (ReaderT, runReaderT, local, asks) import Control.Monad.State.Strict (StateT, runStateT, get, gets, modify) import Control.Monad.Trans.Class (lift) import Control.Monad.Writer (execWriter, tell) import Data.ByteString (ByteString) import Data.Int (Int32) import Data.List (find) import Data.Map (Map) import Data.Maybe (isNothing, fromMaybe, fromJust) import Data.Monoid (Monoid(mempty, mappend), (<>)) import Data.Text (Text) import Data.Time (UTCTime, getCurrentTime, addUTCTime) import Data.Time.Clock.POSIX (posixSecondsToUTCTime) import System.IO.Unsafe (unsafePerformIO) import System.Posix (nanosleep) import System.Random (Random, StdGen, newStdGen, mkStdGen, split, randomR, randomRs) import qualified Data.Map as M import qualified Data.Text as T import MOO.Command import {-# SOURCE #-} MOO.Database import {-# SOURCE #-} MOO.Network import {-# SOURCE #-} MOO.Connection import MOO.Object import MOO.Types import MOO.Verb import qualified MOO.String as Str -- | This is the basic MOO monad transformer stack. A computation of type -- @'MOO' a@ is an 'STM' transaction that returns a value of type @a@ within -- an environment that supports state, continuations, and local modification. type MOO = ReaderT Environment (ContT TaskDisposition (StateT TaskState STM)) -- | Lift an 'STM' transaction into the 'MOO' monad. liftSTM :: STM a -> MOO a liftSTM = lift . lift . lift -- | The known universe, as far as the MOO server is concerned data World = World { database :: Database -- ^ The database of objects , tasks :: Map TaskId Task -- ^ Queued and running tasks , listeners :: Map Point Listener -- ^ Network listening points , connections :: Map ObjId Connection -- ^ Network connections , nextConnectionId :: ObjId -- ^ The (negative) object number to be assigned to the next inbound or -- outbound connection , outboundNetwork :: Bool -- ^ Is @open_network_connection()@ enabled? , bindAddress :: Maybe HostName -- ^ Interface address to bind to for incoming connections , shutdownMessage :: MVar Text -- ^ Shutdown signal } initWorld = World { database = undefined , tasks = M.empty , listeners = M.empty , connections = M.empty , nextConnectionId = firstConnectionId , outboundNetwork = False , bindAddress = Nothing , shutdownMessage = undefined } newWorld :: Database -> Bool -> IO (TVar World) newWorld db outboundNetworkEnabled = do shutdownVar <- newEmptyMVar world' <- newTVarIO initWorld { database = db , outboundNetwork = outboundNetworkEnabled , shutdownMessage = shutdownVar } runTask =<< newTask world' nothing (loadServerOptions >> return zero) return world' -- | A structure representing a queued or running task data Task = Task { taskId :: TaskId , taskStatus :: TaskStatus , taskThread :: ThreadId , taskWorld :: TVar World , taskPlayer :: ObjId , taskState :: TaskState , taskComputation :: MOO Value } instance Show Task where show task = "" initTask = Task { taskId = 0 , taskStatus = Pending , taskThread = undefined , taskWorld = undefined , taskPlayer = nothing , taskState = initState , taskComputation = return zero } instance Sizeable Task where storageBytes task = storageBytes (taskId task) + storageBytes (taskThread task) + storageBytes (taskWorld task) + storageBytes (taskPlayer task) + storageBytes (taskStatus task) + storageBytes (taskState task) -- storageBytes (taskComputation task) instance Eq Task where Task { taskId = taskId1 } == Task { taskId = taskId2 } = taskId1 == taskId2 instance Ord Task where Task { taskState = state1 } `compare` Task { taskState = state2 } = startTime state1 `compare` startTime state2 type TaskId = Int32 -- | Generate a (random) 'TaskId' not currently in use by any existing task. newTaskId :: World -> StdGen -> TaskId newTaskId world = fromJust . find unused . randomRs (1, maxBound) where unused = (`M.notMember` tasks world) -- | Create a pending 'Task' for the given computation on behalf of the given -- player. A new 'TaskId' is reserved for the task and the task is added to -- the 'World'. (The task will not actually run until passed to 'runTask'.) newTask :: TVar World -> ObjId -> MOO Value -> IO Task newTask world' player comp = do gen <- newStdGen state <- newState atomically $ do world <- readTVar world' let taskId = newTaskId world gen task = initTask { taskId = taskId , taskWorld = world' , taskPlayer = player , taskState = state , taskComputation = comp } writeTVar world' world { tasks = M.insert taskId task (tasks world) } return task taskOwner :: Task -> ObjId taskOwner = permissions . activeFrame -- | The running state of a task data TaskStatus = Pending | Running | Forked | Suspended Wake | Reading deriving Show isQueued :: TaskStatus -> Bool isQueued Pending = False isQueued Running = False isQueued _ = True isRunning :: TaskStatus -> Bool isRunning Running = True isRunning _ = False queuedTasks :: MOO [Task] queuedTasks = (filter (isQueued . taskStatus) . M.elems . tasks) `liftM` getWorld instance Sizeable TaskStatus where storageBytes (Suspended _) = 2 * storageBytes () storageBytes _ = storageBytes () -- | A function to call in order to wake a suspended task newtype Wake = Wake (Value -> IO ()) instance Show Wake where show _ = "Wake{..}" -- | The intermediate or final result of a running task data TaskDisposition = Complete Value | Suspend (Maybe Integer) (Resume ()) | Read ObjId (Resume Value) | forall a. RequestIO (IO a) (Resume a) | Uncaught Exception | Timeout Resource CallStack | Suicide -- | A continuation to resume the execution of a task where it left off newtype Resume a = Resume (a -> MOO Value) -- | Task resource limits data Resource = Ticks | Seconds showResource :: Resource -> StrT showResource Ticks = "ticks" showResource Seconds = "seconds" timeoutException :: Resource -> CallStack -> Exception timeoutException resource stack = except { exceptionCallStack = stack } where message = "Task ran out of " <> showResource resource except = newException (Err E_QUOTA) message zero stepTask :: Task -> IO (TaskDisposition, Task) stepTask task = do let env = initEnvironment task comp = taskComputation task comp' = callCC $ \k -> Complete `liftM` local (\r -> r { interruptHandler = Interrupt k }) comp state = taskState task contM = runReaderT comp' env stateM = runContT contM return stmM = runStateT stateM state (result, state') <- atomically stmM runDelayed $ delayedIO state' return (result, task { taskState = state' { delayedIO = mempty }}) stepTaskWithIO :: Task -> IO (TaskDisposition, Task) stepTaskWithIO task = do (disposition, task') <- stepTask task case disposition of RequestIO io (Resume resume) -> do result <- io stepTaskWithIO task' { taskComputation = resume result } _ -> return (disposition, task') -- | Run a task in a new Haskell thread, returning either the value produced -- by the task, or 'Nothing' if the task suspends or aborts before producing a -- value. If the task suspends, it may continue running after this function -- returns. The task is removed from the task queue after it is finished. runTask :: Task -> IO (Maybe Value) runTask task = do resultMVar <- newEmptyMVar forkIO $ do threadId <- myThreadId let task' = task { taskThread = threadId } atomically $ modifyTVar (taskWorld task) $ \world -> world { tasks = M.insert (taskId task) task' { taskStatus = Running } $ tasks world } runTask' task' $ putMVar resultMVar atomically $ modifyTVar (taskWorld task) $ \world -> world { tasks = M.delete (taskId task) $ tasks world } takeMVar resultMVar where noOp = const $ return () runTask' :: Task -> (Maybe Value -> IO ()) -> IO () runTask' task putResult = do (disposition, task') <- stepTaskWithIO task case disposition of Complete value -> putResult (Just value) Suspend _ (Resume resume) -> do putResult Nothing -- restart this task only when there are none other running atomically $ do world <- readTVar (taskWorld task') when (any (isRunning . taskStatus) $ M.elems $ tasks world) retry runTask' task' { taskComputation = resume () } noOp Read _ _ -> error "read() not yet implemented" Uncaught exception@Exception { exceptionCode = code , exceptionMessage = message , exceptionValue = value , exceptionCallStack = Stack frames } -> handleAbortedTask task' formatted putResult $ callSystemVerb "handle_uncaught_error" [code, Str message, value, traceback, stringList formatted] where traceback = formatFrames True frames formatted = formatTraceback exception Timeout resource stack@(Stack frames) -> handleAbortedTask task' formatted putResult $ callSystemVerb "handle_task_timeout" [Str $ showResource resource, traceback, stringList formatted] where traceback = formatFrames True frames formatted = formatTraceback $ timeoutException resource stack Suicide -> putResult Nothing handleAbortedTask :: Task -> [StrT] -> (Maybe Value -> IO ()) -> MOO (Maybe Value) -> IO () handleAbortedTask task traceback putResult call = do state <- newState handleAbortedTask' traceback task { taskState = state , taskComputation = fromMaybe zero `fmap` call } where handleAbortedTask' :: [StrT] -> Task -> IO () handleAbortedTask' traceback task = do (disposition, task') <- stepTaskWithIO task case disposition of Complete value -> do unless (truthOf value) $ informPlayer traceback putResult Nothing Suspend _ (Resume resume) -> do -- The aborted task is considered "handled" but continue -- running the suspended handler (which might abort -- again!) putResult Nothing runTask' task' { taskComputation = resume () } noOp Read _ _ -> error "read() not yet implemented" Uncaught exception -> do informPlayer traceback informPlayer $ formatTraceback exception putResult Nothing Timeout resource stack -> do informPlayer traceback informPlayer $ formatTraceback $ timeoutException resource stack putResult Nothing Suicide -> putResult Nothing informPlayer :: [StrT] -> IO () informPlayer lines = atomically $ do world <- readTVar (taskWorld task) let maybeConnection = M.lookup (taskPlayer task) (connections world) case maybeConnection of Just conn -> mapM_ (sendToConnection conn . Str.toText) lines Nothing -> return () -- XXX write to server log? -- | Create and queue a task to run the given computation after the given -- microsecond delay. 'E_INVARG' may be raised if the delay is out of -- acceptable range. (The given 'TaskId' should have been reserved by a call -- to 'newTaskId'.) forkTask :: TaskId -> Integer -> MOO Value -> MOO () forkTask taskId usecs code = do state <- get let now = startTime state estimatedWakeup = (fromIntegral usecs / 1000000) `addUTCTime` now when (estimatedWakeup < now || estimatedWakeup > endOfTime) $ raise E_INVARG task <- asks task gen <- newRandomGen let frame = currentFrame (stack state) frame' = frame { depthLeft = depthLeft initFrame , contextStack = contextStack initFrame , lineNumber = lineNumber frame + 1 } state' = initState { ticksLeft = 15000 -- XXX , stack = Stack [frame'] , startTime = estimatedWakeup , randomGen = gen } task' = task { taskId = taskId , taskStatus = Forked , taskState = state' , taskComputation = code } -- make sure the forked task doesn't start before the current task commits startSignal <- liftSTM newEmptyTMVar threadId <- requestIO $ forkIO $ do delay usecs atomically $ takeTMVar startSignal now <- getCurrentTime void $ runTask task' { taskState = state' { startTime = now } } modifyWorld $ \world -> world { tasks = M.insert taskId task' { taskThread = threadId } $ tasks world } liftSTM $ putTMVar startSignal () -- | Wait for the given number of microseconds to elapse. delay :: Integer -> IO () delay usecs = if usecs <= fromIntegral (maxBound :: Int) then threadDelay (fromIntegral usecs) else nanosleep (usecs * 1000) -- | A continuation for returning to the task dispatcher to handle an -- interrupt request. Note that calling this continuation implies a commit to -- the current task's transaction. newtype InterruptHandler = Interrupt (TaskDisposition -> MOO TaskDisposition) -- | Commit the current task's transaction, and return to the task dispatcher -- with an interrupt request. The task dispatcher may resume execution of the -- task later if the request is one which supplies an appropriate -- continuation. interrupt :: TaskDisposition -> MOO a interrupt disp = do Interrupt handler <- asks interruptHandler handler disp error "Returned from interrupt handler" -- | An 'IO' computation to be performed after the current task commits its -- 'STM' transaction newtype DelayedIO = DelayedIO { runDelayed :: IO () } instance Monoid DelayedIO where mempty = DelayedIO $ return () DelayedIO a `mappend` DelayedIO b = DelayedIO (a >> b) -- | Interrupt the current task to perform the given IO computation, and -- return the result. Note this implies a commit of the task's 'STM' -- transaction. requestIO :: IO a -> MOO a requestIO io = callCC $ interrupt . RequestIO io . Resume -- | Perform the given IO computation after the current task commits its 'STM' -- transaction. -- -- Since general IO can't be performed within a transaction, this is a simple -- alternative when the value returned by the IO isn't needed. delayIO :: IO () -> MOO () delayIO io = modify $ \state -> state { delayedIO = delayedIO state `mappend` DelayedIO io } -- | Unsafely perform the given IO computation within the current 'STM' -- transaction. -- -- Since 'STM' transactions may be aborted at any time, the IO is performed in -- a separate thread in order to guarantee consistency with any finalizers, -- brackets, and so forth. The IO must be idempotent as it may be run more -- than once. unsafeIOtoMOO :: IO a -> MOO a unsafeIOtoMOO io = liftSTM $ do let result = unsafePerformIO $ do r <- newEmptyMVar forkIO $ (io >>= putMVar r . Right) `catch` \e -> putMVar r $ Left (e :: SomeException) takeMVar r case result of Right x -> return x Left e -> throwSTM e -- | A 'Reader' environment for state that either doesn't change, or can be -- locally modified for subcomputations data Environment = Env { task :: Task , interruptHandler :: InterruptHandler , exceptionHandler :: ExceptionHandler , indexLength :: MOO Int } initEnvironment :: Task -> Environment initEnvironment task = Env { task = task , interruptHandler = error "Undefined interrupt handler" , exceptionHandler = Handler $ interrupt . Uncaught , indexLength = error "Invalid index context" } -- | A 'State' structure for data that may normally change during computation data TaskState = State { ticksLeft :: Int , stack :: CallStack , startTime :: UTCTime , randomGen :: StdGen , delayedIO :: DelayedIO } initState = State { ticksLeft = 30000 , stack = Stack [] , startTime = posixSecondsToUTCTime 0 , randomGen = mkStdGen 0 , delayedIO = mempty } instance Sizeable TaskState where storageBytes state = storageBytes (ticksLeft state) + storageBytes (stack state) + storageBytes (startTime state) + storageBytes (randomGen state) -- storageBytes (delayedIO state) newState :: IO TaskState newState = do startTime <- getCurrentTime gen <- newStdGen return initState { startTime = startTime , randomGen = gen } getWorld :: MOO World getWorld = liftSTM . readTVar . taskWorld =<< asks task getWorld' :: MOO (TVar World) getWorld' = asks (taskWorld . task) putWorld :: World -> MOO () putWorld world = do world' <- getWorld' liftSTM $ writeTVar world' world modifyWorld :: (World -> World) -> MOO () modifyWorld f = do world' <- getWorld' liftSTM $ modifyTVar world' f getTask :: TaskId -> MOO (Maybe Task) getTask taskId = (M.lookup taskId . tasks) `liftM` getWorld putTask :: Task -> MOO () putTask task = modifyWorld $ \world -> world { tasks = M.insert (taskId task) task $ tasks world } purgeTask :: Task -> MOO () purgeTask task = modifyWorld $ \world -> world { tasks = M.delete (taskId task) $ tasks world } getDatabase :: MOO Database getDatabase = database `liftM` getWorld putDatabase :: Database -> MOO () putDatabase db = modifyWorld $ \world -> world { database = db } getPlayer :: MOO ObjId getPlayer = asks (taskPlayer . task) getObject :: ObjId -> MOO (Maybe Object) getObject oid = liftSTM . dbObject oid =<< getDatabase getObjectName :: ObjId -> MOO StrT getObjectName oid = do obj <- getObject oid let objNum = Str.fromText (toText $ Obj oid) return $ maybe objNum (\obj -> Str.concat [objectName obj, " (", objNum, ")"]) obj getProperty :: Object -> StrT -> MOO Property getProperty obj name = do maybeProp <- liftSTM $ lookupProperty obj name maybe (raise E_PROPNF) return maybeProp getVerb :: Object -> Value -> MOO Verb getVerb obj desc@Str{} = do maybeVerb <- liftSTM $ lookupVerb obj desc maybe (raise E_VERBNF) return maybeVerb getVerb obj desc@(Int index) | index < 1 = raise E_INVARG | otherwise = do maybeVerb <- liftSTM $ lookupVerb obj desc maybe (raise E_VERBNF) return maybeVerb getVerb _ _ = raise E_TYPE findVerb :: (Verb -> Bool) -> StrT -> ObjId -> MOO (Maybe ObjId, Maybe Verb) findVerb acceptable name = findVerb' where findVerb' oid = do maybeObj <- getObject oid case maybeObj of Nothing -> return (Nothing, Nothing) Just obj -> do maybeVerb <- searchVerbs (objectVerbs obj) case maybeVerb of Just verb -> return (Just oid, Just verb) Nothing -> maybe (return (Just oid, Nothing)) findVerb' (objectParent obj) searchVerbs ((names,verbTVar):rest) = if verbNameMatch name names then do verb <- liftSTM $ readTVar verbTVar if acceptable verb then return (Just verb) else searchVerbs rest else searchVerbs rest searchVerbs [] = return Nothing callSystemVerb :: StrT -> [Value] -> MOO (Maybe Value) callSystemVerb name args = callSystemVerb' systemObject name args Str.empty callSystemVerb' :: ObjId -> StrT -> [Value] -> StrT -> MOO (Maybe Value) callSystemVerb' object name args argstr = do player <- asks (taskPlayer . task) maybeVerb <- findVerb verbPermX name object case maybeVerb of (Just verbOid, Just verb) -> do let vars = mkVariables [ ("player", Obj player) , ("this" , Obj object) , ("verb" , Str name) , ("args" , fromList args) , ("argstr", Str argstr) ] Just `liftM` runVerb verb initFrame { variables = vars , verbName = name , verbLocation = verbOid , initialThis = object , initialPlayer = player } _ -> return Nothing callCommandVerb :: ObjId -> (ObjId, Verb) -> ObjId -> Command -> (ObjId, ObjId) -> MOO Value callCommandVerb player (verbOid, verb) this command (dobj, iobj) = do let vars = mkVariables [ ("player" , Obj player) , ("this" , Obj this) , ("caller" , Obj player) , ("verb" , Str $ commandVerb command) , ("argstr" , Str $ commandArgStr command) , ("args" , stringList $ commandArgs command) , ("dobjstr", Str $ commandDObjStr command) , ("dobj" , Obj dobj) , ("prepstr", Str $ commandPrepStr command) , ("iobjstr", Str $ commandIObjStr command) , ("iobj" , Obj iobj) ] runVerb verb initFrame { variables = vars , verbName = commandVerb command , verbLocation = verbOid , initialThis = this , initialPlayer = player } callVerb' :: (ObjId, Verb) -> ObjId -> StrT -> [Value] -> MOO Value callVerb' (verbOid, verb) this name args = do thisFrame <- frame id wizard <- isWizard (permissions thisFrame) let player = case (wizard, vars M.! "player") of (True, Obj oid) -> oid _ -> initialPlayer thisFrame vars = variables thisFrame vars' = mkVariables [ ("this" , Obj this) , ("verb" , Str name) , ("args" , fromList args) , ("caller" , Obj $ initialThis thisFrame) , ("player" , Obj player) , ("argstr" , vars M.! "argstr") , ("dobjstr", vars M.! "dobjstr") , ("dobj" , vars M.! "dobj") , ("prepstr", vars M.! "prepstr") , ("iobjstr", vars M.! "iobjstr") , ("iobj" , vars M.! "iobj") ] runVerb verb initFrame { variables = vars' , verbName = name , verbLocation = verbOid , initialThis = this , initialPlayer = player } callVerb :: ObjId -> ObjId -> StrT -> [Value] -> MOO Value callVerb verbLoc this name args = do maybeVerb <- findVerb verbPermX name verbLoc case maybeVerb of (Just verbOid, Just verb) -> callVerb' (verbOid, verb) this name args (Nothing , _) -> raise E_INVIND (_ , Nothing) -> raise E_VERBNF callFromFunc :: StrT -> IntT -> (ObjId, StrT) -> [Value] -> MOO (Maybe Value) callFromFunc func index (oid, name) args = do maybeVerb <- findVerb verbPermX name oid case maybeVerb of (Just verbOid, Just verb) -> liftM Just $ evalFromFunc func index $ callVerb' (verbOid, verb) oid name args _ -> return Nothing evalFromFunc :: StrT -> IntT -> MOO Value -> MOO Value evalFromFunc func index code = do (depthLeft, player) <- frame (depthLeft &&& initialPlayer) pushFrame initFrame { depthLeft = depthLeft , verbName = func , initialPlayer = player , builtinFunc = True , lineNumber = index } value <- code `catchException` \except -> do popFrame passException except popFrame return value runVerb :: Verb -> StackFrame -> MOO Value runVerb verb verbFrame = do Stack frames <- gets stack let depthLeft' = depthLeft $ case frames of frame:_ -> frame [] -> initFrame unless (depthLeft' > 0) $ raise E_MAXREC pushFrame verbFrame { depthLeft = depthLeft' - 1 , debugBit = verbPermD verb , permissions = verbOwner verb , verbFullName = verbNames verb } value <- verbCode verb `catchException` \except -> do popFrame passException except popFrame return value runTick :: MOO () runTick = do ticksLeft <- gets ticksLeft unless (ticksLeft > 0) $ interrupt . Timeout Ticks =<< gets stack modify $ \state -> state { ticksLeft = ticksLeft - 1 } modifyProperty :: Object -> StrT -> (Property -> MOO Property) -> MOO () modifyProperty obj name f = case lookupPropertyRef obj name of Nothing -> raise E_PROPNF Just propTVar -> do prop <- liftSTM $ readTVar propTVar prop' <- f prop liftSTM $ writeTVar propTVar prop' modifyVerb :: (ObjId, Object) -> Value -> (Verb -> MOO Verb) -> MOO () modifyVerb (oid, obj) desc f = case lookupVerbRef obj desc of Nothing -> raise E_VERBNF Just (index, verbTVar) -> do verb <- liftSTM $ readTVar verbTVar verb' <- f verb liftSTM $ writeTVar verbTVar verb' unless (verbNames verb `Str.equal` verbNames verb') $ do db <- getDatabase liftSTM $ modifyObject oid db $ replaceVerb index verb' readProperty :: ObjId -> StrT -> MOO (Maybe Value) readProperty oid name = do maybeObj <- getObject oid case maybeObj of Nothing -> return Nothing Just obj -> maybe (search obj) (return . Just . ($ obj)) $ builtinProperty name where search obj = do maybeProp <- liftSTM $ lookupProperty obj name case maybeProp of Nothing -> return Nothing Just prop -> case propertyValue prop of Nothing -> do parentObj <- maybe (return Nothing) getObject (objectParent obj) maybe (error $ "No inherited value for property " ++ Str.toString name) search parentObj just -> return just writeProperty :: ObjId -> StrT -> Value -> MOO () writeProperty oid name value = do maybeObj <- getObject oid case maybeObj of Nothing -> return () Just obj -> if isBuiltinProperty name then setBuiltinProperty (oid, obj) name value else case lookupPropertyRef obj name of Nothing -> return () Just propTVar -> liftSTM $ do prop <- readTVar propTVar writeTVar propTVar prop { propertyValue = Just value } setBuiltinProperty :: (ObjId, Object) -> StrT -> Value -> MOO () setBuiltinProperty (oid, obj) "name" (Str name) = do if objectIsPlayer obj then checkWizard else checkPermission (objectOwner obj) db <- getDatabase liftSTM $ modifyObject oid db $ \obj -> return obj { objectName = name } setBuiltinProperty (oid, _) "owner" (Obj owner) = do checkWizard db <- getDatabase liftSTM $ modifyObject oid db $ \obj -> return obj { objectOwner = owner } setBuiltinProperty _ "location" (Obj _) = raise E_PERM setBuiltinProperty _ "contents" (Lst _) = raise E_PERM setBuiltinProperty (oid, _) "programmer" bit = do checkWizard db <- getDatabase liftSTM $ modifyObject oid db $ \obj -> return obj { objectProgrammer = truthOf bit } setBuiltinProperty (oid, _) "wizard" bit = do checkWizard db <- getDatabase liftSTM $ modifyObject oid db $ \obj -> return obj { objectWizard = truthOf bit } setBuiltinProperty (oid, obj) "r" bit = do checkPermission (objectOwner obj) db <- getDatabase liftSTM $ modifyObject oid db $ \obj -> return obj { objectPermR = truthOf bit } setBuiltinProperty (oid, obj) "w" bit = do checkPermission (objectOwner obj) db <- getDatabase liftSTM $ modifyObject oid db $ \obj -> return obj { objectPermW = truthOf bit } setBuiltinProperty (oid, obj) "f" bit = do checkPermission (objectOwner obj) db <- getDatabase liftSTM $ modifyObject oid db $ \obj -> return obj { objectPermF = truthOf bit } setBuiltinProperty _ _ _ = raise E_TYPE -- | The stack of verb and/or built-in function frames newtype CallStack = Stack [StackFrame] deriving Show instance Sizeable CallStack where storageBytes (Stack stack) = storageBytes stack -- | A local continuation for loop constructs newtype Continuation = Continuation (() -> MOO Value) instance Sizeable Continuation where storageBytes _ = storageBytes () -- | A structure describing a (possibly nested) context for the current frame, -- used to manage loop break/continue and try/finally interactions data Context = Loop { loopName :: Maybe Id , loopBreak :: Continuation , loopContinue :: Continuation } | TryFinally { finally :: MOO Value } instance Sizeable Context where storageBytes context@Loop{} = storageBytes (loopName context) + storageBytes (loopBreak context) + storageBytes (loopContinue context) storageBytes TryFinally{} = storageBytes () -- storageBytes (finally context) instance Show Context where show Loop { loopName = Nothing } = "" show Loop { loopName = Just name } = "" show TryFinally{} = "" -- | The data tracked for each verb and/or built-in function call data StackFrame = Frame { depthLeft :: Int , contextStack :: [Context] , variables :: Map Id Value , debugBit :: Bool , permissions :: ObjId , verbName :: StrT , verbFullName :: StrT , verbLocation :: ObjId , initialThis :: ObjId , initialPlayer :: ObjId , builtinFunc :: Bool , lineNumber :: IntT } deriving Show initFrame = Frame { depthLeft = 50 , contextStack = [] , variables = initVariables , debugBit = True , permissions = nothing , verbName = Str.empty , verbFullName = Str.empty , verbLocation = nothing , initialThis = nothing , initialPlayer = nothing , builtinFunc = False , lineNumber = 0 } instance Sizeable StackFrame where storageBytes frame = storageBytes (depthLeft frame) + storageBytes (contextStack frame) + storageBytes (variables frame) + storageBytes (debugBit frame) + storageBytes (permissions frame) + storageBytes (verbName frame) + storageBytes (verbFullName frame) + storageBytes (verbLocation frame) + storageBytes (initialThis frame) + storageBytes (initialPlayer frame) + storageBytes (builtinFunc frame) + storageBytes (lineNumber frame) formatFrames :: Bool -> [StackFrame] -> Value formatFrames includeLineNumbers = fromListBy formatFrame where formatFrame frame = fromList $ Obj (initialThis frame) : Str (verbName frame) : Obj (permissions frame) : Obj (verbLocation frame) : Obj (initialPlayer frame) : [Int $ lineNumber frame | includeLineNumbers] pushFrame :: StackFrame -> MOO () pushFrame frame = modify $ \state@State { stack = Stack frames } -> state { stack = Stack (frame : frames) } popFrame :: MOO () popFrame = do unwindContexts (const False) modify $ \state@State { stack = Stack (_:frames) } -> state { stack = Stack frames } currentFrame :: CallStack -> StackFrame currentFrame (Stack (frame:_)) = frame currentFrame (Stack []) = error "currentFrame: Empty call stack" previousFrame :: CallStack -> Maybe StackFrame previousFrame (Stack (_:frames)) = previousFrame' frames where previousFrame' (frame:frames) | builtinFunc frame = previousFrame' frames | otherwise = Just frame previousFrame' [] = Nothing previousFrame (Stack []) = error "previousFrame: Empty call stack" activeFrame :: Task -> StackFrame activeFrame = currentFrame . stack . taskState frame :: (StackFrame -> a) -> MOO a frame f = gets (f . currentFrame . stack) caller :: (StackFrame -> a) -> MOO (Maybe a) caller f = gets (fmap f . previousFrame . stack) modifyFrame :: (StackFrame -> StackFrame) -> MOO () modifyFrame f = modify $ \state@State { stack = Stack (frame:stack) } -> state { stack = Stack (f frame : stack) } setLineNumber :: Int -> MOO () setLineNumber lineNumber = modifyFrame $ \frame -> frame { lineNumber = fromIntegral lineNumber } pushContext :: Context -> MOO () pushContext context = modifyFrame $ \frame -> frame { contextStack = context : contextStack frame } pushTryFinallyContext :: MOO Value -> MOO () pushTryFinallyContext finally = pushContext TryFinally { finally = finally } pushLoopContext :: Maybe Id -> Continuation -> MOO () pushLoopContext name break = pushContext Loop { loopName = name , loopBreak = break , loopContinue = undefined } setLoopContinue :: Continuation -> MOO () setLoopContinue continue = modifyFrame $ \frame@Frame { contextStack = loop:loops } -> frame { contextStack = loop { loopContinue = continue } : loops } popContext :: MOO () popContext = modifyFrame $ \frame@Frame { contextStack = _:contexts } -> frame { contextStack = contexts } unwindContexts :: (Context -> Bool) -> MOO [Context] unwindContexts p = do stack <- unwind =<< frame contextStack modifyFrame $ \frame -> frame { contextStack = stack } return stack where unwind stack@(this:next) = if p this then return stack else do case this of TryFinally { finally = finally } -> do modifyFrame $ \frame -> frame { contextStack = next } void finally _ -> return () unwind next unwind [] = return [] unwindLoopContext :: Maybe Id -> MOO Context unwindLoopContext maybeName = do loop:_ <- unwindContexts testContext return loop where testContext Loop { loopName = name } = isNothing maybeName || maybeName == name testContext _ = False breakLoop :: Maybe Id -> MOO Value breakLoop maybeName = do Loop { loopBreak = Continuation break } <- unwindLoopContext maybeName break () continueLoop :: Maybe Id -> MOO Value continueLoop maybeName = do Loop { loopContinue = Continuation continue } <- unwindLoopContext maybeName continue () -- | The default collection of verb variables initVariables :: Map Id Value initVariables = M.fromList $ [ ("player" , Obj nothing) , ("this" , Obj nothing) , ("caller" , Obj nothing) , ("args" , emptyList) , ("argstr" , emptyString) , ("verb" , emptyString) , ("dobjstr", emptyString) , ("dobj" , Obj nothing) , ("prepstr", emptyString) , ("iobjstr", emptyString) , ("iobj" , Obj nothing) ] ++ typeVariables where typeVariables = [ ("INT" , Int $ typeCode TInt) , ("NUM" , Int $ typeCode TInt) , ("FLOAT", Int $ typeCode TFlt) , ("LIST" , Int $ typeCode TLst) , ("STR" , Int $ typeCode TStr) , ("OBJ" , Int $ typeCode TObj) , ("ERR" , Int $ typeCode TErr) ] -- | Create a variable block for a verb by overriding the default. mkVariables :: [(Id, Value)] -> Map Id Value mkVariables = foldr (uncurry M.insert) initVariables newtype ExceptionHandler = Handler (Exception -> MOO Value) instance Show ExceptionHandler where show _ = "" -- | A MOO exception data Exception = Exception { exceptionCode :: Code , exceptionMessage :: Message , exceptionValue :: Value , exceptionCallStack :: CallStack , exceptionDebugBit :: Bool -- ^ A copy of the debug bit from the verb frame in which the exception -- was raised } type Code = Value type Message = StrT initException = Exception { exceptionCode = Err E_NONE , exceptionMessage = Str.fromText (error2text E_NONE) , exceptionValue = zero , exceptionCallStack = Stack [] , exceptionDebugBit = True } newException :: Code -> Message -> Value -> Exception newException code message value = initException { exceptionCode = code , exceptionMessage = message , exceptionValue = value } -- | Install a local exception handler for the duration of the passed -- computation. catchException :: MOO a -> (Exception -> MOO a) -> MOO a catchException action handler = callCC $ \k -> local (mkHandler k) action where mkHandler k env = env { exceptionHandler = Handler $ \e -> local (const env) $ handler e >>= k } -- | Re-raise an exception to the next enclosing handler. passException :: Exception -> MOO a passException except = do Handler handler <- asks exceptionHandler handler except error "Returned from exception handler" -- | Abort execution of the current computation and call the nearest enclosing -- exception handler. raiseException :: Code -> Message -> Value -> MOO a raiseException code message value = do let except = newException code message value callStack <- gets stack debug <- frame debugBit passException except { exceptionCallStack = callStack , exceptionDebugBit = debug } -- | Execute the passed computation, capturing any exception raised in verb -- frames with debug bit unset and returning the error code as an ordinary -- value instead of propagating the exception. handleDebug :: MOO Value -> MOO Value handleDebug = (`catchException` handler) where handler Exception { exceptionDebugBit = False , exceptionCode = code } = return code handler except = passException except -- | Placeholder for features not yet implemented notyet :: StrT -> MOO a notyet = raiseException (Err E_QUOTA) "Not yet implemented" . Str -- | Create and raise an exception for the given MOO error. raise :: Error -> MOO a raise err = raiseException (Err err) (Str.fromText $ error2text err) zero -- | Verify that the given floating point number is neither infinite nor NaN, -- raising 'E_FLOAT' or 'E_INVARG' respectively if so. Also, return the -- corresponding MOO value. checkFloat :: FltT -> MOO Value checkFloat flt | isInfinite flt = raise E_FLOAT | isNaN flt = raise E_INVARG | otherwise = return (Flt flt) -- | Verify that the given object has a programmer bit, raising 'E_PERM' if -- not. checkProgrammer' :: ObjId -> MOO () checkProgrammer' perm = do programmer <- maybe False objectProgrammer `liftM` getObject perm unless programmer $ raise E_PERM -- | Verify that the current task permissions have programmer privileges, -- raising 'E_PERM' if not. checkProgrammer :: MOO () checkProgrammer = checkProgrammer' =<< frame permissions -- | Determine whether the given object has its wizard bit set. isWizard :: ObjId -> MOO Bool isWizard = liftM (maybe False objectWizard) . getObject -- | Verify that the given object is a wizard, raising 'E_PERM' if not. checkWizard' :: ObjId -> MOO () checkWizard' perm = do wizard <- isWizard perm unless wizard $ raise E_PERM -- | Verify that the current task permissions have wizard privileges, raising -- 'E_PERM' if not. checkWizard :: MOO () checkWizard = checkWizard' =<< frame permissions -- | Verify that the current task permissions either have wizard privileges or -- are the same as the given object, raising 'E_PERM' if not. checkPermission :: ObjId -> MOO () checkPermission who = do perm <- frame permissions unless (perm == who) $ checkWizard' perm -- | Verify that the given object is valid, raising 'E_INVARG' if not. Also, -- return the referenced object. checkValid :: ObjId -> MOO Object checkValid = getObject >=> maybe (raise E_INVARG) return -- | Verify that the given object is fertile for the current task permissions, -- raising 'E_PERM' if not. checkFertile :: ObjId -> MOO () checkFertile oid = do maybeObj <- getObject oid case maybeObj of Nothing -> raise E_PERM Just obj -> unless (objectPermF obj) $ checkPermission (objectOwner obj) -- | Verify that the given /object/ does not have a recursive relationship -- with the given /subject/, raising 'E_RECMOVE' if so. checkRecurrence :: (Object -> Maybe ObjId) -- ^ relationship projection -> ObjId -- ^ /subject/ -> ObjId -- ^ /object/ to check -> MOO () checkRecurrence relation subject = checkRecurrence' where checkRecurrence' object = do when (object == subject) $ raise E_RECMOVE maybeObject <- getObject object case join $ relation `fmap` maybeObject of Just oid -> checkRecurrence' oid Nothing -> return () -- | Translate a MOO /binary string/ into a Haskell 'ByteString', raising -- 'E_INVARG' if the MOO string is improperly formatted. binaryString :: StrT -> MOO ByteString binaryString = maybe (raise E_INVARG) return . Str.toBinary -- | Generate and return a pseudorandom number in the given range, modifying -- the local generator state. random :: (Random a) => (a, a) -> MOO a random range = do (r, gen) <- randomR range `liftM` gets randomGen modify $ \state -> state { randomGen = gen } return r -- | Split the local random number generator state in two, updating the local -- state with one of them and returning the other. newRandomGen :: MOO StdGen newRandomGen = do (gen, gen') <- split `liftM` gets randomGen modify $ \state -> state { randomGen = gen } return gen' -- | Generate traceback lines for an exception, suitable for displaying to a -- user. formatTraceback :: Exception -> [StrT] formatTraceback except@Exception { exceptionCallStack = Stack frames } = map Str.fromText $ T.splitOn "\n" $ execWriter (traceback frames) where traceback (frame:frames) = do describeVerb frame tell $ ": " <> Str.toText (exceptionMessage except) traceback' frames traceback [] = traceback' [] traceback' (frame:frames) = do tell "\n... called from " describeVerb frame traceback' frames traceback' [] = tell "\n(End of traceback)" describeVerb Frame { builtinFunc = False , verbLocation = loc, verbFullName = name , initialThis = this, lineNumber = line } = do tell $ "#" <> T.pack (show loc) <> ":" <> Str.toText name when (loc /= this) $ tell $ " (this == #" <> T.pack (show this) <> ")" when (line > 0) $ tell $ ", line " <> T.pack (show line) describeVerb Frame { builtinFunc = True, verbName = name } = tell $ "built-in function " <> Str.toText name <> "()" -- | Begin the server shutdown process. shutdown :: StrT -> MOO () shutdown message = do world <- getWorld mapM_ unlisten (M.keys $ listeners world) delayIO $ void $ tryPutMVar (shutdownMessage world) (Str.toText message)