mirror of
https://github.com/urbit/shrub.git
synced 2025-01-02 01:25:55 +03:00
237774adc5
Forgot to add `hidden` to http options. Without this, they show up on the summary line in `king run --help`.
190 lines
5.3 KiB
Haskell
190 lines
5.3 KiB
Haskell
module AmesTests (tests) where
|
|
|
|
import Data.Conduit
|
|
import Data.Conduit.List hiding (take)
|
|
import Data.Ord.Unicode
|
|
import Test.QuickCheck hiding ((.&.))
|
|
import Test.Tasty
|
|
import Test.Tasty.QuickCheck
|
|
import Test.Tasty.TH
|
|
import Urbit.Arvo
|
|
import Urbit.King.Config
|
|
import Urbit.Noun
|
|
import Urbit.Prelude
|
|
import Urbit.Time
|
|
import Urbit.Vere.Ames
|
|
import Urbit.Vere.Log
|
|
import Urbit.Vere.Pier.Types
|
|
|
|
import Control.Concurrent (runInBoundThread)
|
|
import Data.LargeWord (LargeKey(..))
|
|
import GHC.Natural (Natural)
|
|
import Network.Socket (tupleToHostAddress)
|
|
|
|
import qualified Urbit.Vere.Log as Log
|
|
|
|
|
|
-- Utils -----------------------------------------------------------------------
|
|
|
|
pid :: KingId
|
|
pid = KingId 0
|
|
|
|
turfEf :: NewtEf
|
|
turfEf = NewtEfTurf (0, ()) []
|
|
|
|
sendEf :: Galaxy -> Wen -> Bytes -> NewtEf
|
|
sendEf g w bs = NewtEfSend (0, ()) (EachYes g) bs
|
|
|
|
data NetworkTestApp = NetworkTestApp
|
|
{ _ntaLogFunc :: !LogFunc
|
|
, _ntaNetworkConfig :: !NetworkConfig
|
|
}
|
|
|
|
makeLenses ''NetworkTestApp
|
|
|
|
instance HasLogFunc NetworkTestApp where
|
|
logFuncL = ntaLogFunc
|
|
|
|
instance HasNetworkConfig NetworkTestApp where
|
|
networkConfigL = ntaNetworkConfig
|
|
|
|
runNetworkApp :: RIO NetworkTestApp a -> IO a
|
|
runNetworkApp = runRIO NetworkTestApp
|
|
{ _ntaLogFunc = mkLogFunc l
|
|
, _ntaNetworkConfig = NetworkConfig NMNormal Nothing Nothing Nothing Nothing
|
|
}
|
|
where
|
|
l _ _ _ _ = pure ()
|
|
|
|
runGala :: forall e. (HasLogFunc e, HasNetworkConfig e)
|
|
=> Word8 -> RAcquire e (TQueue Ev, EffCb e NewtEf)
|
|
runGala point =
|
|
do
|
|
q <- newTQueueIO
|
|
let (_, runAmes) =
|
|
ames pid (fromIntegral point) True (writeTQueue q) noStderr
|
|
cb ← runAmes
|
|
rio $ cb turfEf
|
|
pure (q, cb)
|
|
where
|
|
noStderr _ = pure ()
|
|
|
|
waitForPacket :: TQueue Ev -> Bytes -> IO Bool
|
|
waitForPacket q val = go
|
|
where
|
|
go =
|
|
atomically (readTQueue q) >>= \case
|
|
EvBlip (BlipEvNewt (NewtEvBorn (_, ()) ())) -> go
|
|
EvBlip (BlipEvAmes (AmesEvHear () _ bs)) -> pure (bs == val)
|
|
_ -> pure False
|
|
|
|
runRAcquire :: RAcquire e a -> RIO e a
|
|
runRAcquire acq = rwith acq pure
|
|
|
|
sendThread :: EffCb e NewtEf -> (Galaxy, Bytes) -> RAcquire e ()
|
|
sendThread cb (to, val) = void $ mkRAcquire start cancel
|
|
where
|
|
start = async $ forever $ do threadDelay 1_000
|
|
wen <- io $ now
|
|
cb (sendEf to wen val)
|
|
threadDelay 10_000
|
|
|
|
zodSelfMsg :: Property
|
|
zodSelfMsg = forAll arbitrary (ioProperty . runNetworkApp . runTest)
|
|
where
|
|
runTest :: (HasLogFunc e, HasNetworkConfig e) => Bytes -> RIO e Bool
|
|
runTest val = runRAcquire $ do
|
|
(zodQ, zod) <- runGala 0
|
|
() <- sendThread zod (0, val)
|
|
liftIO (waitForPacket zodQ val)
|
|
|
|
twoTalk :: Property
|
|
twoTalk = forAll arbitrary (ioProperty . runNetworkApp . runTest)
|
|
where
|
|
runTest :: (HasLogFunc e, HasNetworkConfig e)
|
|
=> (Word8, Word8, Bytes) -> RIO e Bool
|
|
runTest (aliceShip, bobShip, val) =
|
|
if aliceShip == bobShip
|
|
then pure True
|
|
else go aliceShip bobShip val
|
|
|
|
go :: (HasLogFunc e, HasNetworkConfig e)
|
|
=> Word8 -> Word8 -> Bytes -> RIO e Bool
|
|
go aliceShip bobShip val = runRAcquire $ do
|
|
(aliceQ, alice) <- runGala aliceShip
|
|
(bobQ, bob) <- runGala bobShip
|
|
sendThread alice (Patp bobShip, val)
|
|
sendThread bob (Patp aliceShip, val)
|
|
liftIO (waitForPacket aliceQ val >> waitForPacket bobQ val)
|
|
|
|
tests :: TestTree
|
|
tests =
|
|
testGroup "Ames"
|
|
[ localOption (QuickCheckTests 10) $
|
|
testProperty "Zod can send a message to itself" $
|
|
zodSelfMsg
|
|
|
|
-- TODO Why doesn't this work in CI?
|
|
-- , localOption (QuickCheckTests 10) $
|
|
-- testProperty "Two galaxies can talk" $
|
|
-- twoTalk
|
|
]
|
|
|
|
|
|
-- Generate Arbitrary Values ---------------------------------------------------
|
|
|
|
arb :: Arbitrary a => Gen a
|
|
arb = arbitrary
|
|
|
|
instance Arbitrary Ipv4 where arbitrary = Ipv4 <$> arb
|
|
instance Arbitrary Port where arbitrary = Port <$> arb
|
|
instance Arbitrary Wen where arbitrary = Wen <$> arb
|
|
instance Arbitrary Gap where arbitrary = Gap . abs <$> arb
|
|
instance Arbitrary Bytes where arbitrary = pure (MkBytes "wtfbbq")
|
|
-- MkBytes . take 100 <$> arb
|
|
|
|
instance Arbitrary a => Arbitrary (Patp a) where
|
|
arbitrary = Patp <$> arb
|
|
|
|
instance Arbitrary ByteString where
|
|
arbitrary = pack <$> arbitrary
|
|
|
|
instance Arbitrary Natural where
|
|
arbitrary = fromIntegral . abs <$> (arbitrary :: Gen Integer)
|
|
|
|
instance (Arbitrary a, Arbitrary b) => Arbitrary (LargeKey a b) where
|
|
arbitrary = LargeKey <$> arb <*> arb
|
|
|
|
genIpv4 :: Gen Ipv4
|
|
genIpv4 = do
|
|
x <- arbitrary
|
|
if (x == 0 || (x≥256 && x≤512))
|
|
then genIpv4
|
|
else pure (Ipv4 x)
|
|
|
|
instance Arbitrary Text where
|
|
arbitrary = pack <$> arb
|
|
|
|
instance Arbitrary Cord where
|
|
arbitrary = Cord <$> arb
|
|
|
|
instance Arbitrary BigCord where
|
|
arbitrary = BigCord <$> arb
|
|
|
|
instance Arbitrary AmesDest where
|
|
arbitrary = oneof [ EachYes <$> arb
|
|
, EachNo <$> arb
|
|
]
|
|
|
|
instance Arbitrary a => Arbitrary (Jammed a) where
|
|
arbitrary = Jammed <$> arbitrary
|
|
|
|
instance Arbitrary AmesAddress where
|
|
arbitrary = AAIpv4 <$> arb <*> arb
|
|
|
|
instance Arbitrary Ship where
|
|
arbitrary = Ship <$> arb
|
|
|
|
instance Arbitrary LogIdentity where
|
|
arbitrary = LogIdentity <$> arb <*> arb <*> arb
|