{-# OPTIONS_GHC -Wwarn #-} {-| Top-Level Pier Management This is the code that starts the IO drivers and deals with communication between the serf, the log, and the IO drivers. -} module Urbit.Vere.Pier ( booted , runSerf , resumed , getSnapshot , pier , runPersist , runCompute , generateBootSeq ) where import Urbit.Prelude import RIO.Directory import System.Random import Urbit.Arvo import Urbit.King.Config import Urbit.Vere.Pier.Types import Control.Monad.Trans.Maybe import Data.Text (append) import System.Posix.Files (ownerModes, setFileMode) import Urbit.King.App (HasConfigDir(..), HasStderrLogFunc(..)) -- ort Urbit.Time (Wen) import Urbit.Vere.Ames (ames) import Urbit.Vere.Behn (behn) import Urbit.Vere.Clay (clay) import Urbit.Vere.Http.Client (client) import Urbit.Vere.Http.Server (serv) import Urbit.Vere.Log (EventLog) import Urbit.Vere.Serf (Serf, ComputeRequest(..), SpinState, EvErr(..)) import qualified System.Entropy as Ent import qualified Urbit.King.API as King import qualified Urbit.Time as Time import qualified Urbit.Vere.Log as Log import qualified Urbit.Vere.Serf as Serf import qualified Urbit.Vere.Term as Term import qualified Urbit.Vere.Term.API as Term import qualified Urbit.Vere.Term.Demux as Term import qualified Urbit.Vere.Term.Render as Term -------------------------------------------------------------------------------- setupPierDirectory :: FilePath -> RIO e () setupPierDirectory shipPath = do for_ ["put", "get", "log", "chk"] $ \seg -> do let pax = shipPath <> "/.urb/" <> seg createDirectoryIfMissing True pax io $ setFileMode pax ownerModes -- Load pill into boot sequence. ----------------------------------------------- genEntropy :: RIO e Word512 genEntropy = fromIntegral . bytesAtom <$> io (Ent.getEntropy 64) generateBootSeq :: Ship -> Pill -> Bool -> LegacyBootEvent -> RIO e BootSeq generateBootSeq ship Pill{..} lite boot = do ent <- genEntropy let ovums = preKern ent <> pKernelOvums <> postKern <> pUserspaceOvums pure $ BootSeq ident pBootFormulas ovums where ident = LogIdentity ship isFake (fromIntegral $ length pBootFormulas) preKern ent = [ EvBlip $ BlipEvArvo $ ArvoEvWhom () ship , EvBlip $ BlipEvArvo $ ArvoEvWack () ent ] postKern = [ EvBlip $ BlipEvTerm $ TermEvBoot (1,()) lite boot ] isFake = case boot of Fake _ -> True _ -> False -- Write a batch of jobs into the event log ------------------------------------ writeJobs :: EventLog -> Vector Job -> RIO e () writeJobs log !jobs = do expect <- Log.nextEv log events <- fmap fromList $ traverse fromJob (zip [expect..] $ toList jobs) Log.appendEvents log events where fromJob :: (EventId, Job) -> RIO e ByteString fromJob (expectedId, job) = do unless (expectedId == jobId job) $ error $ show ("bad job id!", expectedId, jobId job) pure $ jamBS $ jobPayload job jobPayload :: Job -> Noun jobPayload (RunNok (LifeCyc _ m n)) = toNoun (m, n) jobPayload (DoWork (Work _ m d o)) = toNoun (m, d, o) -- Boot a new ship. ------------------------------------------------------------ printTank :: (Text -> IO ()) -> Atom -> Tank -> IO () printTank f _ = io . f . unlines . fmap unTape . wash (WashCfg 0 80) runSerf :: HasLogFunc e => TVar (Text -> IO ()) -> FilePath -> [Serf.Flag] -> RAcquire e Serf runSerf vSlog pax fax = do env <- ask Serf.withSerf (config env) where slog txt = join $ atomically (readTVar vSlog >>= pure . ($ txt)) config env = Serf.Config { scSerf = "urbit-worker" -- TODO Find the executable in some proper way. , scPier = pax , scFlag = fax , scSlog = \(pri, tank) -> printTank slog pri tank , scStdr = \line -> runRIO env $ logTrace (display ("SERF: " <> line)) , scDead = pure () -- TODO: What can be done? } bootSeqJobs :: Time.Wen -> BootSeq -> [Job] bootSeqJobs now (BootSeq ident nocks ovums) = zipWith ($) bootSeqFns [1 ..] where wen :: EventId -> Time.Wen wen off = Time.addGap now ((fromIntegral off - 1) ^. from Time.microSecs) bootSeqFns :: [EventId -> Job] bootSeqFns = fmap muckNock nocks <> fmap muckOvum ovums where muckNock nok eId = RunNok $ LifeCyc eId 0 nok muckOvum ov eId = DoWork $ Work eId 0 (wen eId) ov bootNewShip :: (HasPierConfig e, HasStderrLogFunc e, HasLogFunc e) => Pill -> Bool -> [Serf.Flag] -> Ship -> LegacyBootEvent -> RIO e () bootNewShip pill lite flags ship bootEv = do seq@(BootSeq ident x y) <- generateBootSeq ship pill lite bootEv logTrace "BootSeq Computed" pierPath <- view pierPathL liftRIO (setupPierDirectory pierPath) logTrace "Directory setup." rwith (Log.new (pierPath <> "/.urb/log") ident) $ \log -> do logTrace "Event log initialized." jobs <- (\now -> bootSeqJobs now seq) <$> io Time.now writeJobs log (fromList jobs) logTrace "Finsihed populating event log with boot sequence" booted :: (HasPierConfig e, HasStderrLogFunc e, HasLogFunc e) => TVar (Text -> IO ()) -> Pill -> Bool -> [Serf.Flag] -> Ship -> LegacyBootEvent -> RAcquire e (Serf, EventLog) booted vSlog pill lite flags ship boot = do rio $ bootNewShip pill lite flags ship boot resumed vSlog Nothing flags -- Resume an existing ship. ---------------------------------------------------- resumed :: (HasStderrLogFunc e, HasPierConfig e, HasLogFunc e) => TVar (Text -> IO ()) -> Maybe Word64 -> [Serf.Flag] -> RAcquire e (Serf, EventLog) resumed vSlog replayUntil flags = do rio $ logTrace "Resuming ship" top <- view pierPathL tap <- fmap (fromMaybe top) $ rio $ runMaybeT $ do ev <- MaybeT (pure replayUntil) MaybeT (getSnapshot top ev) rio $ logTrace $ display @Text ("pier: " <> pack top) rio $ logTrace $ display @Text ("running serf in: " <> pack tap) log <- Log.existing (top <> "/.urb/log") serf <- runSerf vSlog tap flags rio $ do logTrace "Replaying events" Serf.execReplay serf log replayUntil logTrace "Taking snapshot" io (Serf.snapshot serf) logTrace "Shuting down the serf" pure (serf, log) getSnapshot :: forall e. FilePath -> Word64 -> RIO e (Maybe FilePath) getSnapshot top last = do lastSnapshot <- lastMay <$> listReplays pure (replayToPath <$> lastSnapshot) where replayDir = top ".partial-replay" replayToPath eId = replayDir show eId listReplays :: RIO e [Word64] listReplays = do createDirectoryIfMissing True replayDir snapshotNums <- mapMaybe readMay <$> listDirectory replayDir pure $ sort (filter (<= fromIntegral last) snapshotNums) -- Run Pier -------------------------------------------------------------------- acquireWorker :: RIO e () -> RAcquire e (Async ()) acquireWorker act = mkRAcquire (async act) cancel acquireWorkerBound :: RIO e () -> RAcquire e (Async ()) acquireWorkerBound act = mkRAcquire (asyncBound act) cancel pier :: ∀e. (HasConfigDir e, HasLogFunc e, HasNetworkConfig e, HasPierConfig e) => (Serf, EventLog) -> TVar (Text -> IO ()) -> MVar () -> RAcquire e () pier (serf, log) vSlog mStart = do computeQ <- newTQueueIO persistQ <- newTQueueIO executeQ <- newTQueueIO saveM <- newEmptyTMVarIO shutdownM <- newEmptyTMVarIO kapi <- King.kingAPI termApiQ <- atomically $ do q <- newTQueue writeTVar (King.kTermConn kapi) (Just $ writeTQueue q) pure q let shutdownEvent = putTMVar shutdownM () inst <- io (KingId . UV . fromIntegral <$> randomIO @Word16) -- (sz, local) <- Term.localClient -- (waitExternalTerm, termServPort) <- Term.termServer (demux, muxed) <- atomically $ do res <- Term.mkDemux -- Term.addDemux local res pure (res, Term.useDemux res) -- rio $ logInfo $ display $ -- "TERMSERV Terminal Server running on port: " <> tshow termServPort acquireWorker $ forever $ do logTrace "TERMSERV Waiting for external terminal." atomically $ do ext <- Term.connClient <$> readTQueue termApiQ Term.addDemux ext demux logTrace "TERMSERV External terminal connected." -- Slogs go to both stderr and to the terminal. atomically $ do oldSlog <- readTVar vSlog writeTVar vSlog $ \txt -> do atomically $ Term.trace muxed txt oldSlog txt let logId = Log.identity log let ship = who logId -- Our call above to set the logging function which echos errors from the -- Serf doesn't have the appended \r\n because those \r\n s are added in -- the c serf code. Logging output from our haskell process must manually -- add them. let showErr = atomically . Term.trace muxed . (flip append "\r\n") let (bootEvents, startDrivers) = drivers inst ship (isFake logId) (writeTQueue computeQ) shutdownEvent (Term.TSize{tsWide=80, tsTall=24}, muxed) showErr io $ atomically $ for_ bootEvents (writeTQueue computeQ) let stubErrCallback = \_ -> pure () let computeConfig = ComputeConfig { ccOnWork = (\x -> EvErr x stubErrCallback) <$> readTQueue computeQ , ccOnKill = takeTMVar shutdownM , ccOnSave = takeTMVar saveM , ccPutResult = writeTQueue persistQ , ccShowSpinner = Term.spin muxed , ccHideSpinner = Term.stopSpin muxed } tExe <- startDrivers >>= acquireWorker . router (readTQueue executeQ) tDisk <- acquireWorkerBound (runPersist log persistQ (writeTQueue executeQ)) tCpu <- acquireWorker (runCompute serf computeConfig) tSaveSignal <- saveSignalThread saveM putMVar mStart () -- Wait for something to die. let ded = asum [ death "effect thread" tExe , death "persist thread" tDisk , death "compute thread" tCpu ] atomically ded >>= \case Left (txt, exn) -> logError $ displayShow ("Somthing died", txt, exn) Right tag -> logError $ displayShow ("something simply exited", tag) atomically $ (Term.spin muxed) (Just "shutdown") death :: Text -> Async () -> STM (Either (Text, SomeException) Text) death tag tid = do waitCatchSTM tid <&> \case Left exn -> Left (tag, exn) Right () -> Right tag saveSignalThread :: TMVar () -> RAcquire e (Async ()) saveSignalThread tm = mkRAcquire start cancel where start = async $ forever $ do threadDelay (120 * 1000000) -- 120 seconds atomically $ putTMVar tm () -- Start All Drivers ----------------------------------------------------------- data Drivers e = Drivers { dAmes :: EffCb e AmesEf , dBehn :: EffCb e BehnEf , dHttpClient :: EffCb e HttpClientEf , dHttpServer :: EffCb e HttpServerEf , dNewt :: EffCb e NewtEf , dSync :: EffCb e SyncEf , dTerm :: EffCb e TermEf } drivers :: (HasLogFunc e, HasNetworkConfig e, HasPierConfig e) => KingId -> Ship -> Bool -> (Ev -> STM ()) -> STM() -> (Term.TSize, Term.Client) -> (Text -> RIO e ()) -> ([Ev], RAcquire e (Drivers e)) drivers inst who isFake plan shutdownSTM termSys stderr = (initialEvents, runDrivers) where (behnBorn, runBehn) = behn inst plan (amesBorn, runAmes) = ames inst who isFake plan stderr (httpBorn, runHttp) = serv inst plan isFake (clayBorn, runClay) = clay inst plan (irisBorn, runIris) = client inst plan (termBorn, runTerm) = Term.term termSys shutdownSTM inst plan initialEvents = mconcat [behnBorn, clayBorn, amesBorn, httpBorn, termBorn, irisBorn] runDrivers = do dNewt <- runAmes dBehn <- liftAcquire $ runBehn dAmes <- pure $ const $ pure () dHttpClient <- runIris dHttpServer <- runHttp dSync <- runClay dTerm <- runTerm pure (Drivers{..}) -- Route Effects to Drivers ---------------------------------------------------- router :: HasLogFunc e => STM FX -> Drivers e -> RIO e () router waitFx Drivers {..} = forever $ do fx <- atomically waitFx for_ fx $ \ef -> do logEffect ef case ef of GoodParse (EfVega _ _ ) -> error "TODO" GoodParse (EfExit _ _ ) -> error "TODO" GoodParse (EfVane (VEAmes ef)) -> dAmes ef GoodParse (EfVane (VEBehn ef)) -> dBehn ef GoodParse (EfVane (VEBoat ef)) -> dSync ef GoodParse (EfVane (VEClay ef)) -> dSync ef GoodParse (EfVane (VEHttpClient ef)) -> dHttpClient ef GoodParse (EfVane (VEHttpServer ef)) -> dHttpServer ef GoodParse (EfVane (VENewt ef)) -> dNewt ef GoodParse (EfVane (VESync ef)) -> dSync ef GoodParse (EfVane (VETerm ef)) -> dTerm ef FailParse n -> logError $ display $ pack @Text (ppShow n) -- Compute Thread -------------------------------------------------------------- logEvent :: HasLogFunc e => Ev -> RIO e () logEvent ev = logDebug $ display $ "[EVENT]\n" <> pretty where pretty :: Text pretty = pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow ev logEffect :: HasLogFunc e => Lenient Ef -> RIO e () logEffect ef = logDebug $ display $ "[EFFECT]\n" <> pretty ef where pretty :: Lenient Ef -> Text pretty = \case GoodParse e -> pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow e FailParse n -> pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow n data ComputeConfig = ComputeConfig { ccOnWork :: STM EvErr , ccOnKill :: STM () , ccOnSave :: STM () , ccPutResult :: (Fact, FX) -> STM () , ccShowSpinner :: Maybe Text -> STM () , ccHideSpinner :: STM () } runCompute :: forall e . HasLogFunc e => Serf.Serf -> ComputeConfig -> RIO e () runCompute serf ComputeConfig {..} = do logTrace "runCompute" let onCR = asum [ CRKill <$> ccOnKill , CRSave <$> ccOnSave , CRWork <$> ccOnWork ] let onOutput :: Serf.RunOutput -> STM () onOutput (Serf.RunOutput e m w nounEv fx) = do ccPutResult (Fact e m w nounEv, GoodParse <$> fx) -- TODO GoodParse let onSpin :: SpinState -> STM () onSpin Nothing = ccHideSpinner onSpin (Just ev) = ccShowSpinner (getSpinnerNameForEvent ev) io (Serf.swimming serf onCR onOutput onSpin) -- Persist Thread -------------------------------------------------------------- data PersistExn = BadEventId EventId EventId deriving Show instance Exception PersistExn where displayException (BadEventId expected got) = unlines [ "Out-of-order event id send to persist thread." , "\tExpected " <> show expected <> " but got " <> show got ] runPersist :: forall e . (HasPierConfig e, HasLogFunc e) => EventLog -> TQueue (Fact, FX) -> (FX -> STM ()) -> RIO e () runPersist log inpQ out = do dryRun <- view dryRunL forever $ do writs <- atomically getBatchFromQueue events <- validateFactsAndGetBytes (fst <$> toNullable writs) unless dryRun (Log.appendEvents log events) atomically $ for_ writs $ \(_, fx) -> do out fx where validateFactsAndGetBytes :: [Fact] -> RIO e (Vector ByteString) validateFactsAndGetBytes facts = do expect <- Log.nextEv log lis <- for (zip [expect ..] facts) $ \(expectedId, Fact eve mug wen non) -> do unless (expectedId == eve) $ do throwIO (BadEventId expectedId eve) pure $ jamBS $ toNoun (mug, wen, non) pure (fromList lis) getBatchFromQueue :: STM (NonNull [(Fact, FX)]) getBatchFromQueue = readTQueue inpQ >>= go . singleton where go acc = tryReadTQueue inpQ >>= \case Nothing -> pure (reverse acc) Just item -> go (item <| acc)