Merge 0f069a08e8 into release/next-js

This commit is contained in:
janeway-bot 2021-01-30 02:48:42 +04:00 committed by GitHub
commit ddb0592b38
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
33 changed files with 950 additions and 456 deletions

View File

@ -155,8 +155,7 @@ let
contents = { contents = {
"${name}/urbit" = "${urbit}/bin/urbit"; "${name}/urbit" = "${urbit}/bin/urbit";
"${name}/urbit-worker" = "${urbit}/bin/urbit-worker"; "${name}/urbit-worker" = "${urbit}/bin/urbit-worker";
# temporarily removed for compatibility reasons "${name}/urbit-king" = "${urbit-king}/bin/urbit-king";
# "${name}/urbit-king" = "${urbit-king}/bin/urbit-king";
}; };
}; };

View File

@ -12,6 +12,7 @@
-} -}
module Urbit.Arvo.Common module Urbit.Arvo.Common
( KingId(..), ServId(..) ( KingId(..), ServId(..)
, Vere(..), Wynn(..)
, Json, JsonNode(..) , Json, JsonNode(..)
, Desk(..), Mime(..) , Desk(..), Mime(..)
, Port(..), Turf(..) , Port(..), Turf(..)
@ -21,9 +22,10 @@ module Urbit.Arvo.Common
, AmesDest, Ipv4(..), Ipv6(..), Patp(..), Galaxy, AmesAddress(..) , AmesDest, Ipv4(..), Ipv6(..), Patp(..), Galaxy, AmesAddress(..)
) where ) where
import Urbit.Prelude hiding (Term) import Urbit.Prelude
import Control.Monad.Fail (fail) import Control.Monad.Fail (fail)
import Data.Bits
import qualified Network.HTTP.Types.Method as H import qualified Network.HTTP.Types.Method as H
import qualified Urbit.Ob as Ob import qualified Urbit.Ob as Ob
@ -45,6 +47,25 @@ newtype KingId = KingId { unKingId :: UV }
newtype ServId = ServId { unServId :: UV } newtype ServId = ServId { unServId :: UV }
deriving newtype (Eq, Ord, Show, Num, Enum, Integral, Real, FromNoun, ToNoun) deriving newtype (Eq, Ord, Show, Num, Enum, Integral, Real, FromNoun, ToNoun)
-- Arvo Version Negotiation ----------------------------------------------------
-- Information about the king runtime passed to Arvo.
data Vere = Vere { vereName :: Term,
vereRev :: [Cord],
vereWynn :: Wynn }
deriving (Eq, Ord, Show)
instance ToNoun Vere where
toNoun Vere{..} = toNoun ((vereName, vereRev), vereWynn)
instance FromNoun Vere where
parseNoun n = named "Vere" $ do
((vereName, vereRev), vereWynn) <- parseNoun n
pure $ Vere {..}
-- A list of names and their kelvin numbers, used in version negotiations.
newtype Wynn = Wynn { unWynn :: [(Term, Word)] }
deriving newtype (Eq, Ord, Show, FromNoun, ToNoun)
-- Http Common ----------------------------------------------------------------- -- Http Common -----------------------------------------------------------------
@ -112,7 +133,7 @@ deriveNoun ''HttpServerConf
-- Desk and Mime --------------------------------------------------------------- -- Desk and Mime ---------------------------------------------------------------
newtype Desk = Desk { unDesk :: Cord } newtype Desk = Desk { unDesk :: Cord }
deriving newtype (Eq, Ord, Show, ToNoun, FromNoun) deriving newtype (Eq, Ord, Show, ToNoun, FromNoun, IsString)
data Mime = Mime Path File data Mime = Mime Path File
deriving (Eq, Ord, Show) deriving (Eq, Ord, Show)
@ -146,7 +167,14 @@ newtype Port = Port { unPort :: Word16 }
-- @if -- @if
newtype Ipv4 = Ipv4 { unIpv4 :: Word32 } newtype Ipv4 = Ipv4 { unIpv4 :: Word32 }
deriving newtype (Eq, Ord, Show, Enum, Real, Integral, Num, ToNoun, FromNoun) deriving newtype (Eq, Ord, Enum, Real, Integral, Num, ToNoun, FromNoun)
instance Show Ipv4 where
show (Ipv4 i) =
show ((shiftL i 24) .&. 0xff) ++ "." ++
show ((shiftL i 16) .&. 0xff) ++ "." ++
show ((shiftL i 8) .&. 0xff) ++ "." ++
show (i .&. 0xff)
-- @is -- @is
newtype Ipv6 = Ipv6 { unIpv6 :: Word128 } newtype Ipv6 = Ipv6 { unIpv6 :: Word128 }

View File

@ -18,7 +18,7 @@ import Urbit.Arvo.Common (KingId(..), ServId(..))
import Urbit.Arvo.Common (Header, HttpEvent, HttpServerConf, Method, Mime) import Urbit.Arvo.Common (Header, HttpEvent, HttpServerConf, Method, Mime)
import Urbit.Arvo.Common (AmesDest, Turf) import Urbit.Arvo.Common (AmesDest, Turf)
import Urbit.Arvo.Common (ReOrg(..), reorgThroughNoun) import Urbit.Arvo.Common (ReOrg(..), reorgThroughNoun)
import Urbit.Arvo.Common (Desk) import Urbit.Arvo.Common (Desk, Wynn)
-- Newt Effects ---------------------------------------------------------------- -- Newt Effects ----------------------------------------------------------------
@ -259,20 +259,32 @@ data Ef
= EfVane VaneEf = EfVane VaneEf
| EfVega Cord EvilPath -- second path component, rest of path | EfVega Cord EvilPath -- second path component, rest of path
| EfExit Cord EvilPath -- second path component, rest of path | EfExit Cord EvilPath -- second path component, rest of path
| EfWend Wynn
deriving (Eq, Ord, Show) deriving (Eq, Ord, Show)
-- XX HACK
clip :: Noun -> Noun
clip (C (C _ x) y) = C x y
clip _ = error "misclip"
tack :: Noun -> Noun
tack (C x y) = C (C (A 0) x) y
tack _ = error "mistack"
instance ToNoun Ef where instance ToNoun Ef where
toNoun = \case toNoun = clip . \case
EfVane v -> toNoun $ reorgThroughNoun ("", v) EfVane v -> toNoun $ reorgThroughNoun ("", v)
EfExit s p -> toNoun $ ReOrg "" s "exit" p (A 0) EfExit s p -> toNoun $ ReOrg "" s "exit" p (A 0)
EfVega s p -> toNoun $ ReOrg "" s "vega" p (A 0) EfVega s p -> toNoun $ ReOrg "" s "vega" p (A 0)
EfWend w -> toNoun $ reorgThroughNoun ("", w)
instance FromNoun Ef where instance FromNoun Ef where
parseNoun = parseNoun >=> \case parseNoun = tack >>> parseNoun >=> \case
ReOrg "" s "exit" p (A 0) -> pure (EfExit s p) ReOrg "" s "exit" p (A 0) -> pure (EfExit s p)
ReOrg "" s "exit" p _ -> fail "%exit effect expects nil value" ReOrg "" s "exit" p _ -> fail "%exit effect expects nil value"
ReOrg "" s "vega" p (A 0) -> pure (EfVega s p) ReOrg "" s "vega" p (A 0) -> pure (EfVega s p)
ReOrg "" s "vega" p _ -> fail "%vega effect expects nil value" ReOrg "" s "vega" p _ -> fail "%vega effect expects nil value"
ReOrg "" s "wend" p val -> EfWend <$> parseNoun val
ReOrg "" s tag p val -> EfVane <$> parseNoun (toNoun (s, tag, p, val)) ReOrg "" s tag p val -> EfVane <$> parseNoun (toNoun (s, tag, p, val))
ReOrg _ _ _ _ _ -> fail "Non-empty first path-element" ReOrg _ _ _ _ _ -> fail "Non-empty first path-element"

View File

@ -9,10 +9,10 @@
-} -}
module Urbit.Arvo.Event where module Urbit.Arvo.Event where
import Urbit.Prelude hiding (Term) import Urbit.Prelude
import Control.Monad.Fail (fail) import Control.Monad.Fail (fail)
import Urbit.Arvo.Common (KingId(..), ServId(..)) import Urbit.Arvo.Common (KingId(..), ServId(..), Vere(..))
import Urbit.Arvo.Common (Desk, Mime) import Urbit.Arvo.Common (Desk, Mime)
import Urbit.Arvo.Common (Header(..), HttpEvent) import Urbit.Arvo.Common (Header(..), HttpEvent)
import Urbit.Arvo.Common (AmesDest, Ipv4, Ipv6, Port, Turf) import Urbit.Arvo.Common (AmesDest, Ipv4, Ipv6, Port, Turf)
@ -218,9 +218,12 @@ instance Show Entropy where
data ArvoEv data ArvoEv
= ArvoEvWhom () Ship = ArvoEvWhom () Ship
| ArvoEvWack () Entropy | ArvoEvWack () Entropy
| ArvoEvWarn Path Noun | ArvoEvWyrd () Vere
| ArvoEvCrud Path Noun | ArvoEvCrud Path Noun
| ArvoEvVeer Atom Noun | ArvoEvTrim UD
| ArvoEvWhat [Noun]
| ArvoEvWhey ()
| ArvoEvVerb (Maybe Bool)
deriving (Eq, Ord, Show) deriving (Eq, Ord, Show)
deriveNoun ''ArvoEv deriveNoun ''ArvoEv
@ -318,50 +321,29 @@ data BlipEv
deriveNoun ''BlipEv deriveNoun ''BlipEv
-- Boot Events -----------------------------------------------------------------
data Vane
= VaneVane VaneEv
| VaneZuse ZuseEv
deriving (Eq, Ord, Show)
data VaneName
= Ames | Behn | Clay | Dill | Eyre | Ford | Gall | Iris | Jael
deriving (Eq, Ord, Show, Enum, Bounded)
data ZuseEv
= ZEVeer () Cord Path BigCord
| ZEVoid Void
deriving (Eq, Ord, Show)
data VaneEv
= VEVeer (VaneName, ()) Cord Path BigCord
| VEVoid Void
deriving (Eq, Ord, Show)
deriveNoun ''Vane
deriveNoun ''VaneName
deriveNoun ''VaneEv
deriveNoun ''ZuseEv
-- The Main Event Type --------------------------------------------------------- -- The Main Event Type ---------------------------------------------------------
data Ev data Ev
= EvBlip BlipEv = EvBlip BlipEv
| EvVane Vane
deriving (Eq, Show) deriving (Eq, Show)
instance ToNoun Ev where instance ToNoun Ev where
toNoun = \case toNoun = toNoun . \case
EvBlip v -> toNoun $ reorgThroughNoun (Cord "", v) EvBlip v@BlipEvAmes{} -> reorgThroughNoun ("ames", v)
EvVane v -> toNoun $ reorgThroughNoun (Cord "vane", v) EvBlip v@BlipEvArvo{} -> reorgThroughNoun ("", v)
EvBlip v@BlipEvBehn{} -> reorgThroughNoun ("behn", v)
EvBlip v@BlipEvBoat{} -> reorgThroughNoun ("clay", v)
EvBlip v@BlipEvHttpClient{} -> reorgThroughNoun ("iris", v)
EvBlip v@BlipEvHttpServer{} -> reorgThroughNoun ("eyre", v)
EvBlip v@BlipEvNewt{} -> reorgThroughNoun ("ames", v)
EvBlip v@BlipEvSync{} -> reorgThroughNoun ("clay", v)
EvBlip v@BlipEvTerm{} -> reorgThroughNoun ("dill", v)
-- XX We really should check the first path element, but since this is used only
-- in the event browser, which otherwise is broken, I don't care right now.
instance FromNoun Ev where instance FromNoun Ev where
parseNoun = parseNoun >=> \case parseNoun = parseNoun >=> \case
ReOrg "" s t p v -> fmap EvBlip $ parseNoun $ toNoun (s,t,p,v) ReOrg _ s t p v -> fmap EvBlip $ parseNoun $ toNoun (s,t,p,v)
ReOrg "vane" s t p v -> fmap EvVane $ parseNoun $ toNoun (s,t,p,v)
ReOrg _ _ _ _ _ -> fail "First path-elem must be ?($ %vane)"
-- Short Event Names ----------------------------------------------------------- -- Short Event Names -----------------------------------------------------------
@ -373,7 +355,6 @@ instance FromNoun Ev where
-} -}
getSpinnerNameForEvent :: Ev -> Maybe Text getSpinnerNameForEvent :: Ev -> Maybe Text
getSpinnerNameForEvent = \case getSpinnerNameForEvent = \case
EvVane _ -> Nothing
EvBlip b -> case b of EvBlip b -> case b of
BlipEvAmes _ -> Just "ames" BlipEvAmes _ -> Just "ames"
BlipEvArvo _ -> Just "arvo" BlipEvArvo _ -> Just "arvo"

View File

@ -4,6 +4,7 @@
module Urbit.King.App module Urbit.King.App
( KingEnv ( KingEnv
, runKingEnvStderr , runKingEnvStderr
, runKingEnvStderrRaw
, runKingEnvLogFile , runKingEnvLogFile
, runKingEnvNoLog , runKingEnvNoLog
, kingEnvKillSignal , kingEnvKillSignal
@ -29,6 +30,7 @@ where
import Urbit.King.Config import Urbit.King.Config
import Urbit.Prelude import Urbit.Prelude
import RIO (logGeneric)
import System.Directory ( createDirectoryIfMissing import System.Directory ( createDirectoryIfMissing
, getXdgDirectory , getXdgDirectory
, XdgDirectory(XdgCache) , XdgDirectory(XdgCache)
@ -90,6 +92,22 @@ runKingEnvStderr verb lvl inner = do
<&> setLogMinLevel lvl <&> setLogMinLevel lvl
withLogFunc logOptions $ \logFunc -> runKingEnv logFunc logFunc inner withLogFunc logOptions $ \logFunc -> runKingEnv logFunc logFunc inner
runKingEnvStderrRaw :: Bool -> LogLevel -> RIO KingEnv a -> IO a
runKingEnvStderrRaw verb lvl inner = do
logOptions <-
logOptionsHandle stderr verb
<&> setLogUseTime True
<&> setLogUseLoc False
<&> setLogMinLevel lvl
withLogFunc logOptions $ \logFunc ->
let lf = wrapCarriage logFunc
in runKingEnv lf lf inner
-- XX loses callstack
wrapCarriage :: LogFunc -> LogFunc
wrapCarriage lf = mkLogFunc $ \_ ls ll bldr ->
runRIO lf $ logGeneric ls ll (bldr <> "\r")
runKingEnvLogFile :: Bool -> LogLevel -> Maybe FilePath -> RIO KingEnv a -> IO a runKingEnvLogFile :: Bool -> LogLevel -> Maybe FilePath -> RIO KingEnv a -> IO a
runKingEnvLogFile verb lvl fileM inner = do runKingEnvLogFile verb lvl fileM inner = do
logFile <- case fileM of logFile <- case fileM of
@ -119,7 +137,8 @@ defaultLogFile :: IO FilePath
defaultLogFile = do defaultLogFile = do
logDir <- getXdgDirectory XdgCache "urbit" logDir <- getXdgDirectory XdgCache "urbit"
createDirectoryIfMissing True logDir createDirectoryIfMissing True logDir
pure (logDir </> "king.log") logId :: Word32 <- randomIO
pure (logDir </> "king-" <> show logId <> ".log")
runKingEnvNoLog :: RIO KingEnv a -> IO a runKingEnvNoLog :: RIO KingEnv a -> IO a
runKingEnvNoLog act = runKingEnv mempty mempty act runKingEnvNoLog act = runKingEnv mempty mempty act

View File

@ -382,7 +382,7 @@ replayPartEvs top last = do
{-| {-|
Interesting Interesting
-} -}
testPill :: HasLogFunc e => FilePath -> Bool -> Bool -> RIO e () testPill :: HasKingEnv e => FilePath -> Bool -> Bool -> RIO e ()
testPill pax showPil showSeq = do testPill pax showPil showSeq = do
logInfo "Reading pill file." logInfo "Reading pill file."
pillBytes <- readFile pax pillBytes <- readFile pax
@ -678,10 +678,13 @@ main = do
runKingEnv args log = runKingEnv args log =
let let
verb = verboseLogging args verb = verboseLogging args
runStderr = case args of
CLI.CmdRun {} -> runKingEnvStderrRaw
_ -> runKingEnvStderr
CLI.Log {..} = log CLI.Log {..} = log
in case logTarget lTarget args of in case logTarget lTarget args of
CLI.LogFile f -> runKingEnvLogFile verb lLevel f CLI.LogFile f -> runKingEnvLogFile verb lLevel f
CLI.LogStderr -> runKingEnvStderr verb lLevel CLI.LogStderr -> runStderr verb lLevel
CLI.LogOff -> runKingEnvNoLog CLI.LogOff -> runKingEnvNoLog
setupSignalHandlers = do setupSignalHandlers = do

View File

@ -2,30 +2,32 @@
Scry helpers Scry helpers
-} -}
module Urbit.King.Scry (scryNow) where module Urbit.King.Scry
( scryNow
, module Urbit.Vere.Pier.Types
)
where
import Urbit.Prelude import Urbit.Prelude
import Urbit.Vere.Serf.Types import Urbit.Vere.Serf.Types
import qualified Urbit.Noun.Time as Time import Urbit.Arvo.Common (Desk)
import Urbit.Vere.Pier.Types (ScryFunc)
scryNow :: forall e n scryNow :: forall e n
. (HasLogFunc e, FromNoun n) . (HasLogFunc e, FromNoun n)
=> (Time.Wen -> Gang -> Path -> IO (Maybe (Term, Noun))) => ScryFunc
-> Text -- ^ vane + care as two-letter string -> Term -- ^ vane + care as two-letter string
-> Ship -- ^ ship in scry path, usually the local ship -> Desk -- ^ desk in scry path
-> Text -- ^ desk in scry path
-> [Text] -- ^ resource path to scry for -> [Text] -- ^ resource path to scry for
-> RIO e (Maybe n) -> RIO e (Maybe n)
scryNow scry vare ship desk path = do scryNow scry vare desk path =
env <- ask io (scry Nothing (EachNo $ DemiOnce vare desk (Path $ MkKnot <$> path)))
wen <- io Time.now >>= \case
let wan = tshow $ Time.MkDate wen Just ("omen", fromNoun @(Path, Term, n) -> Just (_,_,v)) -> pure $ Just v
let pax = Path $ fmap MkKnot $ vare : (tshow ship) : desk : wan : path Just (_, fromNoun @n -> Just v) -> pure $ Just v
io (scry wen Nothing pax) >>= \case Just (_, n) -> do
Just (_, fromNoun @n -> Just v) -> pure $ Just v logError $ displayShow ("uncanny scry result", vare, path, n)
Just (_, n) -> do pure Nothing
logError $ displayShow ("uncanny scry result", vare, pax, n) Nothing -> pure Nothing
pure Nothing
Nothing -> pure Nothing

View File

@ -16,6 +16,7 @@ module Urbit.Prelude
, io, rio , io, rio
, logTrace , logTrace
, acquireWorker, acquireWorkerBound , acquireWorker, acquireWorkerBound
, hark
) where ) where
import ClassyPrelude import ClassyPrelude
@ -38,6 +39,8 @@ import RIO (HasLogFunc, LogFunc, LogLevel(..), logDebug, logError, logFuncL,
logInfo, logOptionsHandle, logOther, logWarn, mkLogFunc, logInfo, logOptionsHandle, logOther, logWarn, mkLogFunc,
setLogMinLevel, setLogUseLoc, setLogUseTime, withLogFunc) setLogMinLevel, setLogUseLoc, setLogUseTime, withLogFunc)
import qualified RIO
io :: MonadIO m => IO a -> m a io :: MonadIO m => IO a -> m a
io = liftIO io = liftIO
@ -47,6 +50,9 @@ rio = liftRIO
logTrace :: HasLogFunc e => Utf8Builder -> RIO e () logTrace :: HasLogFunc e => Utf8Builder -> RIO e ()
logTrace = logOther "trace" logTrace = logOther "trace"
-- | Composes a log message out of textual components.
hark :: [Text] -> Utf8Builder
hark = RIO.displayBytesUtf8 . foldMap encodeUtf8
-- Utils for Spawning Worker Threads ------------------------------------------- -- Utils for Spawning Worker Threads -------------------------------------------

View File

@ -24,8 +24,7 @@ import Urbit.King.App (HasKingId(..), HasPierEnv(..))
import Urbit.Vere.Ames.DNS (NetworkMode(..), ResolvServ(..)) import Urbit.Vere.Ames.DNS (NetworkMode(..), ResolvServ(..))
import Urbit.Vere.Ames.DNS (galaxyPort, resolvServ) import Urbit.Vere.Ames.DNS (galaxyPort, resolvServ)
import Urbit.Vere.Ames.UDP (UdpServ(..), fakeUdpServ, realUdpServ) import Urbit.Vere.Ames.UDP (UdpServ(..), fakeUdpServ, realUdpServ)
import Urbit.Vere.Stat (AmesStat(..), bump, bump')
import qualified Urbit.Noun.Time as Time
-- Constants ------------------------------------------------------------------- -- Constants -------------------------------------------------------------------
@ -47,7 +46,6 @@ type Version = Word8
data AmesDrv = AmesDrv data AmesDrv = AmesDrv
{ aTurfs :: TVar (Maybe [Turf]) { aTurfs :: TVar (Maybe [Turf])
, aDropped :: TVar Word
, aVersion :: TVar (Maybe Version) , aVersion :: TVar (Maybe Version)
, aUdpServ :: UdpServ , aUdpServ :: UdpServ
, aResolvr :: ResolvServ , aResolvr :: ResolvServ
@ -125,13 +123,14 @@ udpPort isFake who = do
udpServ :: (HasLogFunc e, HasNetworkConfig e, HasPortControlApi e) udpServ :: (HasLogFunc e, HasNetworkConfig e, HasPortControlApi e)
=> Bool => Bool
-> Ship -> Ship
-> AmesStat
-> RIO e UdpServ -> RIO e UdpServ
udpServ isFake who = do udpServ isFake who stat = do
mode <- netMode isFake mode <- netMode isFake
port <- udpPort isFake who port <- udpPort isFake who
case modeAddress mode of case modeAddress mode of
Nothing -> fakeUdpServ Nothing -> fakeUdpServ
Just host -> realUdpServ port host Just host -> realUdpServ port host stat
_bornFailed :: e -> WorkError -> IO () _bornFailed :: e -> WorkError -> IO ()
_bornFailed env _ = runRIO env $ do _bornFailed env _ = runRIO env $ do
@ -141,10 +140,11 @@ ames'
:: HasPierEnv e :: HasPierEnv e
=> Ship => Ship
-> Bool -> Bool
-> (Time.Wen -> Gang -> Path -> IO (Maybe (Term, Noun))) -> AmesStat
-> ScryFunc
-> (Text -> RIO e ()) -> (Text -> RIO e ())
-> RIO e ([Ev], RAcquire e (DriverApi NewtEf)) -> RIO e ([Ev], RAcquire e (DriverApi NewtEf))
ames' who isFake scry stderr = do ames' who isFake stat scry stderr = do
-- Unfortunately, we cannot use TBQueue because the only behavior -- Unfortunately, we cannot use TBQueue because the only behavior
-- provided for when full is to block the writer. The implementation -- provided for when full is to block the writer. The implementation
-- below uses materially the same data structures as TBQueue, however. -- below uses materially the same data structures as TBQueue, however.
@ -164,11 +164,11 @@ ames' who isFake scry stderr = do
pure Ouster pure Ouster
dequeuePacket = do dequeuePacket = do
pM <- tryReadTQueue ventQ pM <- tryReadTQueue ventQ
when (isJust pM) $ modifyTVar avail (+ 1) when (isJust pM) $ modifyTVar' avail (+ 1)
pure pM pure pM
env <- ask env <- ask
let (bornEvs, startDriver) = ames env who isFake scry enqueuePacket stderr let (bornEvs, startDriver) = ames env who isFake stat scry enqueuePacket stderr
let runDriver = do let runDriver = do
diOnEffect <- startDriver diOnEffect <- startDriver
@ -195,11 +195,12 @@ ames
=> e => e
-> Ship -> Ship
-> Bool -> Bool
-> (Time.Wen -> Gang -> Path -> IO (Maybe (Term, Noun))) -> AmesStat
-> ScryFunc
-> (EvErr -> STM PacketOutcome) -> (EvErr -> STM PacketOutcome)
-> (Text -> RIO e ()) -> (Text -> RIO e ())
-> ([Ev], RAcquire e (NewtEf -> IO ())) -> ([Ev], RAcquire e (NewtEf -> IO ()))
ames env who isFake scry enqueueEv stderr = (initialEvents, runAmes) ames env who isFake stat scry enqueueEv stderr = (initialEvents, runAmes)
where where
king = fromIntegral (env ^. kingIdL) king = fromIntegral (env ^. kingIdL)
@ -218,21 +219,28 @@ ames env who isFake scry enqueueEv stderr = (initialEvents, runAmes)
cachedScryLane <- cache scryLane cachedScryLane <- cache scryLane
aTurfs <- newTVarIO Nothing aTurfs <- newTVarIO Nothing
aDropped <- newTVarIO 0
aVersion <- newTVarIO Nothing aVersion <- newTVarIO Nothing
aVersTid <- trackVersionThread aVersion aVersTid <- trackVersionThread aVersion
aUdpServ <- udpServ isFake who aUdpServ <- udpServ isFake who stat
aResolvr <- resolvServ aTurfs (usSend aUdpServ) stderr aResolvr <- resolvServ aTurfs (usSend aUdpServ) stderr
aRecvTid <- queuePacketsThread aRecvTid <- queuePacketsThread
aDropped
aVersion aVersion
cachedScryLane cachedScryLane
(send aUdpServ aResolvr mode) (send aUdpServ aResolvr mode)
aUdpServ aUdpServ
stat
pure (AmesDrv { .. }) pure (AmesDrv { .. })
hearFailed _ = pure () hearFailed AmesStat {..} = runRIO env . \case
RunSwap{} -> bump asSwp
RunBail gs -> do
for gs \(t, es) ->
for es \e ->
logWarn $ hark
["ames: goof: ", unTerm t, ": ", tankToText e]
bump asBal
RunOkay{} -> bump asOky
trackVersionThread :: HasLogFunc e => TVar (Maybe Version) -> RIO e (Async ()) trackVersionThread :: HasLogFunc e => TVar (Maybe Version) -> RIO e (Async ())
trackVersionThread versSlot = async $ forever do trackVersionThread versSlot = async $ forever do
@ -249,34 +257,43 @@ ames env who isFake scry enqueueEv stderr = (initialEvents, runAmes)
threadDelay (10 * 60 * 1_000_000) -- 10m threadDelay (10 * 60 * 1_000_000) -- 10m
queuePacketsThread :: HasLogFunc e queuePacketsThread :: HasLogFunc e
=> TVar Word => TVar (Maybe Version)
-> TVar (Maybe Version)
-> (Ship -> RIO e (Maybe [AmesDest])) -> (Ship -> RIO e (Maybe [AmesDest]))
-> (AmesDest -> ByteString -> RIO e ()) -> (AmesDest -> ByteString -> RIO e ())
-> UdpServ -> UdpServ
-> AmesStat
-> RIO e (Async ()) -> RIO e (Async ())
queuePacketsThread dropCtr vers lan forward UdpServ{..} = async $ forever $ do queuePacketsThread vers lan forward UdpServ{..} s@(AmesStat{..}) = async $ forever $ do
-- port number, host address, bytestring -- port number, host address, bytestring
(p, a, b) <- atomically usRecv (p, a, b) <- atomically (bump' asRcv >> usRecv)
ver <- readTVarIO vers ver <- readTVarIO vers
case decode b of case decode b of
Right (pkt@Packet {..}) | ver == Nothing || ver == Just pktVersion -> do Right (pkt@Packet {..}) | ver == Nothing || ver == Just pktVersion -> do
logDebug $ displayShow ("ames: bon packet", pkt, showUD $ bytesAtom b) logDebug $ displayShow ("ames: bon packet", pkt, showUD $ bytesAtom b)
if pktRcvr == who if pktRcvr == who
then serfsUp p a b then do
bump asSup
serfsUp p a b
else lan pktRcvr >>= \case else lan pktRcvr >>= \case
Just ls Just ls
| dest:_ <- filter notSelf ls | dest:_ <- filter notSelf ls
-> forward dest $ encode pkt -> do
{ pktOrigin = pktOrigin <|> Just (ipDest p a) } bump asFwd
forward dest $ encode pkt
{ pktOrigin = pktOrigin
<|> Just (AAIpv4 (Ipv4 a) (fromIntegral p)) }
where where
notSelf (EachYes g) = who /= Ship (fromIntegral g) notSelf (EachYes g) = who /= Ship (fromIntegral g)
notSelf (EachNo _) = True notSelf (EachNo _) = True
_ -> logInfo $ displayShow ("ames: dropping unroutable", pkt)
Right pkt -> logInfo $ displayShow ("ames: dropping ill-versed", pkt, ver) _ -> do
bump asDrt
logInfo $ displayShow ("ames: dropping unroutable", pkt)
Right pkt -> do
bump asDvr
logInfo $ displayShow ("ames: dropping ill-versed", pkt, ver)
-- XX better handle misversioned or illegible packets. -- XX better handle misversioned or illegible packets.
-- Remarks from 67f06ce5, pkg/urbit/vere/io/ames.c, L1010: -- Remarks from 67f06ce5, pkg/urbit/vere/io/ames.c, L1010:
@ -293,18 +310,19 @@ ames env who isFake scry enqueueEv stderr = (initialEvents, runAmes)
-- trigger printfs suggesting upgrade. -- trigger printfs suggesting upgrade.
-- they cannot be filtered, as we do not know their semantics -- they cannot be filtered, as we do not know their semantics
-- --
Left e -> logInfo $ displayShow ("ames: dropping malformed", e) Left e -> do
bump asDml
logInfo $ displayShow ("ames: dropping malformed", e)
where where
serfsUp p a b = serfsUp p a b =
atomically (enqueueEv (EvErr (hearEv p a b) hearFailed)) >>= \case atomically (enqueueEv (EvErr (hearEv p a b) (hearFailed s))) >>= \case
Intake -> pure () Intake -> bump asSrf
Ouster -> do Ouster -> do
d <- atomically $ do d <- atomically $ do
d <- readTVar dropCtr bump' asQuf
writeTVar dropCtr (d + 1) readTVar asQuf
pure d when (d `rem` packetsDroppedPerComplaint == 1) $
when (d `rem` packetsDroppedPerComplaint == 0) $
logWarn "ames: queue full; dropping inbound packets" logWarn "ames: queue full; dropping inbound packets"
stop :: forall e. AmesDrv -> RIO e () stop :: forall e. AmesDrv -> RIO e ()
@ -342,12 +360,12 @@ ames env who isFake scry enqueueEv stderr = (initialEvents, runAmes)
EachNo addr -> to (ipv4Addr addr) EachNo addr -> to (ipv4Addr addr)
scryVersion :: HasLogFunc e => RIO e (Maybe Version) scryVersion :: HasLogFunc e => RIO e (Maybe Version)
scryVersion = scryNow scry "ax" who "" ["protocol", "version"] scryVersion = scryNow scry "ax" "" ["protocol", "version"]
scryLane :: HasLogFunc e scryLane :: HasLogFunc e
=> Ship => Ship
-> RIO e (Maybe [AmesDest]) -> RIO e (Maybe [AmesDest])
scryLane ship = scryNow scry "ax" who "" ["peers", tshow ship, "forward-lane"] scryLane ship = scryNow scry "ax" "" ["peers", tshow ship, "forward-lane"]
ipv4Addr (Jammed (AAVoid v )) = absurd v ipv4Addr (Jammed (AAVoid v )) = absurd v
ipv4Addr (Jammed (AAIpv4 a p)) = SockAddrInet (fromIntegral p) (unIpv4 a) ipv4Addr (Jammed (AAIpv4 a p)) = SockAddrInet (fromIntegral p) (unIpv4 a)

View File

@ -9,18 +9,19 @@ import Urbit.Prelude
import Control.Monad.Fail import Control.Monad.Fail
import Data.Bits import Data.Bits
import Data.LargeWord import Data.LargeWord
import Data.List (genericIndex)
import Data.Serialize import Data.Serialize
import Urbit.Arvo (AmesDest) import Urbit.Arvo (AmesAddress(..), Ipv4(..), Port(..))
data Packet = Packet data Packet = Packet
{ pktVersion :: Word8 { pktVersion :: Word3
, pktEncrypted :: Bool , pktSndr :: Ship
-- , pktRcvr :: Ship
, pktSndr :: Ship , pktSndrTick :: Word4
, pktRcvr :: Ship , pktRcvrTick :: Word4
, pktOrigin :: Maybe AmesDest , pktOrigin :: Maybe AmesAddress
, pktContent :: Bytes , pktContent :: ByteString
} }
deriving Eq deriving Eq
@ -28,73 +29,140 @@ instance Show Packet where
show Packet {..} show Packet {..}
= "Packet {pktVersion = " = "Packet {pktVersion = "
<> show pktVersion <> show pktVersion
<> ", pktEncrypted = "
<> show pktEncrypted
<> ", pktSndr = " <> ", pktSndr = "
<> show pktSndr <> show pktSndr
<> ", pktRcvr = " <> ", pktRcvr = "
<> show pktRcvr <> show pktRcvr
<> ", pktSndrTick = "
<> show pktSndrTick
<> ", pktRcvrTick = "
<> show pktRcvrTick
<> ", pktOrigin = " <> ", pktOrigin = "
<> show pktOrigin <> show pktOrigin
<> ", pktContent = " <> ", pktContent = "
<> showUD (bytesAtom $ unBytes pktContent) <> showUD (bytesAtom pktContent)
<> "}" <> "}"
{-
-- Wire format
data PacketHeader = PacketHeader
{ pktIsAmes :: Bool -- sim_o
, pktVersion :: Word3 -- ver_y
, pktSndrClass :: ShipClass -- sac_y
, pktRcvrClass :: ShipClass -- rac_y
, pktChecksum :: Word20 -- mug_l
, pktIsRelayed :: Bool -- rel_o
}
deriving Eq
data PacketBody = PacketBody
{ pktSndr :: Ship -- sen_d
, pktRcvr :: Ship -- rec_d
, pktSndrTick :: Word4 -- sic_y
, pktRcvrTick :: Word4 -- ric_y
, pktContent :: ByteString -- (con_s, con_y)
, pktOrigin :: Maybe AmesAddress -- rog_d
}
deriving Eq
-}
type Word3 = Word8
type Word4 = Word8
type Word20 = Word32
data ShipClass
= Lord
| Planet
| Moon
| Comet
deriving (Eq, Show)
muk :: ByteString -> Word20
muk bs = mugBS bs .&. (2 ^ 20 - 1)
-- XX check this
getAmesAddress :: Get AmesAddress
getAmesAddress = AAIpv4 <$> (Ipv4 <$> getWord32le) <*> (Port <$> getWord16le)
putAmesAddress :: Putter AmesAddress
putAmesAddress = \case
AAIpv4 (Ipv4 ip) (Port port) -> putWord32le ip >> putWord16le port
instance Serialize Packet where instance Serialize Packet where
get = do get = do
-- header -- header
head <- getWord32le head <- getWord32le
let pktVersion = head .&. 0b111 & fromIntegral -- skip first three bits
let checksum = shiftR head 3 .&. (2 ^ 20 - 1) let isAmes = testBit head 3 & not
let sndrRank = shiftR head 23 .&. 0b11 let pktVersion = shiftR head 4 .&. 0b111 & fromIntegral
let rcvrRank = shiftR head 25 .&. 0b11 let sndrRank = shiftR head 7 .&. 0b11
let pktEncrypted = testBit head 27 & not -- loobean let rcvrRank = shiftR head 9 .&. 0b11
-- verify checksum let checksum = shiftR head 11 .&. (2 ^ 20 - 1)
let isRelayed = testBit head 31 & not -- loobean
let sndrClass = genericIndex [Lord, Planet, Moon, Comet] sndrRank
let rcvrClass = genericIndex [Lord, Planet, Moon, Comet] rcvrRank
guard isAmes
pktOrigin <- if isRelayed
then Just <$> getAmesAddress
else pure Nothing
-- body
lookAhead $ do lookAhead $ do
len <- remaining len <- remaining
body <- getBytes len body <- getBytes len
let chk = fromIntegral (mugBS body) .&. (2 ^ 20 - 1) let chk = muk body
when (checksum /= chk) $ when (checksum /= chk) $
fail ("checksum mismatch: expected " <> show checksum fail ("checksum mismatch: expected " <> show checksum
<> "; got " <> show chk) <> "; got " <> show chk)
-- body
pktSndr <- getShip sndrRank tick <- getWord8
pktRcvr <- getShip rcvrRank let pktSndrTick = tick .&. 0b1111
len <- remaining let pktRcvrTick = shiftR tick 4
payload <- getBytes len
-- data ("payload") pktSndr <- getShip sndrClass
(pktOrigin, pktContent) <- case cueBS payload of pktRcvr <- getShip rcvrClass
Left e -> fail (show e)
Right n -> case fromNounErr n of len <- remaining
Left e -> fail (show e) pktContent <- getBytes len
Right c -> pure c
pure Packet {..} pure Packet{..}
where where
getShip = fmap Ship . \case getShip = fmap Ship . \case
0 -> fromIntegral <$> getWord16le -- galaxy / star Lord -> fromIntegral <$> getWord16le
1 -> fromIntegral <$> getWord32le -- planet Planet -> fromIntegral <$> getWord32le
2 -> fromIntegral <$> getWord64le -- moon Moon -> fromIntegral <$> getWord64le
3 -> LargeKey <$> getWord64le <*> getWord64le -- comet Comet -> LargeKey <$> getWord64le <*> getWord64le
_ -> fail "impossibiru"
put Packet {..} = do put Packet{..} = do
let load = jamBS $ toNoun (pktOrigin, pktContent)
let (sndR, putSndr) = putShipGetRank pktSndr let (sndR, putSndr) = putShipGetRank pktSndr
let (rcvR, putRcvr) = putShipGetRank pktRcvr let (rcvR, putRcvr) = putShipGetRank pktRcvr
let body = runPut (putSndr <> putRcvr <> putByteString load)
let chek = fromIntegral (mugBS body) .&. (2 ^ 20 - 1) let body = runPut $ do
let encr = pktEncrypted putWord8 $ (pktSndrTick .&. 0b1111)
.|. shiftL (pktRcvrTick .&. 0b1111) 4
putSndr
putRcvr
putByteString pktContent
let vers = fromIntegral pktVersion .&. 0b111 let vers = fromIntegral pktVersion .&. 0b111
let head = vers let chek = muk body
.|. shiftL chek 3
.|. shiftL sndR 23 -- skip first 3 bytes, set 4th to yes (0) for "is ames"
.|. shiftL rcvR 25 let head = shiftL vers 4
.|. if encr then 0 else bit 27 .|. shiftL sndR 7
.|. shiftL rcvR 9
.|. shiftL chek 11
.|. if isJust pktOrigin then 0 else bit 31
putWord32le head putWord32le head
putByteString body -- XX can we avoid copy? case pktOrigin of
Just o -> putAmesAddress o
Nothing -> pure ()
putByteString body
where where
putShipGetRank s@(Ship (LargeKey p q)) = case () of putShipGetRank s@(Ship (LargeKey p q)) = case () of
_ | s < 2 ^ 16 -> (0, putWord16le $ fromIntegral s) -- gar _ | s < 2 ^ 16 -> (0, putWord16le $ fromIntegral s) -- lord
| s < 2 ^ 32 -> (1, putWord32le $ fromIntegral s) -- pan | s < 2 ^ 32 -> (1, putWord32le $ fromIntegral s) -- planet
| s < 2 ^ 64 -> (2, putWord64le $ fromIntegral s) -- mon | s < 2 ^ 64 -> (2, putWord64le $ fromIntegral s) -- moon
| otherwise -> (3, putWord64le p >> putWord64le q) -- com | otherwise -> (3, putWord64le p >> putWord64le q) -- comet

View File

@ -39,7 +39,7 @@ import Network.Socket
import Control.Monad.STM (retry) import Control.Monad.STM (retry)
import Network.Socket.ByteString (recvFrom, sendTo) import Network.Socket.ByteString (recvFrom, sendTo)
import Urbit.Vere.Stat (AmesStat(..), bump)
-- Types ----------------------------------------------------------------------- -- Types -----------------------------------------------------------------------
@ -156,8 +156,9 @@ realUdpServ
. (HasLogFunc e, HasPortControlApi e) . (HasLogFunc e, HasPortControlApi e)
=> PortNumber => PortNumber
-> HostAddress -> HostAddress
-> AmesStat
-> RIO e UdpServ -> RIO e UdpServ
realUdpServ por hos = do realUdpServ por hos sat = do
logInfo $ displayShow ("AMES", "UDP", "Starting real UDP server.") logInfo $ displayShow ("AMES", "UDP", "Starting real UDP server.")
env <- ask env <- ask
@ -192,6 +193,7 @@ realUdpServ por hos = do
enqueueRecvPacket p a b = do enqueueRecvPacket p a b = do
did <- atomically (tryWriteTBQueue qRecv (p, a, b)) did <- atomically (tryWriteTBQueue qRecv (p, a, b))
when (did == False) $ do when (did == False) $ do
bump (asUqf sat)
logWarn $ displayShow $ ("AMES", "UDP",) logWarn $ displayShow $ ("AMES", "UDP",)
"Dropping inbound packet because queue is full." "Dropping inbound packet because queue is full."
@ -232,13 +234,16 @@ realUdpServ por hos = do
Just sk -> do Just sk -> do
recvPacket sk >>= \case recvPacket sk >>= \case
Left exn -> do Left exn -> do
bump (asUdf sat)
logError "AMES: UDP: Failed to receive packet" logError "AMES: UDP: Failed to receive packet"
signalBrokenSocket sk signalBrokenSocket sk
Right Nothing -> do Right Nothing -> do
bump (asUi6 sat)
logError "AMES: UDP: Dropping non-ipv4 packet" logError "AMES: UDP: Dropping non-ipv4 packet"
pure () pure ()
Right (Just (b, p, a)) -> do Right (Just (b, p, a)) -> do
logDebug "AMES: UDP: Received packet." logDebug "AMES: UDP: Received packet."
bump (asUdp sat)
enqueueRecvPacket p a b enqueueRecvPacket p a b
let shutdown = do let shutdown = do

View File

@ -10,7 +10,7 @@ module Urbit.Vere.Behn (behn, DriverApi(..), behn') where
import Data.Time.Clock.System (SystemTime) import Data.Time.Clock.System (SystemTime)
import Urbit.Arvo hiding (Behn) import Urbit.Arvo
import Urbit.Prelude import Urbit.Prelude
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types

View File

@ -13,16 +13,15 @@ import Urbit.Prelude hiding (Builder)
import Data.ByteString.Builder import Data.ByteString.Builder
import Urbit.King.Scry import Urbit.King.Scry
import Urbit.Vere.Serf.Types
import Data.Conduit (ConduitT, Flush(..), yield) import Data.Conduit (ConduitT, Flush(..), yield)
import Data.Text.Encoding (encodeUtf8Builder) import Data.Text.Encoding (encodeUtf8Builder)
import Urbit.Vere.Stat (RenderedStat)
import qualified Data.Text.Encoding as E import qualified Data.Text.Encoding as E
import qualified Network.HTTP.Types as H import qualified Network.HTTP.Types as H
import qualified Network.Wai as W import qualified Network.Wai as W
import qualified Network.Wai.Conduit as W import qualified Network.Wai.Conduit as W
import qualified Urbit.Noun.Time as Time
newtype KingSubsite = KS { runKingSubsite :: W.Application } newtype KingSubsite = KS { runKingSubsite :: W.Application }
@ -43,10 +42,11 @@ streamSlog a = do
kingSubsite :: HasLogFunc e kingSubsite :: HasLogFunc e
=> Ship => Ship
-> (Time.Wen -> Gang -> Path -> IO (Maybe (Term, Noun))) -> ScryFunc
-> IO RenderedStat
-> TVar ((Atom, Tank) -> IO ()) -> TVar ((Atom, Tank) -> IO ())
-> RAcquire e KingSubsite -> RAcquire e KingSubsite
kingSubsite who scry func = do kingSubsite who scry stat func = do
clients <- newTVarIO (mempty :: Map Word (SlogAction -> IO ())) clients <- newTVarIO (mempty :: Map Word (SlogAction -> IO ()))
nextId <- newTVarIO (0 :: Word) nextId <- newTVarIO (0 :: Word)
baton <- newTMVarIO () baton <- newTMVarIO ()
@ -77,15 +77,29 @@ kingSubsite who scry func = do
else else
let loop = yield Flush let loop = yield Flush
>> forever (atomically (readTQueue q) >>= streamSlog) >> forever (atomically (readTQueue q) >>= streamSlog)
in respond $ W.responseSource (H.mkStatus 200 "OK") heads loop) in respond $ W.responseSource (H.mkStatus 200 "OK") slogHeads loop)
["~_~", "stat"] -> do
authed <- authenticated env req
if not authed
then respond $ emptyResponse 403 "Permission Denied"
else do
lines <- stat
let msg = mconcat ((<> "\n") . encodeUtf8Builder <$> lines)
<> "\nRefresh for more current data."
respond $ W.responseBuilder (H.mkStatus 200 "OK") statHeads msg
_ -> respond $ emptyResponse 404 "Not Found" _ -> respond $ emptyResponse 404 "Not Found"
where where
heads = [ ("Content-Type" , "text/event-stream") slogHeads = [ ("Content-Type", "text/event-stream")
, ("Cache-Control", "no-cache") , ("Cache-Control", "no-cache")
, ("Connection" , "keep-alive") , ("Connection", "keep-alive")
] ]
statHeads = [ ("Content-Type", "text/plain")
, ("Cache-Control", "no-cache")
]
emptyResponse cod mes = W.responseLBS (H.mkStatus cod mes) [] "" emptyResponse cod mes = W.responseLBS (H.mkStatus cod mes) [] ""
@ -102,7 +116,7 @@ kingSubsite who scry func = do
=> Text => Text
-> RIO e (Maybe Bool) -> RIO e (Maybe Bool)
scryAuth cookie = scryAuth cookie =
scryNow scry "ex" who "" ["authenticated", "cookie", textAsTa cookie] scryNow scry "ex" "" ["authenticated", "cookie", textAsTa cookie]
fourOhFourSubsite :: Ship -> KingSubsite fourOhFourSubsite :: Ship -> KingSubsite
fourOhFourSubsite who = KS $ \req respond -> fourOhFourSubsite who = KS $ \req respond ->

View File

@ -23,6 +23,7 @@ import RIO.Directory
import Urbit.Arvo import Urbit.Arvo
import Urbit.King.App import Urbit.King.App
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Urbit.Vere.Stat
import Control.Monad.STM (retry) import Control.Monad.STM (retry)
import System.Environment (getExecutablePath) import System.Environment (getExecutablePath)
@ -31,11 +32,11 @@ import System.Posix.Files (ownerModes, setFileMode)
import Urbit.EventLog.LMDB (EventLog) import Urbit.EventLog.LMDB (EventLog)
import Urbit.EventLog.Event (buildLogEvent) import Urbit.EventLog.Event (buildLogEvent)
import Urbit.King.API (TermConn) import Urbit.King.API (TermConn)
import Urbit.Noun.Time (Wen)
import Urbit.TermSize (TermSize(..), termSize) import Urbit.TermSize (TermSize(..), termSize)
import Urbit.Vere.Serf (Serf) import Urbit.Vere.Serf (Serf)
import qualified Data.Text as T import qualified Data.Text as T
import qualified Data.List as L
import qualified System.Entropy as Ent import qualified System.Entropy as Ent
import qualified Urbit.EventLog.LMDB as Log import qualified Urbit.EventLog.LMDB as Log
import qualified Urbit.King.API as King import qualified Urbit.King.API as King
@ -71,16 +72,22 @@ setupPierDirectory shipPath = do
-- Load pill into boot sequence. ----------------------------------------------- -- Load pill into boot sequence. -----------------------------------------------
data CannotBootFromIvoryPill = CannotBootFromIvoryPill
deriving (Show, Exception)
genEntropy :: MonadIO m => m Entropy genEntropy :: MonadIO m => m Entropy
genEntropy = Entropy . fromIntegral . bytesAtom <$> io (Ent.getEntropy 64) genEntropy = Entropy . fromIntegral . bytesAtom <$> io (Ent.getEntropy 64)
genBootSeq :: MonadIO m => Ship -> Pill -> Bool -> LegacyBootEvent -> m BootSeq genBootSeq :: HasKingEnv e
genBootSeq ship Pill {..} lite boot = io $ do => Ship -> Pill -> Bool -> LegacyBootEvent -> RIO e BootSeq
ent <- genEntropy genBootSeq _ PillIvory {} _ _ = throwIO CannotBootFromIvoryPill
let ovums = preKern ent <> pKernelOvums <> postKern <> pUserspaceOvums genBootSeq ship PillPill {..} lite boot = do
pure $ BootSeq ident pBootFormulas ovums ent <- io genEntropy
wyr <- wyrd
let ova = preKern ent <> [wyr] <> pKernelOva <> postKern <> pUserspaceOva
pure $ BootSeq ident pBootFormulae ova
where where
ident = LogIdentity ship isFake (fromIntegral $ length pBootFormulas) ident = LogIdentity ship isFake (fromIntegral $ length pBootFormulae)
preKern ent = preKern ent =
[ EvBlip $ BlipEvArvo $ ArvoEvWhom () ship [ EvBlip $ BlipEvArvo $ ArvoEvWhom () ship
, EvBlip $ BlipEvArvo $ ArvoEvWack () ent , EvBlip $ BlipEvArvo $ ArvoEvWack () ent
@ -295,14 +302,18 @@ pier (serf, log) vSlog startedSig injected = do
let execute = writeTQueue executeQ let execute = writeTQueue executeQ
let persist = writeTQueue persistQ let persist = writeTQueue persistQ
let sigint = Serf.sendSIGINT serf let sigint = Serf.sendSIGINT serf
let scry = \w b g -> do let scry = \g r -> do
res <- newEmptyMVar res <- newEmptyMVar
atomically $ writeTQueue scryQ (w, b, g, putMVar res) atomically $ writeTQueue scryQ (g, r, putMVar res)
takeMVar res takeMVar res
-- Set up the runtime stat counters.
stat <- newStat
-- Set up the runtime subsite server and its capability to slog -- Set up the runtime subsite server and its capability to slog
-- and display stats.
siteSlog <- newTVarIO (const $ pure ()) siteSlog <- newTVarIO (const $ pure ())
runtimeSubsite <- Site.kingSubsite ship scry siteSlog runtimeSubsite <- Site.kingSubsite ship scry (renderStat stat) siteSlog
-- Slogs go to stderr, to the runtime subsite, and to the terminal. -- Slogs go to stderr, to the runtime subsite, and to the terminal.
env <- ask env <- ask
@ -311,12 +322,12 @@ pier (serf, log) vSlog startedSig injected = do
io $ readTVarIO siteSlog >>= ($ s) io $ readTVarIO siteSlog >>= ($ s)
logOther "serf" (display $ T.strip $ tankToText tank) logOther "serf" (display $ T.strip $ tankToText tank)
let err = atomically . Term.trace muxed . (<> "\r\n")
(bootEvents, startDrivers) <- do (bootEvents, startDrivers) <- do
env <- ask env <- ask
let err = atomically . Term.trace muxed . (<> "\r\n")
siz <- atomically $ Term.curDemuxSize demux siz <- atomically $ Term.curDemuxSize demux
let fak = isFake logId let fak = isFake logId
drivers env ship fak compute scry (siz, muxed) err sigint runtimeSubsite drivers env ship fak compute scry (siz, muxed) err sigint stat runtimeSubsite
let computeConfig = ComputeConfig { ccOnWork = takeTMVar computeQ let computeConfig = ComputeConfig { ccOnWork = takeTMVar computeQ
, ccOnKill = onKill , ccOnKill = onKill
@ -330,6 +341,8 @@ pier (serf, log) vSlog startedSig injected = do
tSerf <- acquireWorker "Serf" (runCompute serf computeConfig) tSerf <- acquireWorker "Serf" (runCompute serf computeConfig)
doVersionNegotiation compute err
-- Run all born events and retry them until they succeed. -- Run all born events and retry them until they succeed.
wackEv <- EvBlip . BlipEvArvo . ArvoEvWack () <$> genEntropy wackEv <- EvBlip . BlipEvArvo . ArvoEvWack () <$> genEntropy
rio $ for_ (wackEv : bootEvents) $ \ev -> do rio $ for_ (wackEv : bootEvents) $ \ev -> do
@ -341,7 +354,7 @@ pier (serf, log) vSlog startedSig injected = do
cb :: Int -> WorkError -> IO () cb :: Int -> WorkError -> IO ()
cb n | n >= 3 = error ("boot event failed: " <> show ev) cb n | n >= 3 = error ("boot event failed: " <> show ev)
cb n = \case cb n = \case
RunOkay _ -> putMVar okaySig () RunOkay _ _ -> putMVar okaySig ()
RunSwap _ _ _ _ _ -> putMVar okaySig () RunSwap _ _ _ _ _ -> putMVar okaySig ()
RunBail _ -> inject (n + 1) RunBail _ -> inject (n + 1)
@ -368,7 +381,7 @@ pier (serf, log) vSlog startedSig injected = do
let inject = atomically $ compute $ RRWork $ EvErr ev $ cb let inject = atomically $ compute $ RRWork $ EvErr ev $ cb
cb :: WorkError -> IO () cb :: WorkError -> IO ()
cb = \case cb = \case
RunOkay _ -> putMVar okaySig (Right ()) RunOkay _ _ -> putMVar okaySig (Right ())
RunSwap _ _ _ _ _ -> putMVar okaySig (Right ()) RunSwap _ _ _ _ _ -> putMVar okaySig (Right ())
RunBail goofs -> putMVar okaySig (Left goofs) RunBail goofs -> putMVar okaySig (Left goofs)
@ -409,6 +422,68 @@ death tag tid = do
Left exn -> Left (tag, exn) Left exn -> Left (tag, exn)
Right () -> Right tag Right () -> Right tag
-- %wyrd version negotiation ---------------------------------------------------
data PierVersionNegotiationFailed = PierVersionNegotiationFailed
deriving (Show, Exception)
zuseVersion :: Word
zuseVersion = 420
wyrd :: HasKingEnv e => RIO e Ev
wyrd = do
king <- tshow <$> view kingIdL
let k = Wynn [("zuse", zuseVersion),
("lull", 330),
("arvo", 240),
("hoon", 140),
("nock", 4)]
sen = MkTerm king
v = Vere sen [Cord "king-haskell", Cord "1.0"] k
pure $ EvBlip $ BlipEvArvo $ ArvoEvWyrd () v
doVersionNegotiation
:: HasPierEnv e
=> (RunReq -> STM ())
-> (Text -> RIO e ())
-> RAcquire e ()
doVersionNegotiation compute stderr = do
ev <- rio wyrd
okaySig :: MVar (Either [Goof] FX) <- newEmptyMVar
let inject = atomically $ compute $ RRWork $ EvErr ev $ cb
cb :: WorkError -> IO ()
cb = \case
RunOkay _ fx -> putMVar okaySig (Right fx)
RunSwap _ _ _ _ fx -> putMVar okaySig (Right fx)
RunBail goofs -> putMVar okaySig (Left goofs)
rio $ stderr "vere: checking version compatibility"
io inject
takeMVar okaySig >>= \case
Left goof -> do
rio $ stderr "pier: version negotation failed"
logError $ display @Text ("Goof in wyrd event: " <> tshow goof)
throwIO PierVersionNegotiationFailed
Right fx -> do
-- Walk through the returned fx looking for a wend effect. If we find
-- one, check the zuse versions.
rio $ for_ fx $ \case
GoodParse (EfWend (Wynn xs)) -> case L.lookup "zuse" xs of
Nothing -> pure ()
Just zuseVerInWynn ->
if zuseVerInWynn /= zuseVersion
then do
rio $ stderr "pier: pier: version negotiation failed; downgrade"
throwIO PierVersionNegotiationFailed
else
pure ()
_ -> pure ()
-- Start All Drivers ----------------------------------------------------------- -- Start All Drivers -----------------------------------------------------------
@ -427,16 +502,19 @@ drivers
-> Ship -> Ship
-> Bool -> Bool
-> (RunReq -> STM ()) -> (RunReq -> STM ())
-> (Wen -> Gang -> Path -> IO (Maybe (Term, Noun))) -> ScryFunc
-> (TermSize, Term.Client) -> (TermSize, Term.Client)
-> (Text -> RIO e ()) -> (Text -> RIO e ())
-> IO () -> IO ()
-> Stat
-> Site.KingSubsite -> Site.KingSubsite
-> RAcquire e ([Ev], RAcquire e Drivers) -> RAcquire e ([Ev], RAcquire e Drivers)
drivers env who isFake plan scry termSys stderr serfSIGINT sub = do drivers env who isFake plan scry termSys stderr serfSIGINT stat sub = do
let Stat{..} = stat
(behnBorn, runBehn) <- rio Behn.behn' (behnBorn, runBehn) <- rio Behn.behn'
(termBorn, runTerm) <- rio (Term.term' termSys serfSIGINT) (termBorn, runTerm) <- rio (Term.term' termSys (renderStat stat) serfSIGINT)
(amesBorn, runAmes) <- rio (Ames.ames' who isFake scry stderr) (amesBorn, runAmes) <- rio (Ames.ames' who isFake statAmes scry stderr)
(httpBorn, runEyre) <- rio (Eyre.eyre' who isFake stderr sub) (httpBorn, runEyre) <- rio (Eyre.eyre' who isFake stderr sub)
(clayBorn, runClay) <- rio Clay.clay' (clayBorn, runClay) <- rio Clay.clay'
(irisBorn, runIris) <- rio Iris.client' (irisBorn, runIris) <- rio Iris.client'
@ -494,6 +572,7 @@ router slog waitFx Drivers {..} = do
case ef of case ef of
GoodParse (EfVega _ _ ) -> vega GoodParse (EfVega _ _ ) -> vega
GoodParse (EfExit _ _ ) -> exit GoodParse (EfExit _ _ ) -> exit
GoodParse (EfWend _ ) -> pure ()
GoodParse (EfVane (VEBehn ef)) -> io (dBehn ef) GoodParse (EfVane (VEBehn ef)) -> io (dBehn ef)
GoodParse (EfVane (VEBoat ef)) -> io (dSync ef) GoodParse (EfVane (VEBoat ef)) -> io (dSync ef)
GoodParse (EfVane (VEClay ef)) -> io (dSync ef) GoodParse (EfVane (VEClay ef)) -> io (dSync ef)
@ -529,7 +608,7 @@ data ComputeConfig = ComputeConfig
{ ccOnWork :: STM RunReq { ccOnWork :: STM RunReq
, ccOnKill :: STM () , ccOnKill :: STM ()
, ccOnSave :: STM () , ccOnSave :: STM ()
, ccOnScry :: STM (Wen, Gang, Path, Maybe (Term, Noun) -> IO ()) , ccOnScry :: STM (Gang, ScryReq, Maybe (Term, Noun) -> IO ())
, ccPutResult :: (Fact, FX) -> STM () , ccPutResult :: (Fact, FX) -> STM ()
, ccShowSpinner :: Maybe Text -> STM () , ccShowSpinner :: Maybe Text -> STM ()
, ccHideSpinner :: STM () , ccHideSpinner :: STM ()
@ -543,7 +622,7 @@ runCompute serf ComputeConfig {..} = do
let onRR = asum [ ccOnKill <&> Serf.RRKill let onRR = asum [ ccOnKill <&> Serf.RRKill
, ccOnSave <&> Serf.RRSave , ccOnSave <&> Serf.RRSave
, ccOnWork , ccOnWork
, ccOnScry <&> \(w,g,p,k) -> Serf.RRScry w g p k , ccOnScry <&> \(g,r,k) -> Serf.RRScry g r k
] ]
vEvProcessing :: TMVar Ev <- newEmptyTMVarIO vEvProcessing :: TMVar Ev <- newEmptyTMVarIO

View File

@ -14,10 +14,11 @@ module Urbit.Vere.Pier.Types
, jobId , jobId
, jobMug , jobMug
, DriverApi(..) , DriverApi(..)
, ScryFunc
) )
where where
import Urbit.Prelude hiding (Term) import Urbit.Prelude
import Urbit.Arvo import Urbit.Arvo
import Urbit.Noun.Time import Urbit.Noun.Time
@ -44,11 +45,14 @@ instance Show Nock where
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
data Pill = Pill data Pill
{ pBootFormulas :: ![Nock] = PillIvory [Noun]
, pKernelOvums :: ![Ev] | PillPill
, pUserspaceOvums :: ![Ev] { pName :: Noun
} , pBootFormulae :: ![Nock] -- XX not actually nock, semantically
, pKernelOva :: ![Ev]
, pUserspaceOva :: ![Ev]
}
deriving (Eq, Show) deriving (Eq, Show)
data BootSeq = BootSeq !LogIdentity ![Nock] ![Ev] data BootSeq = BootSeq !LogIdentity ![Nock] ![Ev]
@ -87,6 +91,10 @@ data DriverApi ef = DriverApi
} }
-- Scrying --------------------------------------------------------------------
type ScryFunc = Gang -> ScryReq -> IO (Maybe (Term, Noun))
-- Instances ------------------------------------------------------------------- -- Instances -------------------------------------------------------------------
instance ToNoun Work where instance ToNoun Work where

View File

@ -16,15 +16,16 @@
|% |%
:: +writ: from king to serf :: +writ: from king to serf
:: ::
+$ gang (unit (set ship))
+$ writ +$ writ
$% $: %live $% $: %live
$% [%cram eve=@] $% [%cram eve=@]
[%exit cod=@] [%exit cod=@]
[%save eve=@] [%save eve=@]
[%meld ~]
[%pack ~] [%pack ~]
== == == ==
[%peek mil=@ now=@da lyc=gang pat=path] :: sam=[gang (each path $%([%once @tas @tas path] [beam @tas beam]))]
[%peek mil=@ sam=*]
[%play eve=@ lit=(list ?((pair @da ovum) *))] [%play eve=@ lit=(list ?((pair @da ovum) *))]
[%work mil=@ job=(pair @da ovum)] [%work mil=@ job=(pair @da ovum)]
== ==
@ -33,7 +34,8 @@
+$ plea +$ plea
$% [%live ~] $% [%live ~]
[%ripe [pro=%1 hon=@ nok=@] eve=@ mug=@] [%ripe [pro=%1 hon=@ nok=@] eve=@ mug=@]
[%slog pri=@ ?(cord tank)] [%slog pri=@ tank]
[%flog cord]
$: %peek $: %peek
$% [%done dat=(unit (cask))] $% [%done dat=(unit (cask))]
[%bail dud=goof] [%bail dud=goof]
@ -48,6 +50,7 @@
[%bail lud=(list goof)] [%bail lud=(list goof)]
== == == ==
== ==
--
``` ```
-} -}
@ -84,7 +87,8 @@ import Foreign.Ptr (castPtr)
import Foreign.Storable (peek, poke) import Foreign.Storable (peek, poke)
import RIO.Prelude (decodeUtf8Lenient) import RIO.Prelude (decodeUtf8Lenient)
import System.Posix.Signals (sigINT, sigKILL, signalProcess) import System.Posix.Signals (sigINT, sigKILL, signalProcess)
import Urbit.Arvo (Ev, FX) import Urbit.Arvo (FX)
import Urbit.Arvo.Event
import Urbit.Noun.Time (Wen) import Urbit.Noun.Time (Wen)
import qualified Data.ByteString as BS import qualified Data.ByteString as BS
@ -171,9 +175,9 @@ recvPleaHandlingSlog :: Serf -> IO Plea
recvPleaHandlingSlog serf = loop recvPleaHandlingSlog serf = loop
where where
loop = recvPlea serf >>= \case loop = recvPlea serf >>= \case
PSlog info -> serfSlog serf info >> loop PSlog info -> serfSlog serf info >> loop
other -> pure other PFlog (Cord ofni) -> serfSlog serf (0, Tank $ Leaf $ Tape $ ofni) >> loop
other -> pure other
-- Higher-Level IPC Functions -------------------------------------------------- -- Higher-Level IPC Functions --------------------------------------------------
@ -219,9 +223,9 @@ sendCompactionRequest serf = do
sendWrit serf (WLive $ LPack ()) sendWrit serf (WLive $ LPack ())
recvLive serf recvLive serf
sendScryRequest :: Serf -> Wen -> Gang -> Path -> IO (Maybe (Term, Noun)) sendScryRequest :: Serf -> Gang -> ScryReq -> IO (Maybe (Term, Noun))
sendScryRequest serf w g p = do sendScryRequest serf g r = do
sendWrit serf (WPeek 0 w g p) sendWrit serf (WPeek 0 g r)
recvPeek serf recvPeek serf
sendShutdownRequest :: Serf -> Atom -> IO () sendShutdownRequest :: Serf -> Atom -> IO ()
@ -370,10 +374,9 @@ compact serf = withSerfLockIO serf $ \ss -> do
{-| {-|
Peek into the serf state. Peek into the serf state.
-} -}
scry :: Serf -> Wen -> Gang -> Path -> IO (Maybe (Term, Noun)) scry :: Serf -> Gang -> ScryReq -> IO (Maybe (Term, Noun))
scry serf w g p = withSerfLockIO serf $ \ss -> do scry serf g r = withSerfLockIO serf $ \ss -> do
(ss,) <$> sendScryRequest serf w g p (ss,) <$> sendScryRequest serf g r
{-| {-|
Given a list of boot events, send them to to the serf in a single Given a list of boot events, send them to to the serf in a single
@ -493,7 +496,7 @@ run serf maxBatchSize getLastEvInLog onInput sendOn spin = topLoop
RRSave () -> doSave RRSave () -> doSave
RRKill () -> doKill RRKill () -> doKill
RRPack () -> doPack RRPack () -> doPack
RRScry w g p k -> doScry w g p k RRScry g r k -> doScry g r k
doPack :: IO () doPack :: IO ()
doPack = compact serf >> topLoop doPack = compact serf >> topLoop
@ -511,8 +514,8 @@ run serf maxBatchSize getLastEvInLog onInput sendOn spin = topLoop
doKill :: IO () doKill :: IO ()
doKill = waitForLog >> snapshot serf >> pure () doKill = waitForLog >> snapshot serf >> pure ()
doScry :: Wen -> Gang -> Path -> (Maybe (Term, Noun) -> IO ()) -> IO () doScry :: Gang -> ScryReq -> (Maybe (Term, Noun) -> IO ()) -> IO ()
doScry w g p k = (scry serf w g p >>= k) >> topLoop doScry g r k = (scry serf g r >>= k) >> topLoop
doWork :: EvErr -> IO () doWork :: EvErr -> IO ()
doWork firstWorkErr = do doWork firstWorkErr = do
@ -529,13 +532,13 @@ run serf maxBatchSize getLastEvInLog onInput sendOn spin = topLoop
RRKill () -> atomically (closeTBMQueue que) >> pure doKill RRKill () -> atomically (closeTBMQueue que) >> pure doKill
RRSave () -> atomically (closeTBMQueue que) >> pure doSave RRSave () -> atomically (closeTBMQueue que) >> pure doSave
RRPack () -> atomically (closeTBMQueue que) >> pure doPack RRPack () -> atomically (closeTBMQueue que) >> pure doPack
RRScry w g p k -> atomically (closeTBMQueue que) >> pure (doScry w g p k) RRScry g r k -> atomically (closeTBMQueue que) >> pure (doScry g r k)
RRWork workErr -> atomically (writeTBMQueue que workErr) >> workLoop que RRWork workErr -> atomically (writeTBMQueue que workErr) >> workLoop que
onWorkResp :: Wen -> EvErr -> Work -> IO () onWorkResp :: Wen -> EvErr -> Work -> IO ()
onWorkResp wen (EvErr evn err) = \case onWorkResp wen (EvErr evn err) = \case
WDone eid hash fx -> do WDone eid hash fx -> do
io $ err (RunOkay eid) io $ err (RunOkay eid fx)
atomically $ sendOn ((Fact eid hash wen (toNoun evn)), fx) atomically $ sendOn ((Fact eid hash wen (toNoun evn)), fx)
WSwap eid hash (wen, noun) fx -> do WSwap eid hash (wen, noun) fx -> do
io $ err (RunSwap eid hash wen noun fx) io $ err (RunSwap eid hash wen noun fx)
@ -543,6 +546,7 @@ run serf maxBatchSize getLastEvInLog onInput sendOn spin = topLoop
WBail goofs -> do WBail goofs -> do
io $ err (RunBail goofs) io $ err (RunBail goofs)
{-| {-|
Given: Given:

View File

@ -35,7 +35,7 @@ data Work
data Writ data Writ
= WLive Live = WLive Live
| WPeek Atom Wen Gang Path | WPeek Atom Gang ScryReq
| WPlay EventId [Noun] | WPlay EventId [Noun]
| WWork Atom Wen Ev | WWork Atom Wen Ev
deriving (Show) deriving (Show)
@ -44,6 +44,7 @@ data Plea
= PLive () = PLive ()
| PRipe SerfInfo | PRipe SerfInfo
| PSlog Slog | PSlog Slog
| PFlog Cord
| PPeek Scry | PPeek Scry
| PPlay Play | PPlay Play
| PWork Work | PWork Work

View File

@ -2,7 +2,7 @@ module Urbit.Vere.Serf.Types where
import Urbit.Prelude import Urbit.Prelude
import Urbit.Arvo (Ev, FX) import Urbit.Arvo (Desk, Ev, FX)
import Urbit.Noun.Time (Wen) import Urbit.Noun.Time (Wen)
@ -82,7 +82,7 @@ data EvErr = EvErr Ev (WorkError -> IO ())
data WorkError -- TODO Rename type and constructors data WorkError -- TODO Rename type and constructors
= RunSwap EventId Mug Wen Noun FX -- TODO Maybe provide less info here? = RunSwap EventId Mug Wen Noun FX -- TODO Maybe provide less info here?
| RunBail [Goof] | RunBail [Goof]
| RunOkay EventId | RunOkay EventId FX
{- {-
- RRWork: Ask the serf to do work, will output (Fact, FX) if work - RRWork: Ask the serf to do work, will output (Fact, FX) if work
@ -94,7 +94,19 @@ data RunReq
| RRSave () | RRSave ()
| RRKill () | RRKill ()
| RRPack () | RRPack ()
| RRScry Wen Gang Path (Maybe (Term, Noun) -> IO ()) | RRScry Gang ScryReq (Maybe (Term, Noun) -> IO ())
type ScryReq = (Each Path Demi)
data Demi
= DemiOnce Term Desk Path
| DemiBeam Term Beam
deriving (Show)
-- TODO
type Beam = Void
deriveNoun ''Demi
-- Exceptions ------------------------------------------------------------------ -- Exceptions ------------------------------------------------------------------
@ -111,6 +123,8 @@ data SerfExn
| SerfNotRunning | SerfNotRunning
| MissingBootEventsInEventLog Word Word | MissingBootEventsInEventLog Word Word
| SnapshotAheadOfLog EventId EventId | SnapshotAheadOfLog EventId EventId
| BailDuringWyrd [Goof]
| SwapDuringWyrd Mug (Wen, Noun) FX
deriving (Show, Exception) deriving (Show, Exception)

View File

@ -0,0 +1,75 @@
module Urbit.Vere.Stat where
import Urbit.Prelude
data Stat = Stat
{ statAmes :: AmesStat
}
data AmesStat = AmesStat
{ asUdp :: TVar Word
, asUqf :: TVar Word
, asUdf :: TVar Word
, asUi6 :: TVar Word
, asRcv :: TVar Word
, asSup :: TVar Word
, asSrf :: TVar Word
, asQuf :: TVar Word
, asFwd :: TVar Word
, asDrt :: TVar Word
, asDvr :: TVar Word
, asDml :: TVar Word
, asSwp :: TVar Word
, asBal :: TVar Word
, asOky :: TVar Word
}
newStat :: MonadIO m => m Stat
newStat = do
asUdp <- newTVarIO 0
asUqf <- newTVarIO 0
asUdf <- newTVarIO 0
asUi6 <- newTVarIO 0
asRcv <- newTVarIO 0
asSup <- newTVarIO 0
asSrf <- newTVarIO 0
asQuf <- newTVarIO 0
asFwd <- newTVarIO 0
asDrt <- newTVarIO 0
asDvr <- newTVarIO 0
asDml <- newTVarIO 0
asSwp <- newTVarIO 0
asBal <- newTVarIO 0
asOky <- newTVarIO 0
pure Stat{statAmes = AmesStat{..}}
bump :: MonadIO m => TVar Word -> m ()
bump s = atomically $ bump' s
bump' :: TVar Word -> STM ()
bump' s = modifyTVar' s (+ 1)
type RenderedStat = [Text]
renderStat :: MonadIO m => Stat -> m RenderedStat
renderStat Stat{statAmes = AmesStat{..}} =
sequence
[ pure "stat:"
, pure " ames:"
, (" udp ingress: " <>) <$> tshow <$> readTVarIO asUdp
, (" udp queue evict: " <>) <$> tshow <$> readTVarIO asUqf
, (" udp recv fail: " <>) <$> tshow <$> readTVarIO asUdf
, (" udp dropped non-ipv4: " <>) <$> tshow <$> readTVarIO asUi6
, (" driver ingress: " <>) <$> tshow <$> readTVarIO asRcv
, (" enqueued for serf: " <>) <$> tshow <$> readTVarIO asSup
, (" sent to serf: " <>) <$> tshow <$> readTVarIO asSrf
, (" serf queue evict: " <>) <$> tshow <$> readTVarIO asQuf
, (" forwarded: " <>) <$> tshow <$> readTVarIO asFwd
, (" dropped (unroutable): " <>) <$> tshow <$> readTVarIO asDrt
, (" dropped (wrong version): " <>) <$> tshow <$> readTVarIO asDvr
, (" dropped (malformed): " <>) <$> tshow <$> readTVarIO asDml
, (" serf swapped: " <>) <$> tshow <$> readTVarIO asSwp
, (" serf bailed: " <>) <$> tshow <$> readTVarIO asBal
, (" serf okay: " <>) <$> tshow <$> readTVarIO asOky
]

View File

@ -27,6 +27,7 @@ import Urbit.Vere.Pier.Types
import Data.List ((!!)) import Data.List ((!!))
import RIO.Directory (createDirectoryIfMissing) import RIO.Directory (createDirectoryIfMissing)
import Urbit.King.API (readPortsFile) import Urbit.King.API (readPortsFile)
import Urbit.Vere.Stat (RenderedStat)
import Urbit.TermSize (TermSize(TermSize)) import Urbit.TermSize (TermSize(TermSize))
import Urbit.Vere.Term.API (Client(Client), ClientTake(..)) import Urbit.Vere.Term.API (Client(Client), ClientTake(..))
@ -558,7 +559,7 @@ localClient doneSignal = fst <$> mkRAcquire start stop
loop rd loop rd
else if w == 3 then do else if w == 3 then do
-- ETX (^C) -- ETX (^C)
logInfo $ displayShow "Ctrl-c interrupt" logInfo $ "Ctrl-c interrupt"
atomically $ do atomically $ do
writeTQueue wq [Term.Trace "interrupt\r\n"] writeTQueue wq [Term.Trace "interrupt\r\n"]
writeTQueue rq $ Ctl $ Cord "c" writeTQueue rq $ Ctl $ Cord "c"
@ -599,9 +600,10 @@ localClient doneSignal = fst <$> mkRAcquire start stop
term' term'
:: HasPierEnv e :: HasPierEnv e
=> (TermSize, Client) => (TermSize, Client)
-> IO RenderedStat
-> IO () -> IO ()
-> RIO e ([Ev], RAcquire e (DriverApi TermEf)) -> RIO e ([Ev], RAcquire e (DriverApi TermEf))
term' (tsize, client) serfSIGINT = do term' (tsize, client) stat serfSIGINT = do
let TermSize wi hi = tsize let TermSize wi hi = tsize
initEv = [blewEvent wi hi, initialHail] initEv = [blewEvent wi hi, initialHail]
@ -610,7 +612,7 @@ term' (tsize, client) serfSIGINT = do
runDriver = do runDriver = do
env <- ask env <- ask
ventQ :: TQueue EvErr <- newTQueueIO ventQ :: TQueue EvErr <- newTQueueIO
diOnEffect <- term env (tsize, client) (writeTQueue ventQ) serfSIGINT diOnEffect <- term env (tsize, client) (writeTQueue ventQ) stat serfSIGINT
let diEventSource = fmap RRWork <$> tryReadTQueue ventQ let diEventSource = fmap RRWork <$> tryReadTQueue ventQ
@ -623,9 +625,10 @@ term :: forall e. (HasPierEnv e)
=> e => e
-> (TermSize, Client) -> (TermSize, Client)
-> (EvErr -> STM ()) -> (EvErr -> STM ())
-> IO RenderedStat
-> IO () -> IO ()
-> RAcquire e (TermEf -> IO ()) -> RAcquire e (TermEf -> IO ())
term env (tsize, Client{..}) plan serfSIGINT = runTerm term env (tsize, Client{..}) plan stat serfSIGINT = runTerm
where where
runTerm :: RAcquire e (TermEf -> IO ()) runTerm :: RAcquire e (TermEf -> IO ())
runTerm = do runTerm = do

View File

@ -1,5 +1,5 @@
name: urbit-king name: urbit-king
version: 0.10.8 version: 1.1
license: MIT license: MIT
license-file: LICENSE license-file: LICENSE
data-files: data-files:

View File

@ -108,9 +108,10 @@ instance Arbitrary LogIdentity where
instance Arbitrary Packet where instance Arbitrary Packet where
arbitrary = do arbitrary = do
pktVersion <- suchThat arb (< 8) pktVersion <- suchThat arb (< 8)
pktEncrypted <- arb
pktSndr <- arb pktSndr <- arb
pktRcvr <- arb pktRcvr <- arb
pktSndrTick <- suchThat arb (< 16)
pktRcvrTick <- suchThat arb (< 16)
pktOrigin <- arb pktOrigin <- arb
pktContent <- arb pktContent <- arb
pure Packet {..} pure Packet {..}

View File

@ -31,18 +31,6 @@ roundTrip x = Just x == fromNoun (toNoun x)
nounEq :: (ToNoun a, ToNoun b) => a -> b -> Bool nounEq :: (ToNoun a, ToNoun b) => a -> b -> Bool
nounEq x y = toNoun x == toNoun y nounEq x y = toNoun x == toNoun y
data EvExample = EvEx Ev Noun
deriving (Eq, Show)
eventSanity :: [EvExample] -> Bool
eventSanity = all $ \(EvEx e n) -> toNoun e == n
instance Arbitrary EvExample where
arbitrary = oneof $ fmap pure $
[ EvEx (EvVane $ VaneVane $ VEVeer (Jael, ()) "" (Path []) "")
(toNoun (Path ["vane", "vane", "jael"], Cord "veer", (), (), ()))
]
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
tests :: TestTree tests :: TestTree
@ -51,7 +39,6 @@ tests =
[ testProperty "Round Trip Effect" (roundTrip @Ef) [ testProperty "Round Trip Effect" (roundTrip @Ef)
, testProperty "Round Trip Event" (roundTrip @Ev) , testProperty "Round Trip Event" (roundTrip @Ev)
, testProperty "Round Trip AmesDest" (roundTrip @AmesDest) , testProperty "Round Trip AmesDest" (roundTrip @AmesDest)
, testProperty "Basic Event Sanity" eventSanity
] ]
@ -131,24 +118,9 @@ instance Arbitrary BlipEv where
] ]
instance Arbitrary Ev where instance Arbitrary Ev where
arbitrary = oneof [ EvVane <$> arb arbitrary = oneof [ EvBlip <$> arb
, EvBlip <$> arb
] ]
instance Arbitrary Vane where
arbitrary = oneof [ VaneVane <$> arb
, VaneZuse <$> arb
]
instance Arbitrary VaneName where
arbitrary = oneof $ pure <$> [minBound .. maxBound]
instance Arbitrary VaneEv where
arbitrary = VEVeer <$> arb <*> arb <*> arb <*> arb
instance Arbitrary ZuseEv where
arbitrary = ZEVeer () <$> arb <*> arb <*> arb
instance Arbitrary StdMethod where instance Arbitrary StdMethod where
arbitrary = oneof $ pure <$> [ minBound .. maxBound ] arbitrary = oneof $ pure <$> [ minBound .. maxBound ]

View File

@ -35,7 +35,7 @@ instance KnownSymbol name => Options.IsOption (Pill name) where
) )
defaultValue = defaultValue =
Pill ( "../../../bin" Pill ( "../../../bin/"
++ TypeLits.symbolVal (Proxy @name) ++ TypeLits.symbolVal (Proxy @name)
++ ".pill" ++ ".pill"
) )

View File

@ -1 +1 @@
2082167031 233234490

View File

@ -5,6 +5,7 @@ module Urbit.Noun.Mug where
import ClassyPrelude import ClassyPrelude
import Data.Bits import Data.Bits
import Data.ByteString.Builder
import Urbit.Atom import Urbit.Atom
import Data.Hash.Murmur (murmur3) import Data.Hash.Murmur (murmur3)
@ -13,14 +14,7 @@ type Mug = Word32
{-# INLINE mugBS #-} {-# INLINE mugBS #-}
mugBS :: ByteString -> Word32 mugBS :: ByteString -> Word32
mugBS = go 0xcafebabe mugBS = mum 0xcafe_babe 0x7fff
where
go seed buf =
let haz = murmur3 seed buf
ham = shiftR haz 31 `xor` (haz .&. 0x7fff_ffff)
in if ham == 0
then go (seed + 1) buf
else ham
-- XX is there a way to do this without copy? -- XX is there a way to do this without copy?
{-# INLINE mugAtom #-} {-# INLINE mugAtom #-}
@ -29,4 +23,16 @@ mugAtom = mugBS . atomBytes
{-# INLINE mugBoth #-} {-# INLINE mugBoth #-}
mugBoth :: Word32 -> Word32 -> Word32 mugBoth :: Word32 -> Word32 -> Word32
mugBoth m n = mugAtom $ fromIntegral $ m `xor` 0x7fff_ffff `xor` n mugBoth m n = mum 0xdead_beef 0xfffe
$ toStrict $ toLazyByteString (word32LE m <> word32LE n)
mum :: Word32 -> Word32 -> ByteString -> Word32
mum syd fal key = go syd 0
where
go syd 8 = fal
go syd i =
let haz = murmur3 syd key
ham = shiftR haz 31 `xor` (haz .&. 0x7fff_ffff)
in if ham /= 0
then ham
else go (syd + 1) (i + 1)

View File

@ -147,7 +147,9 @@ enumFromAtom :: [(String, Name)] -> Exp
enumFromAtom cons = LamE [VarP x] body enumFromAtom cons = LamE [VarP x] body
where where
(x, c) = (mkName "x", mkName "c") (x, c) = (mkName "x", mkName "c")
getTag = BindS (VarP c) $ AppE (VarE 'parseNounUtf8Atom) (VarE x) getTag = BindS (VarP c)
$ AppE (AppE (VarE 'named) matchFail)
$ AppE (VarE 'parseNounUtf8Atom) (VarE x)
examine = NoBindS $ CaseE (VarE c) (matches ++ [fallback]) examine = NoBindS $ CaseE (VarE c) (matches ++ [fallback])
matches = mkMatch <$> cons matches = mkMatch <$> cons
fallback = Match WildP (NormalB $ AppE (VarE 'fail) matchFail) [] fallback = Match WildP (NormalB $ AppE (VarE 'fail) matchFail) []
@ -194,6 +196,7 @@ taggedFromNoun cons = LamE [VarP n] (DoE [getHead, getTag, examine])
$ AppE (VarE 'parseNoun) (VarE n) $ AppE (VarE 'parseNoun) (VarE n)
getTag = BindS (SigP (VarP c) (ConT ''Text)) getTag = BindS (SigP (VarP c) (ConT ''Text))
$ AppE (AppE (VarE 'named) tagFail)
$ AppE (VarE 'parseNounUtf8Atom) (VarE h) $ AppE (VarE 'parseNounUtf8Atom) (VarE h)
examine = NoBindS examine = NoBindS
@ -208,6 +211,8 @@ taggedFromNoun cons = LamE [VarP n] (DoE [getHead, getTag, examine])
fallback = Match WildP (NormalB $ AppE (VarE 'fail) matchFail) [] fallback = Match WildP (NormalB $ AppE (VarE 'fail) matchFail) []
matchFail = unexpectedTag (fst <$> cons) (VarE c) matchFail = unexpectedTag (fst <$> cons) (VarE c)
tagFail = LitE $ StringL (intercalate " " (('%':) <$> (fst <$> cons)))
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
tagString :: Int -> Name -> String tagString :: Int -> Name -> String

File diff suppressed because it is too large Load Diff

View File

@ -1796,13 +1796,13 @@ _cj_minx(u3_noun cey, u3_noun cor)
} }
static void static void
_cj_print_tas(FILE* fh, u3_noun tas) _cj_print_tas(u3_noun tas)
{ {
c3_w met_w = u3r_met(3, tas); c3_w met_w = u3r_met(3, tas);
c3_c* str_c = alloca(met_w + 1); c3_c* str_c = alloca(met_w + 1);
u3r_bytes(0, met_w, (c3_y*)str_c, tas); u3r_bytes(0, met_w, (c3_y*)str_c, tas);
str_c[met_w] = 0; str_c[met_w] = 0;
fprintf(fh, "/%s", str_c); u3l_log("/%s", str_c);
} }
/* _cj_mine(): declare a core and produce location. RETAIN. /* _cj_mine(): declare a core and produce location. RETAIN.
@ -1851,7 +1851,7 @@ _cj_mine(u3_noun cey, u3_noun cor, u3_noun bas)
u3_noun i = bal; u3_noun i = bal;
u3l_log("hot jet: "); u3l_log("hot jet: ");
while ( i != u3_nul ) { while ( i != u3_nul ) {
_cj_print_tas(stderr, u3h(i)); _cj_print_tas(u3h(i));
i = u3t(i); i = u3t(i);
} }
u3l_log("\r\n axe %d, jax %d,\r\n bash ", axe, jax_l); u3l_log("\r\n axe %d, jax %d,\r\n bash ", axe, jax_l);

View File

@ -121,14 +121,10 @@ _fore_io_talk(u3_auto* car_u)
// set verbose as per -v // set verbose as per -v
// //
// XX should be explicit, not a toggle {
// c3_o lac_o = ( c3y == u3_Host.ops_u.veb ) ? c3n : c3y;
if ( c3y == u3_Host.ops_u.veb ) { wir = u3nc(c3__arvo, u3_nul);
// XX this path shouldn't be necessary cad = u3nt(c3__verb, u3_nul, lac_o);
//
wir = u3nt(c3__term, '1', u3_nul);
cad = u3nc(c3__verb, u3_nul);
u3_auto_plan(car_u, u3_ovum_init(0, u3_blip, wir, cad)); u3_auto_plan(car_u, u3_ovum_init(0, u3_blip, wir, cad));
} }

View File

@ -79,7 +79,7 @@ struct _u3_ufil;
} u3_unix; } u3_unix;
void void
u3_unix_ef_look(u3_unix* unx_u, u3_noun all); u3_unix_ef_look(u3_unix* unx_u, u3_noun mon, u3_noun all);
/* u3_readdir_r(): /* u3_readdir_r():
*/ */
@ -609,8 +609,7 @@ static void
_unix_commit_mount_point(u3_unix* unx_u, u3_noun mon) _unix_commit_mount_point(u3_unix* unx_u, u3_noun mon)
{ {
unx_u->dyr = c3y; unx_u->dyr = c3y;
u3z(mon); u3_unix_ef_look(unx_u, mon, c3n);
u3_unix_ef_look(unx_u, c3n);
return; return;
} }
@ -1355,19 +1354,25 @@ u3_unix_release(c3_c* pax_c)
c3_free(paf_c); c3_free(paf_c);
} }
/* u3_unix_ef_look(): update the root. /* u3_unix_ef_look(): update the root of a specific mount point.
*/ */
void void
u3_unix_ef_look(u3_unix* unx_u, u3_noun all) u3_unix_ef_look(u3_unix* unx_u, u3_noun mon, u3_noun all)
{ {
if ( c3y == unx_u->dyr ) { if ( c3y == unx_u->dyr ) {
unx_u->dyr = c3n; unx_u->dyr = c3n;
u3_umon* mon_u; u3_umon* mon_u = unx_u->mon_u;
for ( mon_u = unx_u->mon_u; mon_u; mon_u = mon_u->nex_u ) { while ( mon_u && ( c3n == u3r_sing_c(mon_u->nam_c, mon) ) ) {
mon_u = mon_u->nex_u;
}
if ( mon_u ) {
_unix_update_mount(unx_u, mon_u, all); _unix_update_mount(unx_u, mon_u, all);
} }
} }
u3z(mon);
} }
/* _unix_io_talk(): start listening for fs events. /* _unix_io_talk(): start listening for fs events.

View File

@ -470,6 +470,14 @@ u3_pier_peek_last(u3_pier* pir_u,
_pier_peek_plan(pir_u, pic_u); _pier_peek_plan(pir_u, pic_u);
} }
/* _pier_stab(): parse path
*/
static u3_noun
_pier_stab(u3_noun pac)
{
return u3do("stab", pac);
}
/* _pier_on_scry_done(): scry callback. /* _pier_on_scry_done(): scry callback.
*/ */
static void static void
@ -482,48 +490,64 @@ _pier_on_scry_done(void* ptr_v, u3_noun nun)
u3l_log("pier: scry failed\n"); u3l_log("pier: scry failed\n");
} }
else { else {
u3_weak out, pad;
c3_c *ext_c, *pac_c;
u3l_log("pier: scry succeeded\n"); u3l_log("pier: scry succeeded\n");
// serialize as desired if ( u3_Host.ops_u.puk_c ) {
pac_c = u3_Host.ops_u.puk_c;
}
else {
pac_c = u3_Host.ops_u.pek_c;
}
// try to serialize as requested
// //
u3_atom out;
c3_c* ext_c;
{ {
u3_atom puf = u3i_string(u3_Host.ops_u.puf_c); u3_atom puf = u3i_string(u3_Host.ops_u.puf_c);
if ( c3y == u3r_sing(c3__jam, puf) ) { if ( c3y == u3r_sing(c3__jam, puf) ) {
out = u3qe_jam(res); out = u3qe_jam(res);
ext_c = "jam"; ext_c = "jam";
} }
else { else if ( c3y == u3a_is_atom(res) ) {
out = u3dc("scot", u3k(puf), u3k(res)); out = u3dc("scot", u3k(puf), u3k(res));
ext_c = "txt"; ext_c = "txt";
} }
else {
u3l_log("pier: cannot export cell as %s\n", u3_Host.ops_u.puf_c);
out = u3_none;
}
u3z(puf); u3z(puf);
} }
c3_c* pac_c = u3_Host.ops_u.puk_c; // try to build export target path
if (!pac_c) { //
pac_c = u3_Host.ops_u.pek_c;
}
u3_noun pad;
{ {
// XX crashes if [pac_c] is not a valid path u3_noun pro = u3m_soft(0, _pier_stab, u3i_string(pac_c));
// XX virtualize or fix if ( 0 == u3h(pro) ) {
// c3_w len_w = u3kb_lent(u3k(u3t(pro)));
u3_noun pax = u3do("stab", u3i_string(pac_c)); pad = u3nt(c3_s4('.', 'u', 'r', 'b'),
c3_w len_w = u3kb_lent(u3k(pax)); c3_s3('p', 'u', 't'),
pad = u3nt(c3_s4('.','u','r','b'), u3qb_scag(len_w - 1, u3t(pro)));
c3_s3('p','u','t'), }
u3qb_scag(len_w - 1, pax)); else {
u3z(pax); u3l_log("pier: invalid export path %s\n", pac_c);
pad = u3_none;
}
u3z(pro);
} }
c3_c fil_c[2048]; // if serialization and export path succeeded, write to disk
snprintf(fil_c, 2048, "%s/.urb/put/%s.%s", pir_u->pax_c, pac_c+1, ext_c); //
if ( (u3_none != out) && (u3_none != pad) ) {
c3_c fil_c[2048];
snprintf(fil_c, 2048, "%s/.urb/put/%s.%s",
pir_u->pax_c, pac_c+1, ext_c);
u3_walk_save(fil_c, 0, out, pir_u->pax_c, pad); u3_walk_save(fil_c, 0, out, pir_u->pax_c, pad);
u3l_log("pier: scry in %s\n", fil_c); u3l_log("pier: scry result in %s\n", fil_c);
}
} }
u3l_log("pier: exit\n"); u3l_log("pier: exit\n");
@ -1707,20 +1731,21 @@ _pier_boot_make(u3_noun who, u3_noun wyr, u3_noun ven, u3_noun pil)
// prepend entropy and identity to the module sequence // prepend entropy and identity to the module sequence
// //
{ {
u3_noun wir, cad; u3_noun cad, wir = u3nt(u3_blip, c3__arvo, u3_nul);
c3_w eny_w[16]; c3_w eny_w[16];
c3_rand(eny_w); c3_rand(eny_w);
wir = u3nt(u3_blip, c3__arvo, u3_nul);
cad = u3nt(c3__verb, u3_nul, ( c3y == u3_Host.ops_u.veb ) ? c3n : c3y);
bot_u.mod = u3nc(u3nc(u3k(wir), cad), bot_u.mod);
cad = u3nc(c3__wack, u3i_words(16, eny_w)); cad = u3nc(c3__wack, u3i_words(16, eny_w));
bot_u.mod = u3nc(u3nc(wir, cad), bot_u.mod); bot_u.mod = u3nc(u3nc(u3k(wir), cad), bot_u.mod);
cad = u3nc(c3__whom, who); // transfer [who]
bot_u.mod = u3nc(u3nc(u3k(wir), cad), bot_u.mod);
wir = u3nt(u3_blip, c3__arvo, u3_nul); wir = u3nt(u3_blip, c3__arvo, u3_nul);
cad = u3nc(c3__whom, who); // transfer bot_u.mod = u3nc(u3nc(wir, wyr), bot_u.mod); // transfer [wir] and [wyr]
bot_u.mod = u3nc(u3nc(wir, cad), bot_u.mod);
wir = u3nt(u3_blip, c3__arvo, u3_nul);
bot_u.mod = u3nc(u3nc(wir, wyr), bot_u.mod);
} }
// prepend legacy boot event to the userspace sequence // prepend legacy boot event to the userspace sequence

View File

@ -1 +1 @@
1.0 1.1