This commit is contained in:
Tyler Brown Cifu Shuster 2020-07-24 11:50:57 -07:00
commit e7b759267a
154 changed files with 17557 additions and 12175 deletions

View File

@ -1,6 +1,13 @@
stages:
- compile
# Don't run the combine stage in pull requests, because deploy is disabled there.
- name: combine
if: type != pull_request
jobs: jobs:
include: include:
- os: linux - stage: compile
os: linux
language: nix language: nix
nix: 2.3.6 nix: 2.3.6
before_install: before_install:
@ -15,7 +22,8 @@ jobs:
- make release - make release
- sh/ci-tests - sh/ci-tests
- os: linux - stage: compile
os: linux
language: generic language: generic
env: STACK_YAML=pkg/hs/stack.yaml env: STACK_YAML=pkg/hs/stack.yaml
cache: cache:
@ -32,7 +40,8 @@ jobs:
- stack test - stack test
- sh/release-king-linux64-dynamic - sh/release-king-linux64-dynamic
- os: osx - stage: compile
os: osx
language: generic language: generic
sudo: required sudo: required
env: STACK_YAML=pkg/hs/stack.yaml env: STACK_YAML=pkg/hs/stack.yaml
@ -50,6 +59,12 @@ jobs:
- stack test - stack test
- sh/release-king-darwin-dynamic - sh/release-king-darwin-dynamic
- stage: combine
os: linux
language: generic
script:
- sh/combine-release-builds
deploy: deploy:
- skip_cleanup: true - skip_cleanup: true
provider: gcs provider: gcs

View File

@ -280,8 +280,8 @@ separate releases.
(**Note**: the following steps are automated by some other Tlon-internal (**Note**: the following steps are automated by some other Tlon-internal
tooling. Just ask `~nidsut-tomdun` for details.) tooling. Just ask `~nidsut-tomdun` for details.)
For Urbit OS updates, this means copying the files into ~zod's %base desk. The For Urbit OS updates, this means copying the files into ~zod's %home desk. The
changes will be synced to /~zod/kids and then propagated through other galaxies changes should be merged into /~zod/kids and then propagated through other galaxies
and stars to the rest of the network. and stars to the rest of the network.
For consistency, I create a release tarball and then rsync the files in. For consistency, I create a release tarball and then rsync the files in.
@ -289,9 +289,10 @@ For consistency, I create a release tarball and then rsync the files in.
``` ```
$ wget https://github.com/urbit/urbit/archive/urbit-os-vx.y.z.tar.gz $ wget https://github.com/urbit/urbit/archive/urbit-os-vx.y.z.tar.gz
$ tar xzf urbit-os-vx.y.z.tar.gz $ tar xzf urbit-os-vx.y.z.tar.gz
$ herb zod -p hood -d "+hood/mount /=base=" $ herb zod -p hood -d "+hood/mount /=home="
$ rsync -zr --delete urbit-urbit-os-vx.y.z/pkg/arvo/ zod/base $ rsync -zr --delete urbit-urbit-os-vx.y.z/pkg/arvo/ zod/home
$ herb zod -p hood -d "+hood/commit %base" $ herb zod -p hood -d "+hood/commit %home"
$ herb zod -p hood -d "+hood/merge %kids our %home"
``` ```
For Vere updates, this means simply shutting down each desired ship, installing For Vere updates, this means simply shutting down each desired ship, installing

View File

@ -1,3 +1,3 @@
version https://git-lfs.github.com/spec/v1 version https://git-lfs.github.com/spec/v1
oid sha256:f738f60e9e028081864f317106f623d2f21a5fe5c2f6fdd83576e22d21a8c6a6 oid sha256:35d8930b9b35364605196d99766ec713154af9105ce7b9fabfaa50e8ca29a5fd
size 14862847 size 4448128

View File

@ -1,3 +1,3 @@
version https://git-lfs.github.com/spec/v1 version https://git-lfs.github.com/spec/v1
oid sha256:9fbfbe75a6685df444621ebd27677716fd0abf7113020f3274c3b5c209e3616e oid sha256:e5c82dea80aa7c5593f43fa4294db7974211abceedd907663da73889857642e7
size 1304972 size 1309381

View File

@ -1,3 +1,3 @@
version https://git-lfs.github.com/spec/v1 version https://git-lfs.github.com/spec/v1
oid sha256:59786d78805460632c4de60275b994260d255be7b721ccf47140d7647a46e66c oid sha256:ecf3f8593815742e409008421f318b664124e672b1eecd131e4a1e49864a1c2a
size 6244195 size 6175676

View File

@ -13,7 +13,6 @@ let
murmur3-src = deps.murmur3.src; murmur3-src = deps.murmur3.src;
scrypt-src = deps.scrypt.src; scrypt-src = deps.scrypt.src;
secp256k1-src = deps.secp256k1.src; secp256k1-src = deps.secp256k1.src;
sni-src = deps.sni.src;
softfloat3-src = deps.softfloat3.src; softfloat3-src = deps.softfloat3.src;
uv-src = deps.uv.src; uv-src = deps.uv.src;
}; };

7
nix/cachix/tests.nix Normal file
View File

@ -0,0 +1,7 @@
let
ops = import ../ops/default.nix {};
in
{
results = ops.test;
fakebus = ops.bus;
}

View File

@ -5,7 +5,6 @@ rec {
murmur3 = import ./deps/murmur3/cross.nix { inherit crossenv; }; murmur3 = import ./deps/murmur3/cross.nix { inherit crossenv; };
uv = import ./deps/uv/cross.nix { inherit crossenv; }; uv = import ./deps/uv/cross.nix { inherit crossenv; };
ed25519 = import ./deps/ed25519/cross.nix { inherit crossenv; }; ed25519 = import ./deps/ed25519/cross.nix { inherit crossenv; };
sni = import ./deps/sni/cross.nix { inherit crossenv; };
scrypt = import ./deps/scrypt/cross.nix { inherit crossenv; }; scrypt = import ./deps/scrypt/cross.nix { inherit crossenv; };
softfloat3 = import ./deps/softfloat3/cross.nix { inherit crossenv; }; softfloat3 = import ./deps/softfloat3/cross.nix { inherit crossenv; };
secp256k1 = import ./deps/secp256k1/cross.nix { inherit crossenv; }; secp256k1 = import ./deps/secp256k1/cross.nix { inherit crossenv; };

View File

@ -10,7 +10,7 @@ let
libs = libs =
with pkgs; with pkgs;
[ openssl zlib curl gmp scrypt libsigsegv ncurses openssl zlib lmdb ]; [ openssl curl gmp scrypt libsigsegv openssl zlib lmdb ];
osx = osx =
with pkgs; with pkgs;
@ -20,7 +20,7 @@ let
vendor = vendor =
with deps; with deps;
[ argon2 ed25519 h2o murmur3 scrypt secp256k1 sni softfloat3 uv ent ge-additions ivory-header ca-header ]; [ argon2 ed25519 h2o murmur3 scrypt secp256k1 softfloat3 uv ent ge-additions ivory-header ca-header ];
in in

View File

@ -5,7 +5,6 @@ rec {
murmur3 = import ./murmur3 { inherit pkgs; }; murmur3 = import ./murmur3 { inherit pkgs; };
uv = import ./uv { inherit pkgs; }; uv = import ./uv { inherit pkgs; };
ed25519 = import ./ed25519 { inherit pkgs; }; ed25519 = import ./ed25519 { inherit pkgs; };
sni = import ./sni { inherit pkgs; };
scrypt = import ./scrypt { inherit pkgs; }; scrypt = import ./scrypt { inherit pkgs; };
softfloat3 = import ./softfloat3 { inherit pkgs; }; softfloat3 = import ./softfloat3 { inherit pkgs; };
secp256k1 = import ./secp256k1 { inherit pkgs; }; secp256k1 = import ./secp256k1 { inherit pkgs; };

View File

@ -1,13 +0,0 @@
source $stdenv/setup
CFLAGS="-O3 -Wall -ffast-math -Wno-unused-const-variable"
echo $CC $CFLAGS -c $src/src/tls.c -o tls.o
$CC $CFLAGS -c $src/src/tls.c -o tls.o
echo $AR rcs libsni.a tls.o
$AR rcs libsni.a tls.o
mkdir -p $out/{lib,include}
cp libsni.a $out/lib/
cp $src/src/tls.h $out/include/

View File

@ -1,18 +0,0 @@
{ crossenv }:
crossenv.make_derivation rec {
name = "sni";
builder = ./builder.sh;
CC = "${crossenv.host}-gcc";
AR = "${crossenv.host}-ar";
src = crossenv.nixpkgs.fetchFromGitHub {
owner = "urbit";
repo = "sniproxy";
rev = "173beb88ee62bddd13874ca04ab338cdec704928";
sha256 = "1ib6p7vhpvbg6d5a2aimppsb09kjg4px4vlw5h3ys9zf9c1if5z4";
};
}

View File

@ -1,12 +0,0 @@
{ pkgs }:
pkgs.stdenv.mkDerivation rec {
name = "sni";
builder = ./builder.sh;
src = pkgs.fetchFromGitHub {
owner = "urbit";
repo = "sniproxy";
rev = "173beb88ee62bddd13874ca04ab338cdec704928";
sha256 = "1ib6p7vhpvbg6d5a2aimppsb09kjg4px4vlw5h3ys9zf9c1if5z4";
};
}

View File

@ -24,10 +24,6 @@ rec {
inherit crossenv; inherit crossenv;
}; };
ncurses = import ./pkgs/ncurses {
inherit crossenv;
};
pdcurses = import ./pkgs/pdcurses { pdcurses = import ./pkgs/pdcurses {
inherit crossenv; inherit crossenv;
}; };

View File

@ -10,10 +10,10 @@ let
tlon = import ../pkgs { inherit pkgs; }; tlon = import ../pkgs { inherit pkgs; };
arvo = tlon.arvo; arvo = tlon.arvo;
urbit = tlon.urbit; urbit = tlon.urbit;
herb = tlon.herb; herb = tlon.herb;
in in
import ./fakeship { import ./fakeship {
inherit pkgs arvo pill ship herb urbit; inherit pkgs arvo pill ship urbit herb;
} }

View File

@ -31,16 +31,16 @@ let
ship = "zod"; ship = "zod";
}; };
in
rec {
bus = import ./fakeship { bus = import ./fakeship {
inherit pkgs herb urbit arvo; inherit pkgs herb urbit arvo;
pill = bootsolid; pill = bootsolid;
ship = "bus"; ship = "bus";
}; };
in
rec {
test = import ./test { test = import ./test {
inherit pkgs herb urbit; inherit pkgs herb urbit;
ship = bus; ship = bus;

View File

@ -13,7 +13,7 @@ check () {
[ 3 -eq "$(herb $out -d 3)" ] [ 3 -eq "$(herb $out -d 3)" ]
} }
if check if check && sleep 10 && check
then then
echo "Boot success." >&2 echo "Boot success." >&2
herb $out -p hood -d '+hood/exit' || true herb $out -p hood -d '+hood/exit' || true

View File

@ -7,13 +7,13 @@ chmod -R u+rw ./pier
$URBIT -d ./pier $URBIT -d ./pier
cleanup () { shutdown () {
if [ -e ./pier/.vere.lock ] if [ -e ./pier/.vere.lock ]
then kill $(< ./pier/.vere.lock) || true; then kill $(< ./pier/.vere.lock) || true;
fi fi
} }
trap cleanup EXIT trap shutdown EXIT
# update pill strategy to ensure correct staging # update pill strategy to ensure correct staging
# #
@ -75,6 +75,8 @@ herb ./pier -p hood -d "+hood/unmount %stage"
herb ./pier -P solid.pill -d '+solid /=stage=/sys, =dub &' herb ./pier -P solid.pill -d '+solid /=stage=/sys, =dub &'
herb ./pier -p hood -d '+hood/exit' || true
mv solid.pill $out mv solid.pill $out
set +x set +x

View File

@ -21,7 +21,7 @@ let
mkUrbit = { debug }: mkUrbit = { debug }:
import ./urbit { import ./urbit {
inherit pkgs ent debug ge-additions libaes_siv; inherit pkgs ent debug ge-additions libaes_siv;
inherit (deps) argon2 murmur3 uv ed25519 sni scrypt softfloat3; inherit (deps) argon2 murmur3 uv ed25519 scrypt softfloat3;
inherit (deps) secp256k1 h2o ivory-header ca-header; inherit (deps) secp256k1 h2o ivory-header ca-header;
}; };

View File

@ -1,7 +1,7 @@
{ {
pkgs, pkgs,
debug, debug,
argon2, ed25519, ent, ge-additions, libaes_siv, h2o, murmur3, scrypt, secp256k1, sni, softfloat3, uv, ivory-header, ca-header argon2, ed25519, ent, ge-additions, libaes_siv, h2o, murmur3, scrypt, secp256k1, softfloat3, uv, ivory-header, ca-header
}: }:
let let
@ -23,10 +23,10 @@ let
deps = deps =
with pkgs; with pkgs;
[ curl gmp sigseg ncurses openssl zlib lmdb ]; [ curl gmp sigseg openssl zlib lmdb ];
vendor = vendor =
[ argon2 softfloat3 ed25519 ent ge-additions libaes_siv h2o scrypt uv murmur3 secp256k1 sni ivory-header ca-header ]; [ argon2 softfloat3 ed25519 ent ge-additions libaes_siv h2o scrypt uv murmur3 secp256k1 ivory-header ca-header ];
urbit = pkgs.stdenv.mkDerivation { urbit = pkgs.stdenv.mkDerivation {
inherit name meta; inherit name meta;

View File

@ -12,11 +12,11 @@ let
crossdeps = crossdeps =
with env; with env;
[ curl libgmp libsigsegv ncurses openssl zlib lmdb ]; [ curl libgmp libsigsegv openssl zlib lmdb ];
vendor = vendor =
with deps; with deps;
[ argon2 softfloat3 ed25519 ge-additions libaes_siv h2o scrypt uv murmur3 secp256k1 sni ivory-header ca-header ]; [ argon2 softfloat3 ed25519 ge-additions libaes_siv h2o scrypt uv murmur3 secp256k1 ivory-header ca-header ];
in in
@ -27,7 +27,6 @@ env.make_derivation {
MEMORY_DEBUG = debug; MEMORY_DEBUG = debug;
CPU_DEBUG = debug; CPU_DEBUG = debug;
EVENT_TIME_DEBUG = false; EVENT_TIME_DEBUG = false;
NCURSES = env.ncurses;
name = "${name}-${env_name}"; name = "${name}-${env_name}";
exename = name; exename = name;

View File

@ -17,6 +17,5 @@ bash ./configure
make build/urbit build/urbit-worker -j8 make build/urbit build/urbit-worker -j8
mkdir -p $out/bin mkdir -p $out/bin
cp -r $NCURSES/share/terminfo $out/bin/$exename-terminfo
cp ./build/urbit $out/bin/$exename cp ./build/urbit $out/bin/$exename
cp ./build/urbit-worker $out/bin/$exename-worker cp ./build/urbit-worker $out/bin/$exename-worker

View File

@ -12,5 +12,5 @@ import ./default.nix {
inherit (tlon) inherit (tlon)
ent ge-additions libaes_siv; ent ge-additions libaes_siv;
inherit (deps) inherit (deps)
argon2 ed25519 h2o murmur3 scrypt secp256k1 sni softfloat3 uv ivory-header ca-header; argon2 ed25519 h2o murmur3 scrypt secp256k1 softfloat3 uv ivory-header ca-header;
} }

View File

@ -33,7 +33,7 @@ let
builds-for-platform = plat: builds-for-platform = plat:
plat.deps // { plat.deps // {
inherit (plat.env) curl libgmp libsigsegv ncurses openssl zlib lmdb; inherit (plat.env) curl libgmp libsigsegv openssl zlib lmdb;
inherit (plat.env) cmake_toolchain; inherit (plat.env) cmake_toolchain;
ent = ent plat; ent = ent plat;
ge-additions = ge-additions plat; ge-additions = ge-additions plat;

View File

@ -69,13 +69,37 @@
|= [ovo=ovum ken=*] |= [ovo=ovum ken=*]
[~ (slum ken [now ovo])] [~ (slum ken [now ovo])]
:: ::
:: our boot-ova is a list containing one massive formula: :: boot-one: lifecycle formula (from +brass)
::
=/ boot-one
=> [boot-formula=** full-sequence=**]
!= =+ [state-gate main-sequence]=.*(full-sequence boot-formula)
|-
?@ main-sequence
state-gate
%= $
main-sequence +.main-sequence
state-gate .*(state-gate [%9 2 %10 [6 %1 -.main-sequence] %0 1])
==
::
:: kernel-formula
:: ::
:: We evaluate :arvo-formula (for jet registration), :: We evaluate :arvo-formula (for jet registration),
:: then ignore the result and produce :installed :: then ignore the result and produce .installed
::
=/ kernel-formula
[%7 arvo-formula %1 installed]
::
:: boot-two: startup formula
::
=/ boot-two
=> [kernel-formula=** main-sequence=**]
!= [.*(0 kernel-formula) main-sequence]
::
:: boot-ova
:: ::
=/ boot-ova=(list) =/ boot-ova=(list)
[[%7 arvo-formula %1 installed] ~] [boot-one boot-two kernel-formula ~]
:: ::
:: a pill is a 3-tuple of event-lists: [boot kernel userspace] :: a pill is a 3-tuple of event-lists: [boot kernel userspace]
:: ::

1
pkg/hs/.gitignore vendored Normal file
View File

@ -0,0 +1 @@
stack.yaml.lock

View File

@ -19,7 +19,7 @@ dependencies:
- transformers - transformers
- transformers-compat - transformers-compat
- unordered-containers - unordered-containers
- urbit-king - urbit-noun
default-extensions: default-extensions:
- ApplicativeDo - ApplicativeDo

3
pkg/hs/racquire/.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
.stack-work
*.cabal
test/gold/*.writ

21
pkg/hs/racquire/LICENSE Normal file
View File

@ -0,0 +1,21 @@
The MIT License (MIT)
Copyright (c) 2016 urbit
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

View File

@ -0,0 +1,66 @@
name: racquire
version: 0.10.4
license: MIT
license-file: LICENSE
library:
source-dirs: lib
ghc-options:
- -fwarn-incomplete-patterns
- -fwarn-unused-binds
- -fwarn-unused-imports
- -Werror
- -O2
dependencies:
- base
- mtl
- unliftio-core
- resourcet
- exceptions
- rio
default-extensions:
- ApplicativeDo
- BangPatterns
- BlockArguments
- ConstraintKinds
- DataKinds
- DefaultSignatures
- DeriveAnyClass
- DeriveDataTypeable
- DeriveFoldable
- DeriveGeneric
- DeriveTraversable
- DerivingStrategies
- EmptyCase
- EmptyDataDecls
- FlexibleContexts
- FlexibleInstances
- FunctionalDependencies
- GADTs
- GeneralizedNewtypeDeriving
- LambdaCase
- MagicHash
- MultiParamTypeClasses
- NamedFieldPuns
- NoImplicitPrelude
- NumericUnderscores
- OverloadedStrings
- PackageImports
- PartialTypeSignatures
- PatternSynonyms
- QuasiQuotes
- Rank2Types
- RankNTypes
- RecordWildCards
- ScopedTypeVariables
- StandaloneDeriving
- TemplateHaskell
- TupleSections
- TypeApplications
- TypeFamilies
- TypeOperators
- UnboxedTuples
- UnicodeSyntax
- ViewPatterns

View File

@ -3,10 +3,15 @@ resolver: lts-14.21
packages: packages:
- lmdb-static - lmdb-static
- proto - proto
- racquire
- terminal-progress-bar - terminal-progress-bar
- urbit-atom - urbit-atom
- urbit-azimuth - urbit-azimuth
- urbit-eventlog-lmdb
- urbit-king - urbit-king
- urbit-termsize
- urbit-noun
- urbit-noun-core
extra-deps: extra-deps:
- flat-0.3.4@sha256:002a0e0ae656ea8cc02a772d0bcb6ea7dbd7f2e79070959cc748ad1e7138eb38 - flat-0.3.4@sha256:002a0e0ae656ea8cc02a772d0bcb6ea7dbd7f2e79070959cc748ad1e7138eb38

3
pkg/hs/urbit-eventlog-lmdb/.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
.stack-work
*.cabal
test/gold/*.writ

View File

@ -0,0 +1,21 @@
The MIT License (MIT)
Copyright (c) 2016 urbit
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

View File

@ -4,27 +4,51 @@
TODO Effects storage logic is messy. TODO Effects storage logic is messy.
-} -}
module Urbit.Vere.Log ( EventLog, identity, nextEv, lastEv module Urbit.EventLog.LMDB
, new, existing ( LogIdentity(..)
, streamEvents, appendEvents, trimEvents , EventLog
, streamEffectsRows, writeEffectsRow , identity
) where , nextEv
, lastEv
, new
, existing
, streamEvents
, appendEvents
, trimEvents
, streamEffectsRows
, writeEffectsRow
)
where
import Urbit.Prelude hiding (init) import ClassyPrelude
import Data.Conduit
import Data.RAcquire import Data.RAcquire
import Database.LMDB.Raw import Database.LMDB.Raw
import Foreign.Marshal.Alloc
import Foreign.Ptr
import Urbit.Vere.Pier.Types
import Foreign.Storable (peek, poke, sizeOf) import Data.Conduit (ConduitT, yield)
import Foreign.Marshal.Alloc (allocaBytes)
import Foreign.Ptr (Ptr, castPtr, nullPtr)
import Foreign.Storable (peek, poke, sizeOf)
import RIO (HasLogFunc, RIO, display, logDebug, runRIO)
import Urbit.Noun (DecodeErr, Noun, Ship)
import Urbit.Noun (deriveNoun, fromNounExn, toNoun)
import Urbit.Noun (cueBS, jamBS)
import qualified Data.ByteString.Unsafe as BU import qualified Data.ByteString.Unsafe as BU
import qualified Data.Vector as V import qualified Data.Vector as V
-- Public Types ----------------------------------------------------------------
data LogIdentity = LogIdentity
{ who :: Ship
, isFake :: Bool
, lifecycleLen :: Word
} deriving (Eq, Ord, Show)
deriveNoun ''LogIdentity
-- Types ----------------------------------------------------------------------- -- Types -----------------------------------------------------------------------
type Env = MDB_env type Env = MDB_env
@ -34,29 +58,29 @@ type Dbi = MDB_dbi
type Cur = MDB_cursor type Cur = MDB_cursor
data EventLog = EventLog data EventLog = EventLog
{ env :: Env { env :: Env
, _metaTbl :: Dbi , _metaTbl :: Dbi
, eventsTbl :: Dbi , eventsTbl :: Dbi
, effectsTbl :: Dbi , effectsTbl :: Dbi
, identity :: LogIdentity , identity :: LogIdentity
, numEvents :: IORef EventId , numEvents :: TVar Word64
} }
nextEv :: EventLog -> RIO e EventId nextEv :: EventLog -> STM Word64
nextEv = fmap succ . readIORef . numEvents nextEv = fmap (+1) . lastEv
lastEv :: EventLog -> RIO e EventId lastEv :: EventLog -> STM Word64
lastEv = readIORef . numEvents lastEv = readTVar . numEvents
data EventLogExn data EventLogExn
= NoLogIdentity = NoLogIdentity
| MissingEvent EventId | MissingEvent Word64
| BadNounInLogIdentity ByteString DecodeErr ByteString | BadNounInLogIdentity ByteString DecodeErr ByteString
| BadKeyInEventLog | BadKeyInEventLog
| BadWriteLogIdentity LogIdentity | BadWriteLogIdentity LogIdentity
| BadWriteEvent EventId | BadWriteEvent Word64
| BadWriteEffect EventId | BadWriteEffect Word64
deriving Show deriving Show
-- Instances ------------------------------------------------------------------- -- Instances -------------------------------------------------------------------
@ -64,6 +88,12 @@ data EventLogExn
instance Exception EventLogExn where instance Exception EventLogExn where
-- Utils -----------------------------------------------------------------------
io :: MonadIO m => IO a -> m a
io = liftIO
-- Open/Close an Event Log ----------------------------------------------------- -- Open/Close an Event Log -----------------------------------------------------
rawOpen :: MonadIO m => FilePath -> m Env rawOpen :: MonadIO m => FilePath -> m Env
@ -82,7 +112,7 @@ create dir id = do
(m, e, f) <- createTables env (m, e, f) <- createTables env
clearEvents env e clearEvents env e
writeIdent env m id writeIdent env m id
EventLog env m e f id <$> newIORef 0 EventLog env m e f id <$> newTVarIO 0
where where
createTables env = createTables env =
rwith (writeTxn env) $ \txn -> io $ rwith (writeTxn env) $ \txn -> io $
@ -98,7 +128,7 @@ open dir = do
id <- getIdent env m id <- getIdent env m
logDebug $ display (pack @Text $ "Log Identity: " <> show id) logDebug $ display (pack @Text $ "Log Identity: " <> show id)
numEvs <- getNumEvents env e numEvs <- getNumEvents env e
EventLog env m e f id <$> newIORef numEvs EventLog env m e f id <$> newTVarIO numEvs
where where
openTables env = openTables env =
rwith (writeTxn env) $ \txn -> io $ rwith (writeTxn env) $ \txn -> io $
@ -227,10 +257,10 @@ clearEvents env eventsTbl =
appendEvents :: EventLog -> Vector ByteString -> RIO e () appendEvents :: EventLog -> Vector ByteString -> RIO e ()
appendEvents log !events = do appendEvents log !events = do
numEvs <- readIORef (numEvents log) numEvs <- atomically $ readTVar (numEvents log)
next <- pure (numEvs + 1) next <- pure (numEvs + 1)
doAppend $ zip [next..] $ toList events doAppend $ zip [next..] $ toList events
writeIORef (numEvents log) (numEvs + word (length events)) atomically $ writeTVar (numEvents log) (numEvs + word (length events))
where where
flags = compileWriteFlags [MDB_NOOVERWRITE] flags = compileWriteFlags [MDB_NOOVERWRITE]
doAppend = \kvs -> doAppend = \kvs ->
@ -240,21 +270,20 @@ appendEvents log !events = do
True -> pure () True -> pure ()
False -> throwIO (BadWriteEvent k) False -> throwIO (BadWriteEvent k)
writeEffectsRow :: EventLog -> EventId -> ByteString -> RIO e () writeEffectsRow :: MonadIO m => EventLog -> Word64 -> ByteString -> m ()
writeEffectsRow log k v = do writeEffectsRow log k v = io $ runRIO () $ do
rwith (writeTxn $ env log) $ \txn -> let flags = compileWriteFlags []
putBytes flags txn (effectsTbl log) k v >>= \case rwith (writeTxn $ env log) $ \txn ->
True -> pure () putBytes flags txn (effectsTbl log) k v >>= \case
False -> throwIO (BadWriteEffect k) True -> pure ()
where False -> throwIO (BadWriteEffect k)
flags = compileWriteFlags []
-- Read Events ----------------------------------------------------------------- -- Read Events -----------------------------------------------------------------
trimEvents :: HasLogFunc e => EventLog -> Word64 -> RIO e () trimEvents :: HasLogFunc e => EventLog -> Word64 -> RIO e ()
trimEvents log start = do trimEvents log start = do
last <- lastEv log last <- atomically (lastEv log)
rwith (writeTxn $ env log) $ \txn -> rwith (writeTxn $ env log) $ \txn ->
for_ [start..last] $ \eId -> for_ [start..last] $ \eId ->
withWordPtr eId $ \pKey -> do withWordPtr eId $ \pKey -> do
@ -262,23 +291,21 @@ trimEvents log start = do
found <- io $ mdb_del txn (eventsTbl log) key Nothing found <- io $ mdb_del txn (eventsTbl log) key Nothing
unless found $ unless found $
throwIO (MissingEvent eId) throwIO (MissingEvent eId)
writeIORef (numEvents log) (pred start) atomically $ writeTVar (numEvents log) (pred start)
streamEvents :: HasLogFunc e streamEvents :: MonadIO m => EventLog -> Word64 -> ConduitT () ByteString m ()
=> EventLog -> Word64
-> ConduitT () ByteString (RIO e) ()
streamEvents log first = do streamEvents log first = do
batch <- lift $ readBatch log first batch <- io $ runRIO () $ readBatch log first
unless (null batch) $ do unless (null batch) $ do
for_ batch yield for_ batch yield
streamEvents log (first + word (length batch)) streamEvents log (first + word (length batch))
streamEffectsRows :: e. HasLogFunc e streamEffectsRows :: e. HasLogFunc e
=> EventLog -> EventId => EventLog -> Word64
-> ConduitT () (Word64, ByteString) (RIO e) () -> ConduitT () (Word64, ByteString) (RIO e) ()
streamEffectsRows log = go streamEffectsRows log = go
where where
go :: EventId -> ConduitT () (Word64, ByteString) (RIO e) () go :: Word64 -> ConduitT () (Word64, ByteString) (RIO e) ()
go next = do go next = do
batch <- lift $ readRowsBatch (env log) (effectsTbl log) next batch <- lift $ readRowsBatch (env log) (effectsTbl log) next
unless (null batch) $ do unless (null batch) $ do
@ -294,12 +321,12 @@ readBatch :: EventLog -> Word64 -> RIO e (V.Vector ByteString)
readBatch log first = start readBatch log first = start
where where
start = do start = do
last <- lastEv log last <- atomically (lastEv log)
if (first > last) if (first > last)
then pure mempty then pure mempty
else readRows $ fromIntegral $ min 1000 $ ((last+1) - first) else readRows $ fromIntegral $ min 1000 $ ((last+1) - first)
assertFound :: EventId -> Bool -> RIO e () assertFound :: Word64 -> Bool -> RIO e ()
assertFound id found = do assertFound id found = do
unless found $ throwIO $ MissingEvent id unless found $ throwIO $ MissingEvent id

View File

@ -0,0 +1,71 @@
name: urbit-eventlog-lmdb
version: 0.10.4
license: MIT
license-file: LICENSE
library:
source-dirs: lib
ghc-options:
- -fwarn-incomplete-patterns
- -fwarn-unused-binds
- -fwarn-unused-imports
- -Werror
- -O2
dependencies:
- base
- classy-prelude
- stm
- rio
- vector
- bytestring
- lmdb-static
- conduit
- racquire
- urbit-noun-core
- urbit-noun
default-extensions:
- ApplicativeDo
- BangPatterns
- BlockArguments
- ConstraintKinds
- DataKinds
- DefaultSignatures
- DeriveAnyClass
- DeriveDataTypeable
- DeriveFoldable
- DeriveGeneric
- DeriveTraversable
- DerivingStrategies
- EmptyCase
- EmptyDataDecls
- FlexibleContexts
- FlexibleInstances
- FunctionalDependencies
- GADTs
- GeneralizedNewtypeDeriving
- LambdaCase
- MagicHash
- MultiParamTypeClasses
- NamedFieldPuns
- NoImplicitPrelude
- NumericUnderscores
- OverloadedStrings
- PackageImports
- PartialTypeSignatures
- PatternSynonyms
- QuasiQuotes
- Rank2Types
- RankNTypes
- RecordWildCards
- ScopedTypeVariables
- StandaloneDeriving
- TemplateHaskell
- TupleSections
- TypeApplications
- TypeFamilies
- TypeOperators
- UnboxedTuples
- UnicodeSyntax
- ViewPatterns

119
pkg/hs/urbit-king/TODO.md Normal file
View File

@ -0,0 +1,119 @@
# New IPC Protocol
Stubbed out:
- [x] Handle replacement events (stubbed out now b/c interface can't
handle unparsed nouns)
- [x] Handle IPC errors by killing serf process.
- [x] Handle `peek` and `pack` in `swimming` flow.
- [x] Documentation for `Urbit.Vere.Serf.IPC`.
- [x] Unstub slog/stder/dead callbacks on serf config.
- [x] Remove GoodParse hack in newRunCompute.
- [x] Bring back tank printing.
- [x] Handle serf stderr message correctly.
- [x] Bring back `logEvent`.
- [x] Snapshots should block until that event is commited to disk.
- [x] Hook up error callbacks to IO Drivers.
- [x] Do something useful with error callbacks from IO Drivers.
Bugs:
- [x] In non-daemon mode, serf slogs/stderr output that happens *before*
the terminal connects should still go to stderr.
- [x] Serf stderr should also be send (along with slogs) to all connected
terminals.
- [x] `king new` should reject pier directories that already exist.
- [x] In non-daemon-mode, ^D doesn't bring down Urbit properly.
- [x] Spinner updated multiple times with the same event, and this causes
logging of events to contain duplicates.
King-Haskell specific features:
- [x] Re-implement `collectFX` flow in Serf/Pier.
- [x] Hook up `collectFX` to CLI.
- [ ] Get `collect-all-fx` flow working again.
Performance:
- [x] Batching during replay.
- [x] Batching during normal operation.
Optimization:
- [x] IO Driver Event Prioritization
Polish:
- [x] Cleanup batching flow.
- [x] Think through how to shutdown the serf on exception.
- [x] King should shutdown promptly on ^C. Always takes 2s in practice.
- [x] Bring back progress bars.
- [x] Make sure replay progress bars go to stderr.
- [x] Logging for new IPC flow.
- [x] Logging for boot sequence.
- [x] Take snapshots on clean shutdown.
# Misc Bugs
- [ ] `king run --collect-fx` flag does nothing. Remove or implement.
- [x] Handle ^C in connected terminals. It should interrupt current
event (send SIGINT to serf, which will cause the current event to
fail promptly).
- [x] The terminal driver seems to have a race condition when spinner
changed too quickly.
# Finding the Serf Executable
- [ ] Right now, `urbit-worker` is found by looking it up in the PATH. This
is wrong, but what is right?
# Take Advantage of New IPC Features
- [ ] Hook up `scry` to drivers.
- Any immediate applications of this?
- [ ] Allow scrys to go into the %work batching flow for better latency.
- Handle event errors in other cases:
- [ ] Ames packet failures should print (but not too often).
- [ ] Incoming Http requests should produce 500 responses.
- [ ] Terminal event errors should be printed in connected terminals.
- [ ] Http client responses should be retried.
# Further IO Driver Startup Flow Betterment
Implement Pier-wide process start events
- [x] Handle %vega and exit effects.
- [x] Handle %trim effect
- [x] Inject entropy event on pier start: ``[//arvo [%wack ENT]]`
- [ ] Verbose flag: `-v` injects `[%verb ~]`
- [ ] CLI event injection: `-I file-path`. The `file-path` is a jammed
noun representing an event: `[wire card]`.
1. Just parse it as an `Ev` for now.
2. Make the serf IPC code not care about the shape of events and effects.
3. Support invalid events throughout the system (use `Lenient`?)
# Polish
- [x] Goot logging output in non-verbose mode.
- [x] Command-Line flag to re-enable verbose output.
# Cleanup
- [x] ShutdownSTM action that's passed to the terminal driver should
live in `KingEnv` and should be available to all drivers.
- [ ] Break most logic from `Main.hs` out into modules.
- [ ] Simplify `Main.hs` flows.
- [ ] Cleanup Terminal Driver code.
- [x] Spin off `racquire` into it's own package.
- [x] Spin off `urbit-noun-core` and `urbit-noun` packages.
- [x] Spin off `urbit-eventlog-lmdb` into it's own package.
- [ ] Spin off `Urbit.Vere.Serf` into it's own package
- Make it care less about the shape of events and effects.
- [ ] Spin off per-pier logic into it's own package.
- Probably `urbit-pier`

View File

@ -79,7 +79,10 @@ instance FromNoun H.StdMethod where
-- Http Server Configuration --------------------------------------------------- -- Http Server Configuration ---------------------------------------------------
newtype PEM = PEM { unPEM :: Wain } newtype PEM = PEM { unPEM :: Wain }
deriving newtype (Eq, Ord, Show, ToNoun, FromNoun) deriving newtype (Eq, Ord, ToNoun, FromNoun)
instance Show PEM where
show _ = "\"PEM (secret)\""
type Key = PEM type Key = PEM
type Cert = PEM type Cert = PEM

View File

@ -3,8 +3,8 @@
-} -}
module Urbit.Arvo.Effect where module Urbit.Arvo.Effect where
import Urbit.Noun.Time
import Urbit.Prelude import Urbit.Prelude
import Urbit.Time
import Urbit.Arvo.Common (KingId(..), ServId(..)) import Urbit.Arvo.Common (KingId(..), ServId(..))
import Urbit.Arvo.Common (Header, HttpEvent, HttpServerConf, Method, Mime) import Urbit.Arvo.Common (Header, HttpEvent, HttpServerConf, Method, Mime)
@ -82,22 +82,6 @@ data SyncEf
deriveNoun ''SyncEf deriveNoun ''SyncEf
-- UDP Effects -----------------------------------------------------------------
{-|
%init -- "I don't think that's something that can happen"
%west -- "Those also shouldn't happen"
%woot -- "Those also shouldn't happen"
-}
data AmesEf
= AmesEfInit Path ()
| AmesEfWest Path Ship Path Noun
| AmesEfWoot Path Ship (Maybe (Maybe (Term, [Tank])))
deriving (Eq, Ord, Show)
deriveNoun ''AmesEf
-- Timer Effects --------------------------------------------------------------- -- Timer Effects ---------------------------------------------------------------
{-| {-|
@ -171,7 +155,6 @@ data VaneEf
| VEHttpClient HttpClientEf | VEHttpClient HttpClientEf
| VEHttpServer HttpServerEf | VEHttpServer HttpServerEf
| VEBehn BehnEf | VEBehn BehnEf
| VEAmes AmesEf
| VETerm TermEf | VETerm TermEf
| VEClay SyncEf | VEClay SyncEf
| VESync SyncEf | VESync SyncEf
@ -203,3 +186,10 @@ instance FromNoun Ef where
ReOrg "" s "vega" p _ -> fail "%vega effect expects nil value" ReOrg "" s "vega" p _ -> fail "%vega effect expects nil value"
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"
summarizeEffect :: Lenient Ef -> Text
summarizeEffect ef =
fromNoun (toNoun ef) & \case
Nothing -> "//invalid %effect"
Just (pax :: [Cord], tag :: Cord, val :: Noun) ->
"/" <> intercalate "/" (unCord <$> pax) <> " %" <> unCord tag

View File

@ -202,9 +202,16 @@ deriveNoun ''AmesEv
-- Arvo Events ----------------------------------------------------------------- -- Arvo Events -----------------------------------------------------------------
newtype Entropy = Entropy { entropyBits :: Word512 }
deriving newtype (Eq, Ord, FromNoun, ToNoun)
instance Show Entropy where
show = const "\"ENTROPY (secret)\""
data ArvoEv data ArvoEv
= ArvoEvWhom () Ship = ArvoEvWhom () Ship
| ArvoEvWack () Word512 | ArvoEvWack () Entropy
| ArvoEvWarn Path Noun | ArvoEvWarn Path Noun
| ArvoEvCrud Path Noun | ArvoEvCrud Path Noun
| ArvoEvVeer Atom Noun | ArvoEvVeer Atom Noun
@ -350,6 +357,7 @@ instance FromNoun Ev where
ReOrg "vane" s t p v -> fmap EvVane $ 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)" ReOrg _ _ _ _ _ -> fail "First path-elem must be ?($ %vane)"
-- Short Event Names ----------------------------------------------------------- -- Short Event Names -----------------------------------------------------------
{- {-
@ -374,3 +382,10 @@ getSpinnerNameForEvent = \case
where where
isRet (TermEvBelt _ (Ret ())) = True isRet (TermEvBelt _ (Ret ())) = True
isRet _ = False isRet _ = False
summarizeEvent :: Ev -> Text
summarizeEvent ev =
fromNoun (toNoun ev) & \case
Nothing -> "//invalid %event"
Just (pax :: [Cord], tag :: Cord, val :: Noun) ->
"/" <> intercalate "/" (unCord <$> pax) <> " %" <> unCord tag

View File

@ -4,7 +4,14 @@
ships. Do it or strip it out. ships. Do it or strip it out.
-} -}
module Urbit.King.API (King(..), kingAPI, readPortsFile) where module Urbit.King.API
( King(..)
, TermConn
, TermConnAPI
, kingAPI
, readPortsFile
)
where
import RIO.Directory import RIO.Directory
import Urbit.Prelude import Urbit.Prelude
@ -12,7 +19,7 @@ import Urbit.Prelude
import Network.Socket (Socket) import Network.Socket (Socket)
import Prelude (read) import Prelude (read)
import Urbit.Arvo (Belt) import Urbit.Arvo (Belt)
import Urbit.King.App (HasConfigDir(..)) import Urbit.King.App (HasPierPath(..))
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
@ -43,16 +50,16 @@ data King = King
{-| {-|
Get the filepath of the urbit config directory and the ports file. Get the filepath of the urbit config directory and the ports file.
-} -}
portsFilePath :: HasConfigDir e => RIO e (FilePath, FilePath) portsFilePath :: HasPierPath e => RIO e (FilePath, FilePath)
portsFilePath = do portsFilePath = do
dir <- view configDirL dir <- view pierPathL
fil <- pure (dir </> ".king.ports") fil <- pure (dir </> ".king.ports")
pure (dir, fil) pure (dir, fil)
{-| {-|
Write the ports file. Write the ports file.
-} -}
portsFile :: HasConfigDir e => Word -> RAcquire e (FilePath, FilePath) portsFile :: HasPierPath e => Word -> RAcquire e (FilePath, FilePath)
portsFile por = portsFile por =
mkRAcquire mkFile (removeFile . snd) mkRAcquire mkFile (removeFile . snd)
where where
@ -65,7 +72,7 @@ portsFile por =
{-| {-|
Get the HTTP port for the running Urbit daemon. Get the HTTP port for the running Urbit daemon.
-} -}
readPortsFile :: HasConfigDir e => RIO e (Maybe Word) readPortsFile :: HasPierPath e => RIO e (Maybe Word)
readPortsFile = do readPortsFile = do
(_, fil) <- portsFilePath (_, fil) <- portsFilePath
bs <- readFile fil bs <- readFile fil
@ -86,7 +93,7 @@ kingServer is =
{-| {-|
Start the HTTP server and write to the ports file. Start the HTTP server and write to the ports file.
-} -}
kingAPI :: (HasConfigDir e, HasLogFunc e) kingAPI :: (HasPierPath e, HasLogFunc e)
=> RAcquire e King => RAcquire e King
kingAPI = do kingAPI = do
(port, sock) <- io $ W.openFreePort (port, sock) <- io $ W.openFreePort

View File

@ -2,139 +2,192 @@
Code for setting up the RIO environment. Code for setting up the RIO environment.
-} -}
module Urbit.King.App module Urbit.King.App
( App ( KingEnv
, runApp , runKingEnvStderr
, runAppLogFile , runKingEnvLogFile
, runAppNoLog , runKingEnvNoLog
, runPierApp , kingEnvKillSignal
, HasConfigDir(..) , killKingActionL
, HasStderrLogFunc(..) , onKillKingSigL
) where , PierEnv
, runPierEnv
, killPierActionL
, onKillPierSigL
, HasStderrLogFunc(..)
, HasKingId(..)
, HasProcId(..)
, HasKingEnv(..)
, HasPierEnv(..)
, module Urbit.King.Config
)
where
import Urbit.King.Config import Urbit.King.Config
import Urbit.Prelude import Urbit.Prelude
import System.Directory (createDirectoryIfMissing, getHomeDirectory) import System.Directory (createDirectoryIfMissing, getHomeDirectory)
import System.Posix.Internals (c_getpid)
import System.Posix.Types (CPid(..))
import System.Random (randomIO)
import Urbit.King.App.Class (HasStderrLogFunc(..))
--------------------------------------------------------------------------------
class HasConfigDir a where -- KingEnv ---------------------------------------------------------------------
configDirL Lens' a FilePath
class HasStderrLogFunc a where class HasKingId a where
stderrLogFuncL :: Lens' a LogFunc kingIdL :: Lens' a Word16
-------------------------------------------------------------------------------- class HasProcId a where
procIdL :: Lens' a Int32
data App = App class (HasLogFunc a, HasStderrLogFunc a, HasKingId a, HasProcId a)
{ _appLogFunc :: !LogFunc => HasKingEnv a
, _appStderrLogFunc :: !LogFunc where
} kingEnvL :: Lens' a KingEnv
makeLenses ''App data KingEnv = KingEnv
{ _kingEnvLogFunc :: !LogFunc
, _kingEnvStderrLogFunc :: !LogFunc
, _kingEnvKingId :: !Word16
, _kingEnvProcId :: !Int32
, _kingEnvKillSignal :: !(TMVar ())
}
instance HasLogFunc App where makeLenses ''KingEnv
logFuncL = appLogFunc
instance HasStderrLogFunc App where instance HasKingEnv KingEnv where
stderrLogFuncL = appStderrLogFunc kingEnvL = id
runApp :: RIO App a -> IO a instance HasLogFunc KingEnv where
runApp inner = do logFuncL = kingEnvLogFunc
logOptions <- logOptionsHandle stderr True
<&> setLogUseTime True
<&> setLogUseLoc False
withLogFunc logOptions $ \logFunc -> instance HasStderrLogFunc KingEnv where
runRIO (App logFunc logFunc) inner stderrLogFuncL = kingEnvStderrLogFunc
runAppLogFile :: RIO App a -> IO a instance HasProcId KingEnv where
runAppLogFile inner = procIdL = kingEnvProcId
withLogFileHandle $ \h -> do
logOptions <- logOptionsHandle h True
<&> setLogUseTime True
<&> setLogUseLoc False
stderrLogOptions <- logOptionsHandle stderr True
<&> setLogUseTime False
<&> setLogUseLoc False
withLogFunc stderrLogOptions $ \stderrLogFunc -> instance HasKingId KingEnv where
withLogFunc logOptions $ \logFunc -> kingIdL = kingEnvKingId
runRIO (App logFunc stderrLogFunc) inner
-- Running KingEnvs ------------------------------------------------------------
runKingEnvStderr :: Bool -> RIO KingEnv a -> IO a
runKingEnvStderr verb inner = do
logOptions <-
logOptionsHandle stderr verb <&> setLogUseTime True <&> setLogUseLoc False
withLogFunc logOptions $ \logFunc -> runKingEnv logFunc logFunc inner
runKingEnvLogFile :: Bool -> RIO KingEnv a -> IO a
runKingEnvLogFile verb inner = withLogFileHandle $ \h -> do
logOptions <-
logOptionsHandle h verb <&> setLogUseTime True <&> setLogUseLoc False
stderrLogOptions <-
logOptionsHandle stderr verb <&> setLogUseTime False <&> setLogUseLoc False
withLogFunc stderrLogOptions $ \stderrLogFunc -> withLogFunc logOptions
$ \logFunc -> runKingEnv logFunc stderrLogFunc inner
withLogFileHandle :: (Handle -> IO a) -> IO a withLogFileHandle :: (Handle -> IO a) -> IO a
withLogFileHandle act = do withLogFileHandle act = do
home <- getHomeDirectory home <- getHomeDirectory
let logDir = home </> ".urbit" let logDir = home </> ".urbit"
createDirectoryIfMissing True logDir createDirectoryIfMissing True logDir
withFile (logDir </> "king.log") AppendMode $ \handle -> do withFile (logDir </> "king.log") AppendMode $ \handle -> do
hSetBuffering handle LineBuffering hSetBuffering handle LineBuffering
act handle act handle
runAppNoLog :: RIO App a -> IO a runKingEnvNoLog :: RIO KingEnv a -> IO a
runAppNoLog act = runKingEnvNoLog act = withFile "/dev/null" AppendMode $ \handle -> do
withFile "/dev/null" AppendMode $ \handle -> do logOptions <- logOptionsHandle handle True
logOptions <- logOptionsHandle handle True withLogFunc logOptions $ \logFunc -> runKingEnv logFunc logFunc act
withLogFunc logOptions $ \logFunc ->
runRIO (App logFunc logFunc) act
-------------------------------------------------------------------------------- runKingEnv :: LogFunc -> LogFunc -> RIO KingEnv a -> IO a
runKingEnv logFunc stderr action = do
kid <- randomIO
CPid pid <- c_getpid
kil <- newEmptyTMVarIO
runRIO (KingEnv logFunc stderr kid pid kil) action
-- | A PierApp is like an App, except that it also provides a PierConfig
data PierApp = PierApp
{ _pierAppLogFunc :: !LogFunc
, _pierAppStderrLogFunc :: !LogFunc
, _pierAppPierConfig :: !PierConfig
, _pierAppNetworkConfig :: !NetworkConfig
}
makeLenses ''PierApp -- KingEnv Utils ---------------------------------------------------------------
instance HasStderrLogFunc PierApp where onKillKingSigL :: HasKingEnv e => Getter e (STM ())
stderrLogFuncL = pierAppStderrLogFunc onKillKingSigL = kingEnvL . kingEnvKillSignal . to readTMVar
instance HasLogFunc PierApp where killKingActionL :: HasKingEnv e => Getter e (STM ())
logFuncL = pierAppLogFunc killKingActionL =
kingEnvL . kingEnvKillSignal . to (\kil -> void (tryPutTMVar kil ()))
instance HasPierConfig PierApp where
pierConfigL = pierAppPierConfig
instance HasNetworkConfig PierApp where -- PierEnv ---------------------------------------------------------------------
networkConfigL = pierAppNetworkConfig
instance HasConfigDir PierApp where class (HasKingEnv a, HasPierConfig a, HasNetworkConfig a) => HasPierEnv a where
configDirL = pierAppPierConfig . pcPierPath pierEnvL :: Lens' a PierEnv
runPierApp :: PierConfig -> NetworkConfig -> Bool -> RIO PierApp a -> IO a data PierEnv = PierEnv
runPierApp pierConfig networkConfig daemon inner = { _pierEnvKingEnv :: !KingEnv
if daemon , _pierEnvPierConfig :: !PierConfig
then execStderr , _pierEnvNetworkConfig :: !NetworkConfig
else withLogFileHandle execFile , _pierEnvKillSignal :: !(TMVar ())
where }
execStderr = do
logOptions <- logOptionsHandle stderr True
<&> setLogUseTime True
<&> setLogUseLoc False
withLogFunc logOptions $ \logFunc -> makeLenses ''PierEnv
go $ PierApp { _pierAppLogFunc = logFunc
, _pierAppStderrLogFunc = logFunc
, _pierAppPierConfig = pierConfig
, _pierAppNetworkConfig = networkConfig
}
execFile logHandle = do instance HasKingEnv PierEnv where
logOptions <- logOptionsHandle logHandle True kingEnvL = pierEnvKingEnv
<&> setLogUseTime True
<&> setLogUseLoc False instance HasPierEnv PierEnv where
logStderrOptions <- logOptionsHandle stderr True pierEnvL = id
<&> setLogUseTime False
<&> setLogUseLoc False instance HasKingId PierEnv where
withLogFunc logStderrOptions $ \logStderr -> kingIdL = kingEnvL . kingEnvKingId
withLogFunc logOptions $ \logFunc ->
go $ PierApp { _pierAppLogFunc = logFunc instance HasStderrLogFunc PierEnv where
, _pierAppStderrLogFunc = logStderr stderrLogFuncL = kingEnvL . stderrLogFuncL
, _pierAppPierConfig = pierConfig
, _pierAppNetworkConfig = networkConfig instance HasLogFunc PierEnv where
} logFuncL = kingEnvL . logFuncL
go app = runRIO app inner
instance HasPierPath PierEnv where
pierPathL = pierEnvPierConfig . pierPathL
instance HasDryRun PierEnv where
dryRunL = pierEnvPierConfig . dryRunL
instance HasPierConfig PierEnv where
pierConfigL = pierEnvPierConfig
instance HasNetworkConfig PierEnv where
networkConfigL = pierEnvNetworkConfig
instance HasProcId PierEnv where
procIdL = kingEnvL . kingEnvProcId
-- PierEnv Utils ---------------------------------------------------------------
onKillPierSigL :: HasPierEnv e => Getter e (STM ())
onKillPierSigL = pierEnvL . pierEnvKillSignal . to readTMVar
killPierActionL :: HasPierEnv e => Getter e (STM ())
killPierActionL =
pierEnvL . pierEnvKillSignal . to (\kil -> void (tryPutTMVar kil ()))
-- Running Pier Envs -----------------------------------------------------------
runPierEnv
:: PierConfig -> NetworkConfig -> TMVar () -> RIO PierEnv a -> RIO KingEnv a
runPierEnv pierConfig networkConfig vKill action = do
app <- ask
let pierEnv = PierEnv { _pierEnvKingEnv = app
, _pierEnvPierConfig = pierConfig
, _pierEnvNetworkConfig = networkConfig
, _pierEnvKillSignal = vKill
}
io (runRIO pierEnv action)

View File

@ -0,0 +1,15 @@
{-|
Code for setting up the RIO environment.
-}
module Urbit.King.App.Class
( HasStderrLogFunc(..)
)
where
import Urbit.Prelude
-- KingEnv ---------------------------------------------------------------------
class HasStderrLogFunc a where
stderrLogFuncL :: Lens' a LogFunc

View File

@ -15,6 +15,12 @@ import System.Environment (getProgName)
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
data KingOpts = KingOpts
{ koSharedHttpPort :: Maybe Word16
, koSharedHttpsPort :: Maybe Word16
}
deriving (Show)
data Opts = Opts data Opts = Opts
{ oQuiet :: Bool { oQuiet :: Bool
, oHashless :: Bool , oHashless :: Bool
@ -23,6 +29,9 @@ data Opts = Opts
, oDryFrom :: Maybe Word64 , oDryFrom :: Maybe Word64
, oVerbose :: Bool , oVerbose :: Bool
, oAmesPort :: Maybe Word16 , oAmesPort :: Maybe Word16
, oNoAmes :: Bool
, oNoHttp :: Bool
, oNoHttps :: Bool
, oTrace :: Bool , oTrace :: Bool
, oCollectFx :: Bool , oCollectFx :: Bool
, oLocalhost :: Bool , oLocalhost :: Bool
@ -31,6 +40,7 @@ data Opts = Opts
, oHttpPort :: Maybe Word16 , oHttpPort :: Maybe Word16
, oHttpsPort :: Maybe Word16 , oHttpsPort :: Maybe Word16
, oLoopbackPort :: Maybe Word16 , oLoopbackPort :: Maybe Word16
, oSerfExe :: Maybe Text
} }
deriving (Show) deriving (Show)
@ -93,7 +103,7 @@ data Bug
data Cmd data Cmd
= CmdNew New Opts = CmdNew New Opts
| CmdRun Run Opts Bool | CmdRun KingOpts [(Run, Opts, Bool)]
| CmdBug Bug | CmdBug Bug
| CmdCon FilePath | CmdCon FilePath
deriving (Show) deriving (Show)
@ -221,6 +231,24 @@ opts = do
<> help "Ames port" <> help "Ames port"
<> hidden <> hidden
oNoAmes <-
switch
$ long "no-ames"
<> help "Run with Ames disabled."
<> hidden
oNoHttp <-
switch
$ long "no-http"
<> help "Run with HTTP disabled."
<> hidden
oNoHttps <-
switch
$ long "no-https"
<> help "Run with HTTPS disabled."
<> hidden
oHttpPort <- oHttpPort <-
optional optional
$ option auto $ option auto
@ -245,13 +273,18 @@ opts = do
<> help "Localhost-only HTTP port" <> help "Localhost-only HTTP port"
<> hidden <> hidden
-- Always disable hashboard. Right now, urbit is almost unusable with this oSerfExe <-
-- flag enabled and it is disabled in vere. optional
let oHashless = True $ strOption
-- oHashless <- switch $ short 'S' $ metavar "PATH"
-- <> long "hashless" <> long "serf"
-- <> help "Disable battery hashing" <> help "Path to Serf"
-- <> hidden <> hidden
oHashless <- switch $ short 'S'
<> long "hashless"
<> help "Disable battery hashing (Ignored for now)"
<> hidden
oQuiet <- switch $ short 'q' oQuiet <- switch $ short 'q'
<> long "quiet" <> long "quiet"
@ -307,15 +340,33 @@ opts = do
newShip :: Parser Cmd newShip :: Parser Cmd
newShip = CmdNew <$> new <*> opts newShip = CmdNew <$> new <*> opts
runOneShip :: Parser (Run, Opts, Bool)
runOneShip = (,,) <$> fmap Run pierPath <*> opts <*> df
where
df = switch (short 'd' <> long "daemon" <> help "Daemon mode" <> hidden)
kingOpts :: Parser KingOpts
kingOpts = do
koSharedHttpPort <-
optional
$ option auto
$ metavar "PORT"
<> long "shared-http-port"
<> help "HTTP port"
<> hidden
koSharedHttpsPort <-
optional
$ option auto
$ metavar "PORT"
<> long "shared-https-port"
<> help "HTTPS port"
<> hidden
pure (KingOpts{..})
runShip :: Parser Cmd runShip :: Parser Cmd
runShip = do runShip = CmdRun <$> kingOpts <*> some runOneShip
rPierPath <- pierPath
o <- opts
daemon <- switch $ short 'd'
<> long "daemon"
<> help "Daemon mode"
<> hidden
pure (CmdRun (Run{..}) o daemon)
valPill :: Parser Bug valPill :: Parser Bug
valPill = do valPill = do

View File

@ -1,29 +1,40 @@
{-| {-|
Pier Configuration Pier Configuration
-} -}
module Urbit.King.Config where module Urbit.King.Config where
import Urbit.Prelude import Urbit.Prelude
import qualified Urbit.Vere.Serf as Serf
{-| {-|
All the configuration data revolving around a ship and the current All the configuration data revolving around a ship and the current
execution options. execution options.
-} -}
data PierConfig = PierConfig data PierConfig = PierConfig
{ _pcPierPath :: FilePath { _pcPierPath :: FilePath
, _pcDryRun :: Bool , _pcDryRun :: Bool
} deriving (Show) , _pcSerfExe :: Text
, _pcSerfFlags :: [Serf.Flag]
} deriving (Show)
makeLenses ''PierConfig makeLenses ''PierConfig
class HasPierConfig env where class HasPierPath a where
pierConfigL :: Lens' env PierConfig pierPathL :: Lens' a FilePath
pierPathL HasPierConfig a => Lens' a FilePath class HasDryRun a where
pierPathL = pierConfigL . pcPierPath dryRunL :: Lens' a Bool
class (HasPierPath a, HasDryRun a) => HasPierConfig a where
pierConfigL :: Lens' a PierConfig
instance HasPierPath PierConfig where
pierPathL = pcPierPath
instance HasDryRun PierConfig where
dryRunL = pcDryRun
dryRunL :: HasPierConfig a => Lens' a Bool
dryRunL = pierConfigL . pcDryRun
------------------------------------------------------------------------------- -------------------------------------------------------------------------------
@ -36,6 +47,9 @@ data NetMode
data NetworkConfig = NetworkConfig data NetworkConfig = NetworkConfig
{ _ncNetMode :: NetMode { _ncNetMode :: NetMode
, _ncAmesPort :: Maybe Word16 , _ncAmesPort :: Maybe Word16
, _ncNoAmes :: Bool
, _ncNoHttp :: Bool
, _ncNoHttps :: Bool
, _ncHttpPort :: Maybe Word16 , _ncHttpPort :: Maybe Word16
, _ncHttpsPort :: Maybe Word16 , _ncHttpsPort :: Maybe Word16
, _ncLocalPort :: Maybe Word16 , _ncLocalPort :: Maybe Word16

View File

@ -10,14 +10,15 @@ import Urbit.Prelude
import Data.Conduit import Data.Conduit
import Urbit.Arvo import Urbit.Arvo
import Urbit.Time import Urbit.Noun.Time
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Control.Monad.Trans.Maybe (MaybeT(..)) import Control.Monad.Trans.Maybe (MaybeT(..))
import Urbit.Vere.Log (EventLog) import Urbit.EventLog.LMDB (EventLog)
import qualified Data.Conduit.Combinators as C import qualified Data.Conduit.Combinators as C
import qualified Urbit.Vere.Log as Log import qualified Urbit.EventLog.LMDB as Log
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
@ -39,7 +40,7 @@ run log = do
hSetEcho stdin False hSetEcho stdin False
logInfo $ displayShow (Log.identity log) logInfo $ displayShow (Log.identity log)
let cycle = fromIntegral $ lifecycleLen $ Log.identity log let cycle = fromIntegral $ lifecycleLen $ Log.identity log
las <- Log.lastEv log las <- atomically (Log.lastEv log)
loop cycle las las loop cycle las las
where where
failRead cur = failRead cur =

View File

@ -1,5 +1,25 @@
{-| {- |
King Haskell Entry Point # Signal Handling (SIGTERM, SIGINT)
We handle SIGTERM by causing the main thread to raise a `UserInterrupt`
exception. This is the same behavior as SIGINT (the signal sent upon
`CTRL-C`).
The main thread is therefore responsible for handling this exception
and causing everything to shut down properly.
# Crashing and Shutting Down
Rule number one: The King never crashes.
This rule is asperational at the moment, but it needs to become as
close to truth as possible. Shut down ships in extreme cases, but
never let the king go down.
-}
{-
TODO These some old scribbled notes. They don't belong here
anymore. Do something about it.
# Event Pruning # Event Pruning
@ -62,18 +82,18 @@ import Urbit.Arvo
import Urbit.King.Config import Urbit.King.Config
import Urbit.Vere.Dawn import Urbit.Vere.Dawn
import Urbit.Vere.Pier import Urbit.Vere.Pier
import Urbit.Vere.Eyre.Multi (multiEyre, MultiEyreApi, MultiEyreConf(..))
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Urbit.Vere.Serf import Urbit.Vere.Serf
import Urbit.King.App
import Control.Concurrent (myThreadId) import Control.Concurrent (myThreadId)
import Control.Exception (AsyncException(UserInterrupt)) import Control.Exception (AsyncException(UserInterrupt))
import Control.Lens ((&)) import Control.Lens ((&))
import System.Process (system) import System.Process (system)
import Text.Show.Pretty (pPrint) import Text.Show.Pretty (pPrint)
import Urbit.King.App (runApp, runAppLogFile, runAppNoLog, runPierApp)
import Urbit.King.App (HasConfigDir(..), HasStderrLogFunc(..))
import Urbit.Noun.Conversions (cordToUW) import Urbit.Noun.Conversions (cordToUW)
import Urbit.Time (Wen) import Urbit.Noun.Time (Wen)
import Urbit.Vere.LockFile (lockFile) import Urbit.Vere.LockFile (lockFile)
import qualified Data.Set as Set import qualified Data.Set as Set
@ -82,18 +102,14 @@ import qualified Network.HTTP.Client as C
import qualified System.Posix.Signals as Sys import qualified System.Posix.Signals as Sys
import qualified System.ProgressBar as PB import qualified System.ProgressBar as PB
import qualified System.Random as Sys import qualified System.Random as Sys
import qualified Urbit.EventLog.LMDB as Log
import qualified Urbit.King.CLI as CLI import qualified Urbit.King.CLI as CLI
import qualified Urbit.King.EventBrowser as EventBrowser import qualified Urbit.King.EventBrowser as EventBrowser
import qualified Urbit.Ob as Ob import qualified Urbit.Ob as Ob
import qualified Urbit.Vere.Log as Log
import qualified Urbit.Vere.Pier as Pier import qualified Urbit.Vere.Pier as Pier
import qualified Urbit.Vere.Serf as Serf import qualified Urbit.Vere.Serf as Serf
import qualified Urbit.Vere.Term as Term import qualified Urbit.Vere.Term as Term
--------------------------------------------------------------------------------
zod :: Ship
zod = 0
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
@ -103,28 +119,33 @@ removeFileIfExists pax = do
when exists $ do when exists $ do
removeFile pax removeFile pax
--------------------------------------------------------------------------------
toSerfFlags :: CLI.Opts -> Serf.Flags -- Compile CLI Flags to Pier Configuration -------------------------------------
{-
TODO: This is not all of the flags.
Urbit is basically useless with hashboard, so we ignore that flag.
-}
toSerfFlags :: CLI.Opts -> [Serf.Flag]
toSerfFlags CLI.Opts{..} = catMaybes m toSerfFlags CLI.Opts{..} = catMaybes m
where where
-- TODO: This is not all the flags. m = [ setFrom oQuiet Serf.Quiet
m = [ from oQuiet Serf.Quiet , setFrom oTrace Serf.Trace
, from oTrace Serf.Trace , setFrom (oHashless || True) Serf.Hashless
, from oHashless Serf.Hashless , setFrom oQuiet Serf.Quiet
, from oQuiet Serf.Quiet , setFrom oVerbose Serf.Verbose
, from oVerbose Serf.Verbose , setFrom (oDryRun || isJust oDryFrom) Serf.DryRun
, from (oDryRun || isJust oDryFrom) Serf.DryRun
] ]
from True flag = Just flag setFrom True flag = Just flag
from False _ = Nothing setFrom False _ = Nothing
toPierConfig :: FilePath -> CLI.Opts -> PierConfig toPierConfig :: FilePath -> CLI.Opts -> PierConfig
toPierConfig pierPath CLI.Opts {..} = PierConfig { .. } toPierConfig pierPath o@(CLI.Opts{..}) = PierConfig { .. }
where where
_pcPierPath = pierPath _pcPierPath = pierPath
_pcDryRun = oDryRun || isJust oDryFrom _pcDryRun = oDryRun || isJust oDryFrom
_pcSerfExe = fromMaybe "urbit-worker" oSerfExe
_pcSerfFlags = toSerfFlags o
toNetworkConfig :: CLI.Opts -> NetworkConfig toNetworkConfig :: CLI.Opts -> NetworkConfig
toNetworkConfig CLI.Opts {..} = NetworkConfig { .. } toNetworkConfig CLI.Opts {..} = NetworkConfig { .. }
@ -143,157 +164,187 @@ toNetworkConfig CLI.Opts {..} = NetworkConfig { .. }
_ncHttpPort = oHttpPort _ncHttpPort = oHttpPort
_ncHttpsPort = oHttpsPort _ncHttpsPort = oHttpsPort
_ncLocalPort = oLoopbackPort _ncLocalPort = oLoopbackPort
_ncNoAmes = oNoAmes
_ncNoHttp = oNoHttp
_ncNoHttps = oNoHttps
tryBootFromPill :: ( HasLogFunc e, HasNetworkConfig e, HasPierConfig e logStderr :: HasStderrLogFunc e => RIO LogFunc a -> RIO e a
, HasConfigDir e, HasStderrLogFunc e logStderr action = do
) logFunc <- view stderrLogFuncL
=> Bool -> Pill -> Bool -> Serf.Flags -> Ship runRIO logFunc action
-> LegacyBootEvent
-> RIO e ()
tryBootFromPill oExit pill lite flags ship boot = do
mStart <- newEmptyMVar
runOrExitImmediately bootedPier oExit mStart
where
bootedPier = do
view pierPathL >>= lockFile
rio $ logTrace "Starting boot"
sls <- Pier.booted pill lite flags ship boot
rio $ logTrace "Completed boot"
pure sls
runOrExitImmediately :: ( HasLogFunc e, HasNetworkConfig e, HasPierConfig e logSlogs :: HasStderrLogFunc e => RIO e (TVar (Text -> IO ()))
, HasConfigDir e logSlogs = logStderr $ do
) env <- ask
=> RAcquire e (Serf e, Log.EventLog, SerfState) newTVarIO (runRIO env . logOther "serf" . display . T.strip)
-> Bool
-> MVar ()
-> RIO e ()
runOrExitImmediately getPier oExit mStart =
rwith getPier $ if oExit then shutdownImmediately else runPier
where
shutdownImmediately (serf, log, ss) = do
logTrace "Sending shutdown signal"
logTrace $ displayShow ss
-- Why is this here? Do I need to force a snapshot to happen? tryBootFromPill
io $ threadDelay 500000 :: Bool
-> Pill
-> Bool
-> Ship
-> LegacyBootEvent
-> MultiEyreApi
-> RIO PierEnv ()
tryBootFromPill oExit pill lite ship boot multi = do
mStart <- newEmptyMVar
vSlog <- logSlogs
runOrExitImmediately vSlog (bootedPier vSlog) oExit mStart multi
where
bootedPier vSlog = do
view pierPathL >>= lockFile
rio $ logDebug "Starting boot"
sls <- Pier.booted vSlog pill lite ship boot
rio $ logDebug "Completed boot"
pure sls
ss <- shutdown serf 0 runOrExitImmediately
logTrace $ displayShow ss :: TVar (Text -> IO ())
logTrace "Shutdown!" -> RAcquire PierEnv (Serf, Log.EventLog)
-> Bool
-> MVar ()
-> MultiEyreApi
-> RIO PierEnv ()
runOrExitImmediately vSlog getPier oExit mStart multi = do
rwith getPier (if oExit then shutdownImmediately else runPier)
where
shutdownImmediately :: (Serf, Log.EventLog) -> RIO PierEnv ()
shutdownImmediately (serf, log) = do
logDebug "Sending shutdown signal"
Serf.stop serf
logDebug "Shutdown!"
runPier sls = do runPier :: (Serf, Log.EventLog) -> RIO PierEnv ()
runRAcquire $ Pier.pier sls mStart runPier serfLog = do
runRAcquire (Pier.pier serfLog vSlog mStart multi)
tryPlayShip :: ( HasStderrLogFunc e, HasLogFunc e, HasNetworkConfig e tryPlayShip
, HasPierConfig e, HasConfigDir e :: Bool
) -> Bool
=> Bool -> Bool -> Maybe Word64 -> Serf.Flags -> MVar () -> RIO e () -> Maybe Word64
tryPlayShip exitImmediately fullReplay playFrom flags mStart = do -> MVar ()
when fullReplay wipeSnapshot -> MultiEyreApi
runOrExitImmediately resumeShip exitImmediately mStart -> RIO PierEnv ()
where tryPlayShip exitImmediately fullReplay playFrom mStart multi = do
wipeSnapshot = do when fullReplay wipeSnapshot
shipPath <- view pierPathL vSlog <- logSlogs
logTrace "wipeSnapshot" runOrExitImmediately vSlog (resumeShip vSlog) exitImmediately mStart multi
logDebug $ display $ pack @Text ("Wiping " <> north shipPath) where
logDebug $ display $ pack @Text ("Wiping " <> south shipPath) wipeSnapshot = do
removeFileIfExists (north shipPath) shipPath <- view pierPathL
removeFileIfExists (south shipPath) logDebug "wipeSnapshot"
logDebug $ display $ pack @Text ("Wiping " <> north shipPath)
logDebug $ display $ pack @Text ("Wiping " <> south shipPath)
removeFileIfExists (north shipPath)
removeFileIfExists (south shipPath)
north shipPath = shipPath <> "/.urb/chk/north.bin" north shipPath = shipPath <> "/.urb/chk/north.bin"
south shipPath = shipPath <> "/.urb/chk/south.bin" south shipPath = shipPath <> "/.urb/chk/south.bin"
resumeShip = do resumeShip :: TVar (Text -> IO ()) -> RAcquire PierEnv (Serf, Log.EventLog)
view pierPathL >>= lockFile resumeShip vSlog = do
rio $ logTrace "RESUMING SHIP" view pierPathL >>= lockFile
sls <- Pier.resumed playFrom flags rio $ logDebug "RESUMING SHIP"
rio $ logTrace "SHIP RESUMED" sls <- Pier.resumed vSlog playFrom
pure sls rio $ logDebug "SHIP RESUMED"
pure sls
runRAcquire :: (MonadUnliftIO (m e), MonadIO (m e), MonadReader e (m e)) runRAcquire :: (MonadUnliftIO (m e), MonadIO (m e), MonadReader e (m e))
=> RAcquire e a -> m e a => RAcquire e a -> m e a
runRAcquire act = rwith act pure runRAcquire act = rwith act pure
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
checkEvs :: forall e. HasLogFunc e => FilePath -> Word64 -> Word64 -> RIO e () checkEvs :: FilePath -> Word64 -> Word64 -> RIO KingEnv ()
checkEvs pierPath first last = do checkEvs pierPath first last = do
rwith (Log.existing logPath) $ \log -> do rwith (Log.existing logPath) $ \log -> do
let ident = Log.identity log let ident = Log.identity log
let pbSty = PB.defStyle { PB.stylePostfix = PB.exact } let pbSty = PB.defStyle { PB.stylePostfix = PB.exact }
logTrace (displayShow ident) logDebug (displayShow ident)
last <- Log.lastEv log <&> \lastReal -> min last lastReal last <- atomically $ Log.lastEv log <&> \lastReal -> min last lastReal
let evCount = fromIntegral (last - first) let evCount = fromIntegral (last - first)
pb <- PB.newProgressBar pbSty 10 (PB.Progress 1 evCount ()) pb <- PB.newProgressBar pbSty 10 (PB.Progress 1 evCount ())
runConduit $ Log.streamEvents log first runConduit $ Log.streamEvents log first .| showEvents
.| showEvents pb first (fromIntegral $ lifecycleLen ident) pb
where first
logPath :: FilePath (fromIntegral $ lifecycleLen ident)
logPath = pierPath <> "/.urb/log" where
logPath :: FilePath
logPath = pierPath <> "/.urb/log"
showEvents :: PB.ProgressBar () -> EventId -> EventId showEvents
-> ConduitT ByteString Void (RIO e) () :: PB.ProgressBar ()
showEvents pb eId _ | eId > last = pure () -> EventId
showEvents pb eId cycle = await >>= \case -> EventId
Nothing -> do -> ConduitT ByteString Void (RIO KingEnv) ()
lift $ PB.killProgressBar pb showEvents pb eId _ | eId > last = pure ()
lift $ logTrace "Everything checks out." showEvents pb eId cycle = await >>= \case
Just bs -> do Nothing -> do
lift $ PB.incProgress pb 1 lift $ PB.killProgressBar pb
lift $ do lift $ logDebug "Everything checks out."
n <- io $ cueBSExn bs Just bs -> do
when (eId > cycle) $ do lift $ PB.incProgress pb 1
(mug, wen, evNoun) <- unpackJob n lift $ do
fromNounErr evNoun & \case n <- io $ cueBSExn bs
Left err -> logError (displayShow (eId, err)) when (eId > cycle) $ do
Right (_ Ev) -> pure () (mug, wen, evNoun) <- unpackJob n
showEvents pb (succ eId) cycle fromNounErr evNoun & \case
Left err -> logError (displayShow (eId, err))
Right (_ :: Ev) -> pure ()
showEvents pb (succ eId) cycle
unpackJob :: Noun -> RIO KingEnv (Mug, Wen, Noun)
unpackJob = io . fromNounExn
unpackJob :: Noun -> RIO e (Mug, Wen, Noun)
unpackJob = io . fromNounExn
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
collectAllFx :: FilePath -> RIO KingEnv ()
collectAllFx = error "TODO"
{-
{-| {-|
This runs the serf at `$top/.tmpdir`, but we disable snapshots, This runs the serf at `$top/.tmpdir`, but we disable snapshots,
so this should never actually be created. We just do this to avoid so this should never actually be created. We just do this to avoid
letting the serf use an existing snapshot. letting the serf use an existing snapshot.
-} -}
collectAllFx :: e. HasLogFunc e => FilePath -> RIO e () collectAllFx :: FilePath -> RIO KingEnv ()
collectAllFx top = do collectAllFx top = do
logTrace $ display $ pack @Text top logDebug $ display $ pack @Text top
rwith collectedFX $ \() -> vSlog <- logSlogs
logTrace "Done collecting effects!" rwith (collectedFX vSlog) $ \() ->
logDebug "Done collecting effects!"
where where
tmpDir :: FilePath tmpDir :: FilePath
tmpDir = top </> ".tmpdir" tmpDir = top </> ".tmpdir"
collectedFX :: RAcquire e () collectedFX :: TVar (Text -> IO ()) -> RAcquire KingEnv ()
collectedFX = do collectedFX vSlog = do
lockFile top lockFile top
log <- Log.existing (top <> "/.urb/log") log <- Log.existing (top <> "/.urb/log")
serf <- Serf.run (Serf.Config tmpDir serfFlags) serf <- Pier.runSerf vSlog tmpDir serfFlags
rio $ Serf.collectFX serf log rio $ Serf.collectFX serf log
serfFlags :: Serf.Flags serfFlags :: [Serf.Flag]
serfFlags = [Serf.Hashless, Serf.DryRun] serfFlags = [Serf.Hashless, Serf.DryRun]
-}
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
replayPartEvs :: e. (HasStderrLogFunc e, HasLogFunc e) replayPartEvs :: FilePath -> Word64 -> RIO KingEnv ()
=> FilePath -> Word64 -> RIO e ()
replayPartEvs top last = do replayPartEvs top last = do
logTrace $ display $ pack @Text top logDebug $ display $ pack @Text top
fetchSnapshot fetchSnapshot
rwith replayedEvs $ \() -> rwith replayedEvs $ \() ->
logTrace "Done replaying events!" logDebug "Done replaying events!"
where where
fetchSnapshot :: RIO e () fetchSnapshot :: RIO KingEnv ()
fetchSnapshot = do fetchSnapshot = do
snap <- Pier.getSnapshot top last snap <- Pier.getSnapshot top last
case snap of case snap of
@ -305,20 +356,28 @@ replayPartEvs top last = do
tmpDir :: FilePath tmpDir :: FilePath
tmpDir = top </> ".partial-replay" </> show last tmpDir = top </> ".partial-replay" </> show last
replayedEvs :: RAcquire e () replayedEvs :: RAcquire KingEnv ()
replayedEvs = do replayedEvs = do
lockFile top lockFile top
log <- Log.existing (top <> "/.urb/log") log <- Log.existing (top <> "/.urb/log")
serf <- Serf.run (Serf.Config tmpDir serfFlags) let onSlog = print
let onStdr = print
let onDead = error "DIED"
let config = Serf.Config "urbit-worker" tmpDir serfFlags onSlog onStdr onDead
(serf, info) <- io (Serf.start config)
rio $ do rio $ do
ss <- Serf.replay serf log $ Just last eSs <- Serf.execReplay serf log (Just last)
Serf.snapshot serf ss case eSs of
Left bail -> error (show bail)
Right 0 -> io (Serf.snapshot serf)
Right num -> pure ()
io $ threadDelay 500000 -- Copied from runOrExitImmediately io $ threadDelay 500000 -- Copied from runOrExitImmediately
pure () pure ()
serfFlags :: Serf.Flags serfFlags :: [Serf.Flag]
serfFlags = [Serf.Hashless] serfFlags = [Serf.Hashless]
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
{-| {-|
@ -326,84 +385,98 @@ replayPartEvs top last = do
-} -}
testPill :: HasLogFunc e => FilePath -> Bool -> Bool -> RIO e () testPill :: HasLogFunc e => FilePath -> Bool -> Bool -> RIO e ()
testPill pax showPil showSeq = do testPill pax showPil showSeq = do
putStrLn "Reading pill file." logDebug "Reading pill file."
pillBytes <- readFile pax pillBytes <- readFile pax
putStrLn "Cueing pill file." logDebug "Cueing pill file."
pillNoun <- io $ cueBS pillBytes & either throwIO pure pillNoun <- io $ cueBS pillBytes & either throwIO pure
putStrLn "Parsing pill file." logDebug "Parsing pill file."
pill <- fromNounErr pillNoun & either (throwIO . uncurry ParseErr) pure pill <- fromNounErr pillNoun & either (throwIO . uncurry ParseErr) pure
putStrLn "Using pill to generate boot sequence." logDebug "Using pill to generate boot sequence."
bootSeq <- generateBootSeq zod pill False (Fake $ Ship 0) bootSeq <- genBootSeq (Ship 0) pill False (Fake (Ship 0))
putStrLn "Validate jam/cue and toNoun/fromNoun on pill value" logDebug "Validate jam/cue and toNoun/fromNoun on pill value"
reJam <- validateNounVal pill reJam <- validateNounVal pill
putStrLn "Checking if round-trip matches input file:" logDebug "Checking if round-trip matches input file:"
unless (reJam == pillBytes) $ do unless (reJam == pillBytes) $ do
putStrLn " Our jam does not match the file...\n" logDebug " Our jam does not match the file...\n"
putStrLn " This is surprising, but it is probably okay." logDebug " This is surprising, but it is probably okay."
when showPil $ do when showPil $ do
putStrLn "\n\n== Pill ==\n" logDebug "\n\n== Pill ==\n"
io $ pPrint pill io $ pPrint pill
when showSeq $ do when showSeq $ do
putStrLn "\n\n== Boot Sequence ==\n" logDebug "\n\n== Boot Sequence ==\n"
io $ pPrint bootSeq io $ pPrint bootSeq
validateNounVal :: (HasLogFunc e, Eq a, ToNoun a, FromNoun a) validateNounVal :: (HasLogFunc e, Eq a, ToNoun a, FromNoun a)
=> a -> RIO e ByteString => a -> RIO e ByteString
validateNounVal inpVal = do validateNounVal inpVal = do
putStrLn " jam" logDebug " jam"
inpByt <- evaluate $ jamBS $ toNoun inpVal inpByt <- evaluate $ jamBS $ toNoun inpVal
putStrLn " cue" logDebug " cue"
outNon <- cueBS inpByt & either throwIO pure outNon <- cueBS inpByt & either throwIO pure
putStrLn " fromNoun" logDebug " fromNoun"
outVal <- fromNounErr outNon & either (throwIO . uncurry ParseErr) pure outVal <- fromNounErr outNon & either (throwIO . uncurry ParseErr) pure
putStrLn " toNoun" logDebug " toNoun"
outNon <- evaluate (toNoun outVal) outNon <- evaluate (toNoun outVal)
putStrLn " jam" logDebug " jam"
outByt <- evaluate $ jamBS outNon outByt <- evaluate $ jamBS outNon
putStrLn "Checking if: x == cue (jam x)" logDebug "Checking if: x == cue (jam x)"
unless (inpVal == outVal) $ unless (inpVal == outVal) $
error "Value fails test: x == cue (jam x)" error "Value fails test: x == cue (jam x)"
putStrLn "Checking if: jam x == jam (cue (jam x))" logDebug "Checking if: jam x == jam (cue (jam x))"
unless (inpByt == outByt) $ unless (inpByt == outByt) $
error "Value fails test: jam x == jam (cue (jam x))" error "Value fails test: jam x == jam (cue (jam x))"
pure outByt pure outByt
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
pillFrom :: CLI.PillSource -> RIO e Pill pillFrom :: CLI.PillSource -> RIO KingEnv Pill
pillFrom = \case
CLI.PillSourceFile pillPath -> do
logDebug $ display $ "boot: reading pill from " ++ (pack pillPath :: Text)
io (loadFile pillPath >>= either throwIO pure)
pillFrom (CLI.PillSourceFile pillPath) = do CLI.PillSourceURL url -> do
putStrLn $ "boot: reading pill from " ++ pack pillPath logDebug $ display $ "boot: retrieving pill from " ++ (pack url :: Text)
io (loadFile pillPath >>= either throwIO pure) -- Get the jamfile with the list of stars accepting comets right now.
manager <- io $ C.newManager tlsManagerSettings
request <- io $ C.parseRequest url
response <- io $ C.httpLbs (C.setRequestCheckStatus request) manager
let body = toStrict $ C.responseBody response
pillFrom (CLI.PillSourceURL url) = do noun <- cueBS body & either throwIO pure
putStrLn $ "boot: retrieving pill from " ++ pack url fromNounErr noun & either (throwIO . uncurry ParseErr) pure
-- Get the jamfile with the list of stars accepting comets right now.
manager <- io $ C.newManager tlsManagerSettings
request <- io $ C.parseRequest url
response <- io $ C.httpLbs (C.setRequestCheckStatus request) manager
let body = toStrict $ C.responseBody response
noun <- cueBS body & either throwIO pure newShip :: CLI.New -> CLI.Opts -> RIO KingEnv ()
fromNounErr noun & either (throwIO . uncurry ParseErr) pure newShip CLI.New{..} opts = do
{-
TODO XXX HACK
newShip :: forall e. HasLogFunc e => CLI.New -> CLI.Opts -> RIO e () Because the "new ship" flow *may* automatically start the ship,
newShip CLI.New{..} opts we need to create this, but it's not actually correct.
| CLI.BootComet <- nBootType = do
The right solution is to separate out the "new ship" flow from the
"run ship" flow, and possibly sequence them from the outside if
that's really needed.
-}
multi <- multiEyre (MultiEyreConf Nothing Nothing True)
case nBootType of
CLI.BootComet -> do
pill <- pillFrom nPillSource pill <- pillFrom nPillSource
putStrLn "boot: retrieving list of stars currently accepting comets" putStrLn "boot: retrieving list of stars currently accepting comets"
starList <- dawnCometList starList <- dawnCometList
@ -413,14 +486,14 @@ newShip CLI.New{..} opts
eny <- io $ Sys.randomIO eny <- io $ Sys.randomIO
let seed = mineComet (Set.fromList starList) eny let seed = mineComet (Set.fromList starList) eny
putStrLn ("boot: found comet " ++ renderShip (sShip seed)) putStrLn ("boot: found comet " ++ renderShip (sShip seed))
bootFromSeed pill seed bootFromSeed multi pill seed
| CLI.BootFake name <- nBootType = do CLI.BootFake name -> do
pill <- pillFrom nPillSource pill <- pillFrom nPillSource
ship <- shipFrom name ship <- shipFrom name
runTryBootFromPill pill name ship (Fake ship) runTryBootFromPill multi pill name ship (Fake ship)
| CLI.BootFromKeyfile keyFile <- nBootType = do CLI.BootFromKeyfile keyFile -> do
text <- readFileUtf8 keyFile text <- readFileUtf8 keyFile
asAtom <- case cordToUW (Cord $ T.strip text) of asAtom <- case cordToUW (Cord $ T.strip text) of
Nothing -> error "Couldn't parse keyfile. Hint: keyfiles start with 0w?" Nothing -> error "Couldn't parse keyfile. Hint: keyfiles start with 0w?"
@ -433,10 +506,10 @@ newShip CLI.New{..} opts
pill <- pillFrom nPillSource pill <- pillFrom nPillSource
bootFromSeed pill seed bootFromSeed multi pill seed
where where
shipFrom :: Text -> RIO e Ship shipFrom :: Text -> RIO KingEnv Ship
shipFrom name = case Ob.parsePatp name of shipFrom name = case Ob.parsePatp name of
Left x -> error "Invalid ship name" Left x -> error "Invalid ship name"
Right p -> pure $ Ship $ fromIntegral $ Ob.fromPatp p Right p -> pure $ Ship $ fromIntegral $ Ob.fromPatp p
@ -446,7 +519,7 @@ newShip CLI.New{..} opts
Just x -> x Just x -> x
Nothing -> "./" <> unpack name Nothing -> "./" <> unpack name
nameFromShip :: Ship -> RIO e Text nameFromShip :: Ship -> RIO KingEnv Text
nameFromShip s = name nameFromShip s = name
where where
nameWithSig = Ob.renderPatp $ Ob.patp $ fromIntegral s nameWithSig = Ob.renderPatp $ Ob.patp $ fromIntegral s
@ -454,8 +527,8 @@ newShip CLI.New{..} opts
Nothing -> error "Urbit.ob didn't produce string with ~" Nothing -> error "Urbit.ob didn't produce string with ~"
Just x -> pure x Just x -> pure x
bootFromSeed :: Pill -> Seed -> RIO e () bootFromSeed :: MultiEyreApi -> Pill -> Seed -> RIO KingEnv ()
bootFromSeed pill seed = do bootFromSeed multi pill seed = do
ethReturn <- dawnVent seed ethReturn <- dawnVent seed
case ethReturn of case ethReturn of
@ -463,43 +536,51 @@ newShip CLI.New{..} opts
Right dawn -> do Right dawn -> do
let ship = sShip $ dSeed dawn let ship = sShip $ dSeed dawn
name <- nameFromShip ship name <- nameFromShip ship
runTryBootFromPill pill name ship (Dawn dawn) runTryBootFromPill multi pill name ship (Dawn dawn)
flags = toSerfFlags opts
-- Now that we have all the information for running an application with a -- Now that we have all the information for running an application with a
-- PierConfig, do so. -- PierConfig, do so.
runTryBootFromPill pill name ship bootEvent = do runTryBootFromPill multi pill name ship bootEvent = do
vKill <- view kingEnvKillSignal
let pierConfig = toPierConfig (pierPath name) opts let pierConfig = toPierConfig (pierPath name) opts
let networkConfig = toNetworkConfig opts let networkConfig = toNetworkConfig opts
io $ runPierApp pierConfig networkConfig True $ runPierEnv pierConfig networkConfig vKill $
tryBootFromPill True pill nLite flags ship bootEvent tryBootFromPill True pill nLite ship bootEvent multi
------ tryBootFromPill (CLI.oExit opts) pill nLite flags ship bootEvent ------ tryBootFromPill (CLI.oExit opts) pill nLite flags ship bootEvent
runShipEnv :: CLI.Run -> CLI.Opts -> TMVar () -> RIO PierEnv a -> RIO KingEnv a
runShipEnv (CLI.Run pierPath) opts vKill act = do
runPierEnv pierConfig netConfig vKill act
where
pierConfig = toPierConfig pierPath opts
netConfig = toNetworkConfig opts
runShip :: CLI.Run -> CLI.Opts -> Bool -> IO () runShip
runShip (CLI.Run pierPath) opts daemon = do :: CLI.Run -> CLI.Opts -> Bool -> MultiEyreApi -> RIO PierEnv ()
tid <- myThreadId runShip (CLI.Run pierPath) opts daemon multi = do
let onTermExit = throwTo tid UserInterrupt mStart <- newEmptyMVar
mStart <- newEmptyMVar
if daemon if daemon
then runPier mStart then runPier mStart
else do else do
-- Wait until the pier has started up, then connect a terminal. If
-- the terminal ever shuts down, ask the ship to go down.
connectionThread <- async $ do connectionThread <- async $ do
readMVar mStart readMVar mStart
finally (runAppNoLog $ connTerm pierPath) onTermExit finally (connTerm pierPath) $ do
finally (runPier mStart) (cancel connectionThread) view killPierActionL >>= atomically
-- Run the pier until it finishes, and then kill the terminal.
finally (runPier mStart) $ do
cancel connectionThread
where where
runPier mStart = runPier :: MVar () -> RIO PierEnv ()
runPierApp pierConfig networkConfig daemon $ runPier mStart = do
tryPlayShip tryPlayShip
(CLI.oExit opts) (CLI.oExit opts)
(CLI.oFullReplay opts) (CLI.oFullReplay opts)
(CLI.oDryFrom opts) (CLI.oDryFrom opts)
(toSerfFlags opts) mStart
mStart multi
pierConfig = toPierConfig pierPath opts
networkConfig = toNetworkConfig opts
startBrowser :: HasLogFunc e => FilePath -> RIO e () startBrowser :: HasLogFunc e => FilePath -> RIO e ()
@ -540,33 +621,200 @@ checkComet = do
main :: IO () main :: IO ()
main = do main = do
args <- CLI.parseArgs
hSetBuffering stdout NoBuffering
setupSignalHandlers
runKingEnv args $ case args of
CLI.CmdRun ko ships -> runShips ko ships
CLI.CmdNew n o -> newShip n o
CLI.CmdBug (CLI.CollectAllFX pax ) -> collectAllFx pax
CLI.CmdBug (CLI.EventBrowser pax ) -> startBrowser pax
CLI.CmdBug (CLI.ValidatePill pax pil s) -> testPill pax pil s
CLI.CmdBug (CLI.ValidateEvents pax f l) -> checkEvs pax f l
CLI.CmdBug (CLI.ValidateFX pax f l) -> checkFx pax f l
CLI.CmdBug (CLI.ReplayEvents pax l ) -> replayPartEvs pax l
CLI.CmdBug (CLI.CheckDawn pax ) -> checkDawn pax
CLI.CmdBug CLI.CheckComet -> checkComet
CLI.CmdCon pier -> connTerm pier
where
runKingEnv args =
let verb = verboseLogging args
in if willRunTerminal args
then runKingEnvLogFile verb
else runKingEnvStderr verb
setupSignalHandlers = do
mainTid <- myThreadId mainTid <- myThreadId
let onKillSig = throwTo mainTid UserInterrupt
for_ [Sys.sigTERM, Sys.sigINT] $ \sig -> do
Sys.installHandler sig (Sys.Catch onKillSig) Nothing
hSetBuffering stdout NoBuffering verboseLogging :: CLI.Cmd -> Bool
verboseLogging = \case
CLI.CmdRun ko ships -> any CLI.oVerbose (ships <&> \(_, o, _) -> o)
_ -> False
let onTermSig = throwTo mainTid UserInterrupt willRunTerminal :: CLI.Cmd -> Bool
willRunTerminal = \case
CLI.CmdCon _ -> True
CLI.CmdRun ko [(_,_,daemon)] -> not daemon
CLI.CmdRun ko _ -> False
_ -> False
Sys.installHandler Sys.sigTERM (Sys.Catch onTermSig) Nothing
CLI.parseArgs >>= \case {-
CLI.CmdRun r o d -> runShip r o d Runs a ship but restarts it if it crashes or shuts down on it's own.
CLI.CmdNew n o -> runApp $ newShip n o
CLI.CmdBug (CLI.CollectAllFX pax) -> runApp $ collectAllFx pax Once `waitForKillRequ` returns, the ship will be terminated and this
CLI.CmdBug (CLI.EventBrowser pax) -> runApp $ startBrowser pax routine will exit.
CLI.CmdBug (CLI.ValidatePill pax pil s) -> runApp $ testPill pax pil s
CLI.CmdBug (CLI.ValidateEvents pax f l) -> runApp $ checkEvs pax f l TODO Use logging system instead of printing.
CLI.CmdBug (CLI.ValidateFX pax f l) -> runApp $ checkFx pax f l -}
CLI.CmdBug (CLI.ReplayEvents pax l) -> runApp $ replayPartEvs pax l runShipRestarting
CLI.CmdBug (CLI.CheckDawn pax) -> runApp $ checkDawn pax :: CLI.Run -> CLI.Opts -> MultiEyreApi -> RIO KingEnv ()
CLI.CmdBug CLI.CheckComet -> runApp $ checkComet runShipRestarting r o multi = do
CLI.CmdCon pier -> runAppLogFile $ connTerm pier let pier = pack (CLI.rPierPath r)
loop = runShipRestarting r o multi
onKill <- view onKillKingSigL
vKillPier <- newEmptyTMVarIO
tid <- asyncBound $ runShipEnv r o vKillPier $ runShip r o True multi
let onShipExit = Left <$> waitCatchSTM tid
onKillRequ = Right <$> onKill
atomically (onShipExit <|> onKillRequ) >>= \case
Left exit -> do
case exit of
Left err -> logError $ display (tshow err <> ": " <> pier)
Right () ->
logError $ display ("Ship exited on it's own. Why? " <> pier)
threadDelay 250_000
loop
Right () -> do
logTrace $ display (pier <> " shutdown requested")
race_ (wait tid) $ do
threadDelay 5_000_000
logDebug $ display (pier <> " not down after 5s, killing with fire.")
cancel tid
logTrace $ display ("Ship terminated: " <> pier)
{-
TODO This is messy and shared a lot of logic with `runShipRestarting`.
-}
runShipNoRestart
:: CLI.Run -> CLI.Opts -> Bool -> MultiEyreApi -> RIO KingEnv ()
runShipNoRestart r o d multi = do
vKill <- view kingEnvKillSignal -- killing ship same as killing king
tid <- asyncBound (runShipEnv r o vKill $ runShip r o d multi)
onKill <- view onKillKingSigL
let pier = pack (CLI.rPierPath r)
let onShipExit = Left <$> waitCatchSTM tid
onKillRequ = Right <$> onKill
atomically (onShipExit <|> onKillRequ) >>= \case
Left (Left err) -> do
logError $ display (tshow err <> ": " <> pier)
Left (Right ()) -> do
logError $ display (pier <> " exited on it's own. Why?")
Right () -> do
logTrace $ display (pier <> " shutdown requested")
race_ (wait tid) $ do
threadDelay 5_000_000
logTrace $ display (pier <> " not down after 5s, killing with fire.")
cancel tid
logTrace $ display (pier <> " terminated.")
runShips :: CLI.KingOpts -> [(CLI.Run, CLI.Opts, Bool)] -> RIO KingEnv ()
runShips CLI.KingOpts {..} ships = do
let meConf = MultiEyreConf
{ mecHttpPort = fromIntegral <$> koSharedHttpPort
, mecHttpsPort = fromIntegral <$> koSharedHttpsPort
, mecLocalhostOnly = False -- TODO Localhost-only needs to be
-- a king-wide option.
}
{-
TODO Need to rework RIO environment to fix this. Should have a
bunch of nested contexts:
- King has started. King has Id. Logging available.
- In running environment. MultiEyre and global config available.
- In pier environment: pier path and config available.
- In running ship environment: serf state, event queue available.
-}
multi <- multiEyre meConf
go multi ships
where
go :: MultiEyreApi -> [(CLI.Run, CLI.Opts, Bool)] -> RIO KingEnv ()
go me = \case
[] -> pure ()
[rod] -> runSingleShip rod me
ships -> runMultipleShips (ships <&> \(r, o, _) -> (r, o)) me
-- TODO Duplicated logic.
runSingleShip :: (CLI.Run, CLI.Opts, Bool) -> MultiEyreApi -> RIO KingEnv ()
runSingleShip (r, o, d) multi = do
shipThread <- async (runShipNoRestart r o d multi)
{-
Wait for the ship to go down.
Since `waitCatch` will never throw an exception, the `onException`
block will only happen if this thread is killed with an async
exception. The one we expect is `UserInterrupt` which will be raised
on this thread upon SIGKILL or SIGTERM.
If this thread is killed, we first ask the ship to go down, wait
for the ship to actually go down, and then go down ourselves.
-}
onException (void $ waitCatch shipThread) $ do
logTrace "KING IS GOING DOWN"
atomically =<< view killKingActionL
waitCatch shipThread
pure ()
runMultipleShips :: [(CLI.Run, CLI.Opts)] -> MultiEyreApi -> RIO KingEnv ()
runMultipleShips ships multi = do
shipThreads <- for ships $ \(r, o) -> do
async (runShipRestarting r o multi)
{-
Since `spin` never returns, this will run until the main
thread is killed with an async exception. The one we expect is
`UserInterrupt` which will be raised on this thread upon SIGKILL
or SIGTERM.
Once that happens, we send a shutdown signal which will cause all
ships to be shut down, and then we `wait` for them to finish before
returning.
This is different than the single-ship flow, because ships never
go down on their own in this flow. If they go down, they just bring
themselves back up.
-}
let spin = forever (threadDelay maxBound)
finally spin $ do
logTrace "KING IS GOING DOWN"
view killKingActionL >>= atomically
for_ shipThreads waitCatch
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
connTerm :: e. HasLogFunc e => FilePath -> RIO e () connTerm :: e. HasLogFunc e => FilePath -> RIO e ()
connTerm pier = connTerm = Term.runTerminalClient
Term.runTerminalClient pier
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------

View File

@ -1,87 +1,162 @@
{-| {-|
Ames IO Driver -- UDP Ames IO Driver
-} -}
module Urbit.Vere.Ames (ames) where module Urbit.Vere.Ames (ames, ames', PacketOutcome(..)) where
import Urbit.Prelude import Urbit.Prelude
import Control.Monad.Extra hiding (mapM_) import Network.Socket hiding (recvFrom, sendTo)
import Network.Socket hiding (recvFrom, sendTo) import Urbit.Arvo hiding (Fake)
import Network.Socket.ByteString
import Urbit.Arvo hiding (Fake)
import Urbit.King.Config import Urbit.King.Config
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import qualified Data.ByteString as BS import Urbit.King.App (HasKingId(..), HasPierEnv(..))
import qualified Data.Map as M import Urbit.Vere.Ames.DNS (NetworkMode(..), ResolvServ(..))
import qualified Urbit.Ob as Ob import Urbit.Vere.Ames.DNS (galaxyPort, resolvServ)
import qualified Urbit.Time as Time import Urbit.Vere.Ames.UDP (UdpServ(..), fakeUdpServ, realUdpServ)
-- Constants -------------------------------------------------------------------
-- | How many unprocessed ames packets to allow in the queue before we start
-- dropping incoming packets.
queueBound :: Word
queueBound = 1000
-- | How often, measured in number of packets dropped, we should announce packet
-- loss.
packetsDroppedPerComplaint :: Word
packetsDroppedPerComplaint = 1000
-- Types ----------------------------------------------------------------------- -- Types -----------------------------------------------------------------------
data AmesDrv = AmesDrv data AmesDrv = AmesDrv
{ aTurfs :: TVar (Maybe [Turf]) { aTurfs :: TVar (Maybe [Turf])
, aGalaxies :: IORef (M.Map Galaxy (Async (), TQueue ByteString)) , aDropped :: TVar Word
, aSocket :: TVar (Maybe Socket) , aUdpServ :: UdpServ
, aListener :: Async () , aResolvr :: ResolvServ
, aSendingQueue :: TQueue (SockAddr, ByteString) , aRecvTid :: Async ()
, aSendingThread :: Async ()
} }
data NetworkMode = Fake | Localhost | Real | NoNetwork data PacketOutcome
deriving (Eq, Ord, Show) = Intake
| Ouster
-- Utils ----------------------------------------------------------------------- -- Utils -----------------------------------------------------------------------
galaxyPort :: NetworkMode -> Galaxy -> PortNumber
galaxyPort Fake (Patp g) = fromIntegral g + 31337
galaxyPort Localhost (Patp g) = fromIntegral g + 13337
galaxyPort Real (Patp g) = fromIntegral g + 13337
galaxyPort NoNetwork _ = fromIntegral 0
listenPort :: NetworkMode -> Ship -> PortNumber listenPort :: NetworkMode -> Ship -> PortNumber
listenPort m s | s < 256 = galaxyPort m (fromIntegral s) listenPort m s | s < 256 = galaxyPort m (fromIntegral s)
listenPort m _ = 0 listenPort m _ = 0 -- I don't care, just give me any port.
localhost :: HostAddress localhost :: HostAddress
localhost = tupleToHostAddress (127,0,0,1) localhost = tupleToHostAddress (127, 0, 0, 1)
inaddrAny :: HostAddress inaddrAny :: HostAddress
inaddrAny = tupleToHostAddress (0,0,0,0) inaddrAny = tupleToHostAddress (0, 0, 0, 0)
okayFakeAddr :: AmesDest -> Bool modeAddress :: NetworkMode -> Maybe HostAddress
okayFakeAddr = \case modeAddress = \case
EachYes _ -> True Fake -> Just localhost
EachNo (Jammed (AAIpv4 (Ipv4 a) _)) -> a == localhost Localhost -> Just localhost
EachNo (Jammed (AAVoid v)) -> absurd v Real -> Just inaddrAny
NoNetwork -> Nothing
localhostSockAddr :: NetworkMode -> AmesDest -> SockAddr okFakeAddr :: AmesDest -> Bool
localhostSockAddr mode = \case okFakeAddr = \case
EachYes g -> SockAddrInet (galaxyPort mode g) localhost EachYes _ -> True
EachNo (Jammed (AAIpv4 _ p)) -> SockAddrInet (fromIntegral p) localhost EachNo (Jammed (AAIpv4 (Ipv4 a) _)) -> a == localhost
EachNo (Jammed (AAVoid v)) -> absurd v EachNo (Jammed (AAVoid v )) -> absurd v
localAddr :: NetworkMode -> AmesDest -> SockAddr
localAddr mode = \case
EachYes g -> SockAddrInet (galaxyPort mode g) localhost
EachNo (Jammed (AAIpv4 _ p)) -> SockAddrInet (fromIntegral p) localhost
EachNo (Jammed (AAVoid v )) -> absurd v
bornEv :: KingId -> Ev bornEv :: KingId -> Ev
bornEv inst = bornEv inst = EvBlip $ BlipEvNewt $ NewtEvBorn (fromIntegral inst, ()) ()
EvBlip $ BlipEvNewt $ NewtEvBorn (fromIntegral inst, ()) ()
hearEv :: PortNumber -> HostAddress -> ByteString -> Ev hearEv :: PortNumber -> HostAddress -> ByteString -> Ev
hearEv p a bs = hearEv p a bs =
EvBlip $ BlipEvAmes $ AmesEvHear () dest (MkBytes bs) EvBlip $ BlipEvAmes $ AmesEvHear () dest (MkBytes bs)
where where
dest = EachNo $ Jammed $ AAIpv4 (Ipv4 a) (fromIntegral p) dest = EachNo $ Jammed $ AAIpv4 (Ipv4 a) (fromIntegral p)
_turfText :: Turf -> Text
_turfText = intercalate "." . reverse . fmap unCord . unTurf
renderGalaxy :: Galaxy -> Text
renderGalaxy = Ob.renderPatp . Ob.patp . fromIntegral . unPatp
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
netMode :: HasNetworkConfig e => Bool -> RIO e NetworkMode
netMode isFake = do
netMode <- view (networkConfigL . ncNetMode)
noAmes <- view (networkConfigL . ncNoAmes)
pure $ case (noAmes, isFake, netMode) of
(True, _ , _ ) -> NoNetwork
(_ , _ , NMNone ) -> NoNetwork
(_ , True, _ ) -> Fake
(_ , _ , NMNormal ) -> Real
(_ , _ , NMLocalhost) -> Localhost
udpPort :: HasNetworkConfig e => Bool -> Ship -> RIO e PortNumber
udpPort isFake who = do
mode <- netMode isFake
mPort <- view (networkConfigL . ncAmesPort)
pure $ maybe (listenPort mode who) fromIntegral mPort
udpServ :: (HasLogFunc e, HasNetworkConfig e) => Bool -> Ship -> RIO e UdpServ
udpServ isFake who = do
mode <- netMode isFake
port <- udpPort isFake who
case modeAddress mode of
Nothing -> fakeUdpServ
Just host -> realUdpServ port host
_bornFailed :: e -> WorkError -> IO ()
_bornFailed env _ = runRIO env $ do
pure () -- TODO What can we do?
ames'
:: HasPierEnv e
=> Ship
-> Bool
-> (Text -> RIO e ())
-> RIO e ([Ev], RAcquire e (DriverApi NewtEf))
ames' who isFake stderr = do
-- Unfortunately, we cannot use TBQueue because the only behavior
-- provided for when full is to block the writer. The implementation
-- below uses materially the same data structures as TBQueue, however.
ventQ :: TQueue EvErr <- newTQueueIO
avail :: TVar Word <- newTVarIO queueBound
let
enqueuePacket p = do
vail <- readTVar avail
if vail > 0
then do
modifyTVar avail (subtract 1)
writeTQueue ventQ p
pure Intake
else do
_ <- readTQueue ventQ
writeTQueue ventQ p
pure Ouster
dequeuePacket = do
pM <- tryReadTQueue ventQ
when (isJust pM) $ modifyTVar avail (+ 1)
pure pM
env <- ask
let (bornEvs, startDriver) = ames env who isFake enqueuePacket stderr
let runDriver = do
diOnEffect <- startDriver
let diEventSource = fmap RRWork <$> dequeuePacket
pure (DriverApi {..})
pure (bornEvs, runDriver)
{-| {-|
inst -- Process instance number. inst -- Process instance number.
who -- Which ship are we? who -- Which ship are we?
@ -93,229 +168,81 @@ renderGalaxy = Ob.renderPatp . Ob.patp . fromIntegral . unPatp
TODO verify that the KingIds match on effects. TODO verify that the KingIds match on effects.
-} -}
ames :: forall e. (HasLogFunc e, HasNetworkConfig e) ames
=> KingId -> Ship -> Bool -> QueueEv :: forall e
-> (Text -> RIO e ()) . (HasLogFunc e, HasNetworkConfig e, HasKingId e)
-> ([Ev], RAcquire e (EffCb e NewtEf)) => e
ames inst who isFake enqueueEv stderr = -> Ship
(initialEvents, runAmes) -> Bool
where -> (EvErr -> STM PacketOutcome)
initialEvents :: [Ev] -> (Text -> RIO e ())
initialEvents = [bornEv inst] -> ([Ev], RAcquire e (NewtEf -> IO ()))
ames env who isFake enqueueEv stderr = (initialEvents, runAmes)
where
king = fromIntegral (env ^. kingIdL)
runAmes :: RAcquire e (EffCb e NewtEf) initialEvents :: [Ev]
runAmes = do initialEvents = [bornEv king]
drv <- mkRAcquire start stop
pure (handleEffect drv)
start :: RIO e AmesDrv runAmes :: RAcquire e (NewtEf -> IO ())
start = do runAmes = do
aTurfs <- newTVarIO Nothing mode <- rio (netMode isFake)
aGalaxies <- newIORef mempty drv <- mkRAcquire start stop
aSocket <- newTVarIO Nothing pure (handleEffect drv mode)
bindSock aSocket
aListener <- async (waitPacket aSocket)
aSendingQueue <- newTQueueIO
aSendingThread <- async (sendingThread aSendingQueue aSocket)
pure $ AmesDrv{..}
netMode :: RIO e NetworkMode start :: HasLogFunc e => RIO e AmesDrv
netMode = do start = do
if isFake aTurfs <- newTVarIO Nothing
then pure Fake aDropped <- newTVarIO 0
else view (networkConfigL . ncNetMode) >>= \case aUdpServ <- udpServ isFake who
NMNormal -> pure Real aRecvTid <- queuePacketsThread aDropped aUdpServ
NMLocalhost -> pure Localhost aResolvr <- resolvServ aTurfs (usSend aUdpServ) stderr
NMNone -> pure NoNetwork pure (AmesDrv { .. })
stop :: AmesDrv -> RIO e () hearFailed _ = pure ()
stop AmesDrv{..} = do
readIORef aGalaxies >>= mapM_ (cancel . fst)
cancel aSendingThread queuePacketsThread :: HasLogFunc e => TVar Word -> UdpServ -> RIO e (Async ())
cancel aListener queuePacketsThread dropCtr UdpServ {..} = async $ forever $ do
socket <- atomically $ readTVar aSocket outcome <- atomically $ do
io $ maybeM (pure ()) (close') (pure socket) (p, a, b) <- usRecv
enqueueEv (EvErr (hearEv p a b) hearFailed)
case outcome of
Intake -> pure ()
Ouster -> do
d <- atomically $ do
d <- readTVar dropCtr
writeTVar dropCtr (d + 1)
pure d
when (d `rem` packetsDroppedPerComplaint == 0) $
logWarn "ames: queue full; dropping inbound packets"
bindSock :: TVar (Maybe Socket) -> RIO e () stop :: AmesDrv -> RIO e ()
bindSock socketVar = getBindAddr >>= doBindSocket stop AmesDrv {..} = io $ do
where usKill aUdpServ
getBindAddr = netMode >>= \case rsKill aResolvr
Fake -> pure $ Just localhost cancel aRecvTid
Localhost -> pure $ Just localhost
Real -> pure $ Just inaddrAny
NoNetwork -> pure Nothing
doBindSocket :: Maybe HostAddress -> RIO e () handleEffect :: AmesDrv -> NetworkMode -> NewtEf -> IO ()
doBindSocket Nothing = atomically $ writeTVar socketVar Nothing handleEffect drv@AmesDrv {..} mode = runRIO env . \case
doBindSocket (Just bindAddr) = do NewtEfTurf (_id, ()) turfs -> do
mode <- netMode atomically $ writeTVar aTurfs (Just turfs)
mPort <- view (networkConfigL . ncAmesPort)
let ourPort = maybe (listenPort mode who) fromIntegral mPort
s <- io $ socket AF_INET Datagram defaultProtocol
logTrace $ displayShow ("(ames) Binding to port ", ourPort) NewtEfSend (_id, ()) dest (MkBytes bs) -> do
let addr = SockAddrInet ourPort bindAddr atomically (readTVar aTurfs) >>= \case
() <- io $ bind s addr Nothing -> pure ()
Just turfs -> sendPacket drv mode dest bs
atomically $ writeTVar socketVar (Just s) sendPacket :: AmesDrv -> NetworkMode -> AmesDest -> ByteString -> RIO e ()
sendPacket AmesDrv {..} mode dest byt = do
let to adr = io (usSend aUdpServ adr byt)
waitPacket :: TVar (Maybe Socket) -> RIO e () case (mode, dest) of
waitPacket socketVar = do (NoNetwork, _ ) -> pure ()
(atomically $ readTVar socketVar) >>= \case (Fake , _ ) -> when (okFakeAddr dest) $ to (localAddr Fake dest)
Nothing -> pure () (Localhost, _ ) -> to (localAddr Localhost dest)
Just s -> do (Real , ra) -> ra & \case
res <- io $ tryIOError $ recvFrom s 4096 EachYes gala -> io (rsSend aResolvr gala byt)
case res of EachNo addr -> to (ipv4Addr addr)
Left exn -> do
-- When we have a socket exception, we need to rebuild the
-- socket.
logTrace $ displayShow ("(ames) Socket exception. Rebinding.")
bindSock socketVar
Right (bs, addr) -> do
logTrace $ displayShow ("(ames) Received packet from ", addr)
case addr of
SockAddrInet p a -> atomically (enqueueEv $ hearEv p a bs)
_ -> pure ()
waitPacket socketVar ipv4Addr (Jammed (AAVoid v )) = absurd v
ipv4Addr (Jammed (AAIpv4 a p)) = SockAddrInet (fromIntegral p) (unIpv4 a)
handleEffect :: AmesDrv -> NewtEf -> RIO e ()
handleEffect drv@AmesDrv{..} = \case
NewtEfTurf (_id, ()) turfs -> do
atomically $ writeTVar aTurfs (Just turfs)
NewtEfSend (_id, ()) dest (MkBytes bs) -> do
atomically (readTVar aTurfs) >>= \case
Nothing -> pure ()
Just turfs -> do
mode <- netMode
(sendPacket drv mode dest bs)
sendPacket :: AmesDrv -> NetworkMode -> AmesDest -> ByteString -> RIO e ()
sendPacket AmesDrv{..} NoNetwork dest bs = pure ()
sendPacket AmesDrv{..} Fake dest bs = do
when (okayFakeAddr dest) $ atomically $
writeTQueue aSendingQueue ((localhostSockAddr Fake dest), bs)
-- In localhost only mode, regardless of the actual destination, send it to
-- localhost.
sendPacket AmesDrv{..} Localhost dest bs = atomically $
writeTQueue aSendingQueue ((localhostSockAddr Localhost dest), bs)
sendPacket AmesDrv{..} Real (EachYes galaxy) bs = do
galaxies <- readIORef aGalaxies
queue <- case M.lookup galaxy galaxies of
Just (_, queue) -> pure queue
Nothing -> do
inQueue <- newTQueueIO
thread <- async $ galaxyResolver galaxy aTurfs inQueue aSendingQueue
modifyIORef (aGalaxies) (M.insert galaxy (thread, inQueue))
pure inQueue
atomically $ writeTQueue queue bs
sendPacket AmesDrv{..} Real (EachNo (Jammed (AAIpv4 a p))) bs = do
let addr = SockAddrInet (fromIntegral p) (unIpv4 a)
atomically $ writeTQueue aSendingQueue (addr, bs)
sendPacket AmesDrv{..} Real (EachNo (Jammed (AAVoid v))) bs = do
pure (absurd v)
-- An outbound queue of messages. We can only write to a socket from one
-- thread, so coalesce those writes here.
sendingThread :: TQueue (SockAddr, ByteString)
-> TVar (Maybe Socket)
-> RIO e ()
sendingThread queue socketVar = forever $
do
(dest, bs) <- atomically $ readTQueue queue
logTrace $ displayShow ("(ames) Sending packet to ", dest)
sendAll bs dest
where
sendAll bs dest = do
mybSocket <- atomically $ readTVar socketVar
case mybSocket of
Nothing -> pure ()
Just socket -> do
bytesSent <- io $ sendTo socket bs dest
when (bytesSent /= BS.length bs) $ do
sendAll (drop bytesSent bs) dest
-- Asynchronous thread per galaxy which handles domain resolution, and can
-- block its own queue of ByteStrings to send.
--
-- Maybe perform the resolution asynchronously, injecting into the resolver
-- queue as a message.
--
-- TODO: Figure out how the real haskell time library works.
galaxyResolver :: Galaxy -> TVar (Maybe [Turf]) -> TQueue ByteString
-> TQueue (SockAddr, ByteString)
-> RIO e ()
galaxyResolver galaxy turfVar incoming outgoing =
loop Nothing Time.unixEpoch
where
loop :: Maybe SockAddr -> Time.Wen -> RIO e ()
loop lastGalaxyIP lastLookupTime = do
packet <- atomically $ readTQueue incoming
checkIP lastGalaxyIP lastLookupTime >>= \case
(Nothing, t) -> do
-- We've failed to lookup the IP. Drop the outbound packet
-- because we have no IP for our galaxy, including possible
-- previous IPs.
logDebug $ displayShow
("(ames) Dropping packet; no ip for galaxy ", galaxy)
loop Nothing t
(Just ip, t) -> do
queueSendToGalaxy ip packet
loop (Just ip) t
checkIP :: Maybe SockAddr -> Time.Wen
-> RIO e (Maybe SockAddr, Time.Wen)
checkIP lastIP lastLookupTime = do
current <- io $ Time.now
if (Time.gap current lastLookupTime ^. Time.secs) < 300
then pure (lastIP, lastLookupTime)
else do
toCheck <- fromMaybe [] <$> atomically (readTVar turfVar)
mybIp <- resolveFirstIP lastIP toCheck
timeAfterResolution <- io $ Time.now
pure (mybIp, timeAfterResolution)
resolveFirstIP :: Maybe SockAddr -> [Turf] -> RIO e (Maybe SockAddr)
resolveFirstIP prevIP [] = do
stderr $ "ames: czar at " ++ renderGalaxy galaxy ++ ": not found"
logDebug $ displayShow
("(ames) Failed to lookup IP for ", galaxy)
pure prevIP
resolveFirstIP prevIP (x:xs) = do
hostname <- buildDNS galaxy x
let portstr = show $ galaxyPort Real galaxy
listIPs <- io $ getAddrInfo Nothing (Just hostname) (Just portstr)
case listIPs of
[] -> resolveFirstIP prevIP xs
(y:ys) -> do
let sockaddr = Just $ addrAddress y
when (sockaddr /= prevIP) $
stderr $ "ames: czar " ++ renderGalaxy galaxy ++ ": ip " ++
(tshow $ addrAddress y)
logDebug $ displayShow
("(ames) Looked up ", hostname, portstr, y)
pure sockaddr
buildDNS :: Galaxy -> Turf -> RIO e String
buildDNS (Patp g) turf = do
let nameWithSig = Ob.renderPatp $ Ob.patp $ fromIntegral g
name <- case stripPrefix "~" nameWithSig of
Nothing -> error "Urbit.ob didn't produce string with ~"
Just x -> pure (unpack x)
pure $ name ++ "." ++ (unpack $ _turfText turf)
queueSendToGalaxy :: SockAddr -> ByteString -> RIO e ()
queueSendToGalaxy inet packet = do
atomically $ writeTQueue outgoing (inet, packet)

View File

@ -0,0 +1,217 @@
{-|
Handles sending packets to galaxies. We need to get their IP addresses
from DNS, which is more complicated.
-- Asynchronous thread per galaxy which handles domain resolution, and can
-- block its own queue of ByteStrings to send.
--
-- Maybe perform the resolution asynchronously, injecting into the resolver
-- queue as a message.
--
-- TODO: Figure out how the real haskell time library works.
-- We've failed to lookup the IP. Drop the outbound packet
-- because we have no IP for our galaxy, including possible
-- previous IPs.
{-
- Sending Packets to Galaxies.
- Each galaxy has it's own DNS resolution thread.
- Initially, no threads are started.
- To send a message to a galaxy,
- Check to see if it already has a resolution thread.
- If it does, pass the packet to that thread.
- If it doesn't, start a new thread and give it the packet.
- Galaxy resolution threads work as follows:
- First, they are given:
- They know which galaxy they are responsible for.
- They have access to the turfs TVar (shared state with Ames driver).
- They can be given packets (to be send to their galaxy).
- They must be given a way to send UDP packets.
- Next, we loop forever
- In the loop we track:
- the last-known IP address.
- the time when we last looked up the IP address.
- We wait to be given a packet.
- We get the IP address.
- If we looked up the IP address in the last 5 minute, use the
cached IP address.
- Just use the one from last time.
- Otherwise,
- Do a DNS lookup.
- Go through the turf list one item at a time.
- Try each one.
- If it resolves to one-or-more IP addresses,
- Use the first one.
- If it resolves to zero IP addresses, move on to the next turf.
- If none of the turfs can be used to resolve the IP address,
then we don't know where the galaxy is.
- Drop the packet.
-}
-}
module Urbit.Vere.Ames.DNS
( NetworkMode(..)
, ResolvServ(..)
, resolvServ
, galaxyPort
, renderGalaxy
)
where
import Urbit.Prelude
import Network.Socket hiding (recvFrom, sendTo)
import Urbit.Arvo hiding (Fake)
import qualified Data.Map as M
import qualified Urbit.Noun.Time as Time
import qualified Urbit.Ob as Ob
-- Types -----------------------------------------------------------------------
data NetworkMode = Fake | Localhost | Real | NoNetwork
deriving (Eq, Ord, Show)
data ResolvServ = ResolvServ
{ rsSend :: Galaxy -> ByteString -> IO ()
, rsKill :: IO ()
}
-- Utils -----------------------------------------------------------------------
galaxyPort :: NetworkMode -> Galaxy -> PortNumber
galaxyPort Fake (Patp g) = fromIntegral g + 31337
galaxyPort Localhost (Patp g) = fromIntegral g + 13337
galaxyPort Real (Patp g) = fromIntegral g + 13337
galaxyPort NoNetwork _ = fromIntegral 0
turfText :: Turf -> Text
turfText = intercalate "." . reverse . fmap unCord . unTurf
renderGalaxy :: Galaxy -> Text
renderGalaxy = Ob.renderPatp . Ob.patp . fromIntegral . unPatp
galaxyHostname :: Galaxy -> Turf -> Text
galaxyHostname g t = galaName g ++ "." ++ turfText t
where
stripSig :: Text -> Text
stripSig inp = fromMaybe inp (stripPrefix "~" inp)
galaName :: Galaxy -> Text
galaName = stripSig . renderGalaxy
resolv :: Galaxy -> [Turf] -> IO (Maybe (Turf, Text, PortNumber, SockAddr))
resolv gal = go
where
go = \case
[] -> pure Nothing
turf : turfs -> do
let host = galaxyHostname gal turf
port = galaxyPort Real gal
getAddrInfo Nothing (Just (unpack host)) (Just (show port)) >>= \case
[] -> go turfs
ip : _ -> pure $ Just (turf, host, port, addrAddress ip)
doResolv
:: HasLogFunc e
=> Galaxy
-> (Time.Wen, Maybe SockAddr)
-> [Turf]
-> (Text -> RIO e ())
-> RIO e (Maybe SockAddr, Time.Wen)
doResolv gal (prevWen, prevIP) turfs stderr = do
current <- io $ Time.now
if (Time.gap current prevWen ^. Time.secs) < 300
then pure (prevIP, prevWen)
else do
tim <- io (Time.now)
io (resolv gal turfs) >>= \case
Nothing -> do
stderr $ "ames: czar at " ++ galStr ++ ": not found"
logDebug $ displayShow ("(ames) Failed to lookup IP for ", gal)
pure (prevIP, tim)
Just (turf, host, port, addr) -> do
when (Just addr /= prevIP) (printCzar addr)
logDebug $ displayShow ("(ames) Looked up ", host, port, turf, addr)
pure (Just addr, tim)
where
galStr = renderGalaxy gal
printCzar addr = stderr $ "ames: czar " ++ galStr ++ ": ip " ++ tshow addr
resolvWorker
:: forall e
. HasLogFunc e
=> Galaxy
-> TVar (Maybe [Turf])
-> TVar (Time.Wen, Maybe SockAddr)
-> STM ByteString
-> (SockAddr -> ByteString -> IO ())
-> (Text -> RIO e ())
-> RIO e (Async ())
resolvWorker gal vTurfs vLast waitMsg send stderr = async (forever go)
where
logDrop =
logDebug $ displayShow ("(ames) Dropping packet; no ip for galaxy ", gal)
go :: RIO e ()
go = do
(packt, turfs, (lastTime, lastAddr)) <- atomically
((,,) <$> waitMsg <*> readTVar vTurfs <*> readTVar vLast)
(newAddr, newTime) <- doResolv gal
(lastTime, lastAddr)
(fromMaybe [] turfs)
stderr
maybe logDrop (\ip -> io (send ip packt)) newAddr
atomically $ writeTVar vLast (newTime, newAddr)
resolvServ
:: HasLogFunc e
=> TVar (Maybe [Turf])
-> (SockAddr -> ByteString -> IO ())
-> (Text -> RIO e ())
-> RIO e ResolvServ
resolvServ vTurfs send stderr = do
vGala <- newTVarIO (mempty :: Map Galaxy (Async (), TQueue ByteString))
vDead <- newTVarIO False
envir <- ask
let spawnWorker :: Galaxy -> IO (Async (), TQueue ByteString)
spawnWorker gal = runRIO envir $ do
que <- newTQueueIO
las <- newTVarIO (Time.unixEpoch, Nothing)
tid <- resolvWorker gal vTurfs las (readTQueue que) send stderr
pure (tid, que)
let getWorker :: Galaxy -> IO (Async (), TQueue ByteString)
getWorker gal = do
(fmap (lookup gal) $ atomically $ readTVar vGala) >>= \case
Just (tid, que) -> do
pure (tid, que)
Nothing -> do
(tid, que) <- spawnWorker gal
atomically $ modifyTVar' vGala (M.insert gal (tid, que))
pure (tid, que)
let doSend :: Galaxy -> ByteString -> IO ()
doSend gal byt = do
dead <- atomically (readTVar vDead)
unless dead $ do
(_, que) <- getWorker gal
atomically (writeTQueue que byt)
let doKill :: IO ()
doKill = do
galas <- atomically $ do
writeTVar vDead True
readTVar vGala
for_ galas (cancel . fst)
pure (ResolvServ doSend doKill)

View File

@ -0,0 +1,243 @@
{- |
Raw UDP Server used by Ames driver.
1. Opens a UDP socket and makes sure that it stays open.
- If can't open the port, wait and try again repeatedly.
- If there is an error reading or writting from the open socket,
close it and open another.
2. Receives packets from the socket.
- When packets come in from the socket, they go into a bounded queue.
- If the queue is full, the packet is dropped.
- If the socket is closed, wait and try again repeatedly.
- `usRecv` gets the first packet from the queue.
3. Sends packets to the socket.
- Packets sent to `usSend` enter a bounded queue.
- If that queue is full, the packet is dropped.
- Packets are taken off the queue one at a time.
- If the socket is closed (or broken), the packet is dropped.
4. Runs until `usKill` is run, then all threads are killed and the
socket is closed.
-}
module Urbit.Vere.Ames.UDP
( UdpServ(..)
, fakeUdpServ
, realUdpServ
)
where
import Urbit.Prelude
import Network.Socket hiding (recvFrom, sendTo)
import Control.Monad.STM (retry)
import Network.Socket.ByteString (recvFrom, sendTo)
-- Types -----------------------------------------------------------------------
data UdpServ = UdpServ
{ usSend :: SockAddr -> ByteString -> IO ()
, usRecv :: STM (PortNumber, HostAddress, ByteString)
, usKill :: IO ()
}
-- Utils -----------------------------------------------------------------------
{- |
Writes to queue and returns `True` unless the queue is full, then do
nothing and return `False`.
-}
tryWriteTBQueue :: TBQueue x -> x -> STM Bool
tryWriteTBQueue q x = do
isFullTBQueue q >>= \case
True -> pure False
False -> writeTBQueue q x $> True
{- |
Open a UDP socket and bind it to a port
-}
doBind :: PortNumber -> HostAddress -> IO (Either IOError Socket)
doBind por hos = tryIOError $ do
sok <- io $ socket AF_INET Datagram defaultProtocol
() <- io $ bind sok (SockAddrInet por hos)
pure sok
{- |
Open a UDP socket and bind it to a port.
If this fails, wait 250ms and repeat forever.
-}
forceBind :: HasLogFunc e => PortNumber -> HostAddress -> RIO e Socket
forceBind por hos = go
where
go = do
logDebug (display ("AMES: UDP: Opening socket on port " <> tshow por))
io (doBind por hos) >>= \case
Right sk -> do
logDebug (display ("AMES: UDP: Opened socket on port " <> tshow por))
pure sk
Left err -> do
logDebug (display ("AMES: UDP: " <> tshow err))
logDebug ("AMES: UDP: Failed to open UDP socket. Waiting")
threadDelay 250_000
go
{- |
Attempt to send a packet to a socket.
If it fails, return `False`. Otherwise, return `True`.
-}
sendPacket :: HasLogFunc e => ByteString -> SockAddr -> Socket -> RIO e Bool
sendPacket fullBytes adr sok = do
logDebug $ displayShow ("AMES", "UDP", "Sending packet.")
res <- io $ tryIOError $ go fullBytes
case res of
Left err -> do
logError $ displayShow ("AMES", "UDP", "Failed to send packet", err)
pure False
Right () -> do
logDebug $ displayShow ("AMES", "UDP", "Packet sent.")
pure True
where
go byt = do
sent <- sendTo sok byt adr
when (sent /= length byt) $ do
go (drop sent byt)
{- |
Attempt to receive a packet from a socket.
- If an exception is throw, return `Left exn`.
- If it wasn't an IPv4 packet, return `Right Nothing`.
- Otherwise, return `Right (Just packet)`.
-}
recvPacket
:: HasLogFunc e
=> Socket
-> RIO e (Either IOError (Maybe (ByteString, PortNumber, HostAddress)))
recvPacket sok = do
io (tryIOError $ recvFrom sok 4096) <&> \case
Left exn -> Left exn
Right (b, SockAddrInet p a) -> Right (Just (b, p, a))
Right (_, _ ) -> Right Nothing
-- Fake Server for No-Networking Mode ------------------------------------------
{- |
Fake UDP API for no-networking configurations.
-}
fakeUdpServ :: HasLogFunc e => RIO e UdpServ
fakeUdpServ = do
logDebug $ displayShow ("AMES", "UDP", "\"Starting\" fake UDP server.")
pure UdpServ { .. }
where
usSend = \_ _ -> pure ()
usRecv = retry
usKill = pure ()
-- Real Server -----------------------------------------------------------------
{- |
Real UDP server. See module-level docs.
-}
realUdpServ
:: forall e . HasLogFunc e => PortNumber -> HostAddress -> RIO e UdpServ
realUdpServ por hos = do
logDebug $ displayShow ("AMES", "UDP", "Starting real UDP server.")
env <- ask
vSock <- newTVarIO Nothing
vFail <- newEmptyTMVarIO
qSend <- newTBQueueIO 100 -- TODO Tuning
qRecv <- newTBQueueIO 100 -- TODO Tuning
{-
If reading or writing to a socket fails, unbind it and tell the
socket-open thread to close it and open another.
This is careful about edge-cases. In any of these cases, do nothing.
- If vSock isn't set to the socket we used, do nothing.
- If vFail is already set (another thread signaled failure already).
-}
let signalBrokenSocket :: Socket -> RIO e ()
signalBrokenSocket sock = do
logDebug $ displayShow ("AMES", "UDP"
, "Socket broken. Requesting new socket"
)
atomically $ do
mSock <- readTVar vSock
mFail <- tryReadTMVar vFail
when (mSock == Just sock && mFail == Nothing) $ do
putTMVar vFail sock
writeTVar vSock Nothing
enqueueRecvPacket :: PortNumber -> HostAddress -> ByteString -> RIO e ()
enqueueRecvPacket p a b = do
did <- atomically (tryWriteTBQueue qRecv (p, a, b))
when (did == False) $ do
logWarn $ displayShow $ ("AMES", "UDP",)
"Dropping inbound packet because queue is full."
enqueueSendPacket :: SockAddr -> ByteString -> RIO e ()
enqueueSendPacket a b = do
did <- atomically (tryWriteTBQueue qSend (a, b))
when (did == False) $ do
logWarn "AMES: UDP: Dropping outbound packet because queue is full."
tOpen <- async $ forever $ do
sk <- forceBind por hos
atomically (writeTVar vSock (Just sk))
broken <- atomically (takeTMVar vFail)
logWarn "AMES: UDP: Closing broken socket."
io (close broken)
tSend <- async $ forever $ join $ atomically $ do
(adr, byt) <- readTBQueue qSend
readTVar vSock <&> \case
Nothing -> pure ()
Just sk -> do
okay <- sendPacket byt adr sk
unless okay (signalBrokenSocket sk)
tRecv <- async $ forever $ do
atomically (readTVar vSock) >>= \case
Nothing -> threadDelay 100_000
Just sk -> do
recvPacket sk >>= \case
Left exn -> do
logError "AMES: UDP: Failed to receive packet"
signalBrokenSocket sk
Right Nothing -> do
logError "AMES: UDP: Dropping non-ipv4 packet"
pure ()
Right (Just (b, p, a)) -> do
logDebug "AMES: UDP: Received packet."
enqueueRecvPacket p a b
let shutdown = do
logDebug "AMES: UDP: Shutting down. (killing threads)"
cancel tOpen
cancel tSend
cancel tRecv
logDebug "AMES: UDP: Shutting down. (closing socket)"
io $ join $ atomically $ do
res <- readTVar vSock <&> maybe (pure ()) close
writeTVar vSock Nothing
pure res
pure $ UdpServ { usSend = \a b -> runRIO env (enqueueSendPacket a b)
, usRecv = readTBQueue qRecv
, usKill = runRIO env shutdown
}

View File

@ -2,21 +2,33 @@
Behn: Timer Driver Behn: Timer Driver
-} -}
module Urbit.Vere.Behn (behn) where module Urbit.Vere.Behn (behn, DriverApi(..), behn') where
import Urbit.Arvo hiding (Behn) import Urbit.Arvo hiding (Behn)
import Urbit.Prelude import Urbit.Prelude
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Urbit.Time (Wen) import Urbit.King.App (HasKingId(..), HasPierEnv(..))
import Urbit.Timer (Timer) import Urbit.Noun.Time (Wen)
import Urbit.Timer (Timer)
import qualified Urbit.Time as Time import qualified Urbit.Noun.Time as Time
import qualified Urbit.Timer as Timer import qualified Urbit.Timer as Timer
-- Behn Stuff ------------------------------------------------------------------ -- Behn Stuff ------------------------------------------------------------------
behn' :: HasPierEnv e => RIO e ([Ev], RAcquire e (DriverApi BehnEf))
behn' = do
env <- ask
pure ([bornEv (fromIntegral (env ^. kingIdL))], runDriver env)
where
runDriver env = do
ventQ :: TQueue EvErr <- newTQueueIO
diOnEffect <- liftAcquire (behn env (writeTQueue ventQ))
let diEventSource = fmap RRWork <$> tryReadTQueue ventQ
pure (DriverApi {..})
bornEv :: KingId -> Ev bornEv :: KingId -> Ev
bornEv king = EvBlip $ BlipEvBehn $ BehnEvBorn (king, ()) () bornEv king = EvBlip $ BlipEvBehn $ BehnEvBorn (king, ()) ()
@ -25,16 +37,22 @@ wakeEv = EvBlip $ BlipEvBehn $ BehnEvWake () ()
sysTime = view Time.systemTime sysTime = view Time.systemTime
behn :: KingId -> QueueEv -> ([Ev], Acquire (EffCb e BehnEf)) wakeErr :: WorkError -> IO ()
behn king enqueueEv = wakeErr _ = pure ()
(initialEvents, runBehn)
where
initialEvents = [bornEv king]
runBehn :: Acquire (EffCb e BehnEf) behn
:: HasKingId e
=> e
-> (EvErr -> STM ())
-> Acquire (BehnEf -> IO ())
behn env enqueueEv = runBehn
where
king = fromIntegral (env ^. kingIdL)
runBehn :: Acquire (BehnEf -> IO ())
runBehn = do runBehn = do
tim <- mkAcquire Timer.init Timer.stop tim <- mkAcquire Timer.init Timer.stop
pure (handleEf tim) pure (runRIO env . handleEf tim)
handleEf :: Timer -> BehnEf -> RIO e () handleEf :: Timer -> BehnEf -> RIO e ()
handleEf b = io . \case handleEf b = io . \case
@ -45,4 +63,4 @@ behn king enqueueEv =
doze :: Timer -> Maybe Wen -> IO () doze :: Timer -> Maybe Wen -> IO ()
doze tim = \case doze tim = \case
Nothing -> Timer.stop tim Nothing -> Timer.stop tim
Just t -> Timer.start tim (sysTime t) $ atomically (enqueueEv wakeEv) Just t -> Timer.start tim (sysTime t) $ atomically (enqueueEv (EvErr wakeEv wakeErr))

View File

@ -2,10 +2,14 @@
UNIX Filesystem Driver UNIX Filesystem Driver
-} -}
module Urbit.Vere.Clay (clay) where module Urbit.Vere.Clay
( clay
, clay'
)
where
import Urbit.Arvo hiding (Term) import Urbit.Arvo hiding (Term)
import Urbit.King.Config import Urbit.King.App
import Urbit.Prelude import Urbit.Prelude
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
@ -112,26 +116,52 @@ buildActionListFromDifferences fp snapshot = do
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
clay :: forall e. (HasPierConfig e, HasLogFunc e) _boatFailed :: e -> WorkError -> IO ()
=> KingId -> QueueEv -> ([Ev], RAcquire e (EffCb e SyncEf)) _boatFailed env _ = runRIO env $ do
clay king enqueueEv = pure () -- TODO What can we do?
clay'
:: HasPierEnv e
=> RIO e ([Ev], RAcquire e (DriverApi SyncEf))
clay' = do
ventQ :: TQueue EvErr <- newTQueueIO
env <- ask
let (bornEvs, startDriver) = clay env (writeTQueue ventQ)
let runDriver = do
diOnEffect <- startDriver
let diEventSource = fmap RRWork <$> tryReadTQueue ventQ
pure (DriverApi {..})
pure (bornEvs, runDriver)
clay
:: forall e
. (HasPierConfig e, HasLogFunc e, HasKingId e)
=> e
-> (EvErr -> STM ())
-> ([Ev], RAcquire e (SyncEf -> IO ()))
clay env plan =
(initialEvents, runSync) (initialEvents, runSync)
where where
initialEvents = [ king = fromIntegral (env ^. kingIdL)
EvBlip $ BlipEvBoat $ BoatEvBoat () ()
-- TODO: In the case of -A, we need to read all the data from the
-- specified directory and shove it into an %into event.
]
runSync :: RAcquire e (EffCb e SyncEf) boatEv = EvBlip $ BlipEvBoat $ BoatEvBoat () ()
-- TODO: In the case of -A, we need to read all the data from the
-- specified directory and shove it into an %into event.
initialEvents = [boatEv]
runSync :: RAcquire e (SyncEf -> IO ())
runSync = handleEffect <$> mkRAcquire start stop runSync = handleEffect <$> mkRAcquire start stop
start :: RIO e ClayDrv start :: RIO e ClayDrv
start = ClayDrv <$> newTVarIO mempty start = ClayDrv <$> newTVarIO mempty
stop c = pure () stop c = pure ()
handleEffect :: ClayDrv -> SyncEf -> RIO e () handleEffect :: ClayDrv -> SyncEf -> IO ()
handleEffect cd = \case handleEffect cd = runRIO env . \case
SyncEfHill _ mountPoints -> do SyncEfHill _ mountPoints -> do
logDebug $ displayShow ("(clay) known mount points:", mountPoints) logDebug $ displayShow ("(clay) known mount points:", mountPoints)
pierPath <- view pierPathL pierPath <- view pierPathL
@ -151,8 +181,15 @@ clay king enqueueEv =
logDebug $ displayShow ("(clay) dirk actions: ", actions) logDebug $ displayShow ("(clay) dirk actions: ", actions)
let !intoList = map (actionsToInto dir) actions let !intoList = map (actionsToInto dir) actions
atomically $ enqueueEv $ EvBlip $ BlipEvSync $
SyncEvInto (Some (king, ())) desk False intoList let syncEv = EvBlip
$ BlipEvSync
$ SyncEvInto (Some (king, ())) desk False intoList
let syncFailed _ = pure ()
atomically $ plan (EvErr syncEv syncFailed)
atomically $ modifyTVar atomically $ modifyTVar
(cdMountPoints cd) (cdMountPoints cd)

View File

@ -0,0 +1,364 @@
{-|
Eyre: Http Server Driver
-}
module Urbit.Vere.Eyre
( eyre
, eyre'
)
where
import Urbit.Prelude hiding (Builder)
import Urbit.Arvo hiding (ServerId, reqUrl, secure)
import Urbit.King.App (HasKingId(..), HasPierEnv(..))
import Urbit.King.Config
import Urbit.Vere.Eyre.Multi
import Urbit.Vere.Eyre.PortsFile
import Urbit.Vere.Eyre.Serv
import Urbit.Vere.Eyre.Service
import Urbit.Vere.Eyre.Wai
import Urbit.Vere.Pier.Types
import Data.List.NonEmpty (NonEmpty((:|)))
import Data.PEM (pemParseBS, pemWriteBS)
import RIO.Prelude (decodeUtf8Lenient)
import System.Random (randomIO)
import Urbit.Vere.Http (convertHeaders, unconvertHeaders)
import qualified Network.HTTP.Types as H
-- Types -----------------------------------------------------------------------
type HasShipEnv e = (HasLogFunc e, HasNetworkConfig e, HasPierConfig e)
type ReqId = UD
newtype Drv = Drv (MVar (Maybe Serv))
data SockOpts = SockOpts
{ soLocalhost :: Bool
, soWhich :: ServPort
}
data PortsToTry = PortsToTry
{ pttSec :: SockOpts
, pttIns :: SockOpts
, pttLop :: SockOpts
}
data Serv = Serv
{ sServId :: ServId
, sConfig :: HttpServerConf
, sLop :: ServApi
, sIns :: ServApi
, sSec :: Maybe ServApi
, sPorts :: Ports
, sPortsFile :: FilePath
, sLiveReqs :: TVar LiveReqs
}
-- Utilities for Constructing Events -------------------------------------------
servEv :: HttpServerEv -> Ev
servEv = EvBlip . BlipEvHttpServer
bornEv :: KingId -> Ev
bornEv king = servEv $ HttpServerEvBorn (king, ()) ()
liveEv :: ServId -> Ports -> Ev
liveEv sId Ports {..} = servEv $ HttpServerEvLive (sId, ()) pHttp pHttps
cancelEv :: ServId -> ReqId -> EvErr
cancelEv sId reqId =
EvErr (servEv (HttpServerEvCancelRequest (sId, reqId, 1, ()) ())) cancelFailed
cancelFailed :: WorkError -> IO ()
cancelFailed _ = pure ()
reqEv :: ServId -> ReqId -> WhichServer -> Address -> HttpRequest -> Ev
reqEv sId reqId which addr req = case which of
Loopback -> servEv $ HttpServerEvRequestLocal (sId, reqId, 1, ())
$ HttpServerReq False addr req
_ -> servEv $ HttpServerEvRequest (sId, reqId, 1, ())
$ HttpServerReq (which == Secure) addr req
-- Based on Pier+Config, which ports should each server run? -------------------
httpServerPorts :: HasShipEnv e => Bool -> RIO e PortsToTry
httpServerPorts fak = do
ins <- view (networkConfigL . ncHttpPort . to (fmap fromIntegral))
sec <- view (networkConfigL . ncHttpsPort . to (fmap fromIntegral))
lop <- view (networkConfigL . ncLocalPort . to (fmap fromIntegral))
localMode <- view (networkConfigL . ncNetMode . to (== NMLocalhost))
let local = localMode || fak
let pttSec = case (sec, fak) of
(Just p , _ ) -> SockOpts local (SPChoices $ singleton p)
(Nothing, False) -> SockOpts local (SPChoices (443 :| [8443 .. 8453]))
(Nothing, True ) -> SockOpts local (SPChoices (8443 :| [8444 .. 8453]))
let pttIns = case (ins, fak) of
(Just p , _ ) -> SockOpts local (SPChoices $ singleton p)
(Nothing, False) -> SockOpts local (SPChoices (80 :| [8080 .. 8090]))
(Nothing, True ) -> SockOpts local (SPChoices (8080 :| [8081 .. 8090]))
let pttLop = case (lop, fak) of
(Just p , _) -> SockOpts local (SPChoices $ singleton p)
(Nothing, _) -> SockOpts local SPAnyPort
pure (PortsToTry { .. })
-- Convert Between Urbit and WAI types. ----------------------------------------
parseTlsConfig :: (Key, Cert) -> Maybe TlsConfig
parseTlsConfig (PEM key, PEM certs) = do
let (cerByt, keyByt) = (wainBytes certs, wainBytes key)
pems <- pemParseBS cerByt & either (const Nothing) Just
(cert, chain) <- case pems of
[] -> Nothing
p : ps -> pure (pemWriteBS p, pemWriteBS <$> ps)
pure $ TlsConfig keyByt cert chain
where
wainBytes :: Wain -> ByteString
wainBytes = encodeUtf8 . unWain
parseHttpEvent :: HttpEvent -> [RespAct]
parseHttpEvent = \case
Start h b True -> [RAFull (hSta h) (hHdr h) (fByt $ fromMaybe "" b)]
Start h b False -> [RAHead (hSta h) (hHdr h) (fByt $ fromMaybe "" b)]
Cancel () -> [RADone]
Continue b done -> toList (RABloc . fByt <$> b)
<> if done then [RADone] else []
where
hHdr :: ResponseHeader -> [H.Header]
hHdr = unconvertHeaders . headers
hSta :: ResponseHeader -> H.Status
hSta = toEnum . fromIntegral . statusCode
fByt :: File -> ByteString
fByt = unOcts . unFile
requestEvent :: ServId -> WhichServer -> Word64 -> ReqInfo -> Ev
requestEvent srvId which reqId ReqInfo{..} = reqEv srvId reqUd which riAdr evReq
where
evBod = bodFile riBod
evHdr = convertHeaders riHdr
evUrl = Cord (decodeUtf8Lenient riUrl)
evReq = HttpRequest riMet evUrl evHdr evBod
reqUd = fromIntegral reqId
bodFile :: ByteString -> Maybe File
bodFile "" = Nothing
bodFile bs = Just $ File $ Octs bs
-- Running Servers -------------------------------------------------------------
execRespActs :: HasLogFunc e => Drv -> Ship -> Word64 -> HttpEvent -> RIO e ()
execRespActs (Drv v) who reqId ev = readMVar v >>= \case
Nothing -> logError "Got a response to a request that does not exist."
Just sv -> do
logDebug $ displayShow ev
for_ (parseHttpEvent ev) $ \act -> do
atomically (routeRespAct who (sLiveReqs sv) reqId act)
startServ
:: (HasPierConfig e, HasLogFunc e, HasNetworkConfig e)
=> MultiEyreApi
-> Ship
-> Bool
-> HttpServerConf
-> (EvErr -> STM ())
-> RIO e Serv
startServ multi who isFake conf plan = do
logDebug (displayShow ("EYRE", "startServ"))
let vLive = meaLive multi
srvId <- io $ ServId . UV . fromIntegral <$> (randomIO :: IO Word32)
let mTls = hscSecure conf >>= parseTlsConfig
mCre <- mTls & \case
Nothing -> pure Nothing
Just tc -> configCreds tc & \case
Right rs -> pure (Just (tc, rs))
Left err -> do
logError "Couldn't Load TLS Credentials."
pure Nothing
ptt <- httpServerPorts isFake
{-
TODO If configuration requests a redirect, get the HTTPS port (if
configuration specifies a specific port, use that. Otherwise, wait
for the HTTPS server to start and then use the port that it chose).
and run an HTTP server that simply redirects to the HTTPS server.
-}
let secRedi = Nothing
let soHost :: SockOpts -> ServHost
soHost so = if soLocalhost so then SHLocalhost else SHAnyHostOk
noHttp <- view (networkConfigL . ncNoHttp)
noHttps <- view (networkConfigL . ncNoHttps)
let reqEvFailed _ = pure ()
let onReq :: WhichServer -> Ship -> Word64 -> ReqInfo -> STM ()
onReq which _ship reqId reqInfo =
plan $ EvErr (requestEvent srvId which reqId reqInfo) reqEvFailed
let onKilReq :: Ship -> Word64 -> STM ()
onKilReq _ship = plan . cancelEv srvId . fromIntegral
logDebug (displayShow ("EYRE", "joinMultiEyre", who, mTls, mCre))
atomically (joinMultiEyre multi who mCre onReq onKilReq)
logDebug $ displayShow ("EYRE", "Starting loopback server")
lop <- serv vLive $ ServConf
{ scHost = soHost (pttLop ptt)
, scPort = soWhich (pttLop ptt)
, scRedi = Nothing
, scFake = False
, scType = STHttp who $ ReqApi
{ rcReq = onReq Loopback
, rcKil = onKilReq
}
}
logDebug $ displayShow ("EYRE", "Starting insecure server")
ins <- serv vLive $ ServConf
{ scHost = soHost (pttIns ptt)
, scPort = soWhich (pttIns ptt)
, scRedi = secRedi
, scFake = noHttp
, scType = STHttp who $ ReqApi
{ rcReq = onReq Insecure
, rcKil = onKilReq
}
}
mSec <- for mTls $ \tls -> do
logDebug "Starting secure server"
serv vLive $ ServConf
{ scHost = soHost (pttSec ptt)
, scPort = soWhich (pttSec ptt)
, scRedi = Nothing
, scFake = noHttps
, scType = STHttps who tls $ ReqApi
{ rcReq = onReq Secure
, rcKil = onKilReq
}
}
pierPath <- view pierPathL
lopPor <- atomically (fmap fromIntegral $ saPor lop)
insPor <- atomically (fmap fromIntegral $ saPor ins)
secPor <- for mSec (fmap fromIntegral . atomically . saPor)
let por = Ports secPor insPor lopPor
fil = pierPath <> "/.http.ports"
logDebug $ displayShow ("EYRE", "All Servers Started.", srvId, por, fil)
pure (Serv srvId conf lop ins mSec por fil vLive)
-- Eyre Driver -----------------------------------------------------------------
_bornFailed :: e -> WorkError -> IO ()
_bornFailed env _ = runRIO env $ do
pure () -- TODO What should this do?
eyre'
:: HasPierEnv e
=> MultiEyreApi
-> Ship
-> Bool
-> RIO e ([Ev], RAcquire e (DriverApi HttpServerEf))
eyre' multi who isFake = do
ventQ :: TQueue EvErr <- newTQueueIO
env <- ask
let (bornEvs, startDriver) = eyre env multi who (writeTQueue ventQ) isFake
let runDriver = do
diOnEffect <- startDriver
let diEventSource = fmap RRWork <$> tryReadTQueue ventQ
pure (DriverApi {..})
pure (bornEvs, runDriver)
{-|
Eyre -- HTTP Server Driver
Inject born events.
Until born events succeeds, ignore effects.
Wait until born event callbacks invoked.
If success, signal success.
If failure, try again several times.
If still failure, bring down ship.
Once born event succeeds:
- Begin normal operation (start accepting requests)
-}
eyre
:: forall e
. (HasPierEnv e)
=> e
-> MultiEyreApi
-> Ship
-> (EvErr -> STM ())
-> Bool
-> ([Ev], RAcquire e (HttpServerEf -> IO ()))
eyre env multi who plan isFake = (initialEvents, runHttpServer)
where
king = fromIntegral (env ^. kingIdL)
initialEvents :: [Ev]
initialEvents = [bornEv king]
runHttpServer :: RAcquire e (HttpServerEf -> IO ())
runHttpServer = handleEf <$> mkRAcquire
(Drv <$> newMVar Nothing)
(\(Drv v) -> stopService v kill >>= fromEither)
kill :: HasLogFunc e => Serv -> RIO e ()
kill Serv{..} = do
atomically (leaveMultiEyre multi who)
atomically (saKil sLop)
atomically (saKil sIns)
for_ sSec (\sec -> atomically (saKil sec))
io (removePortsFile sPortsFile)
restart :: Drv -> HttpServerConf -> RIO e Serv
restart (Drv var) conf = do
logDebug "Restarting http server"
let startAct = startServ multi who isFake conf plan
res <- fromEither =<< restartService var startAct kill
logDebug "Done restating http server"
pure res
liveFailed _ = pure ()
handleEf :: Drv -> HttpServerEf -> IO ()
handleEf drv = runRIO env . \case
HSESetConfig (i, ()) conf -> do
logDebug (displayShow ("EYRE", "%set-config"))
Serv {..} <- restart drv conf
logDebug (displayShow ("EYRE", "%set-config", "Sending %live"))
atomically $ plan (EvErr (liveEv sServId sPorts) liveFailed)
logDebug "Write ports file"
io (writePortsFile sPortsFile sPorts)
HSEResponse (i, req, _seq, ()) ev -> do
logDebug (displayShow ("EYRE", "%response"))
execRespActs drv who (fromIntegral req) ev

View File

@ -0,0 +1,131 @@
{-|
Eyre: Http Server Driver
-}
module Urbit.Vere.Eyre.Multi
( WhichServer(..)
, MultiEyreConf(..)
, OnMultiReq
, OnMultiKil
, MultiEyreApi(..)
, joinMultiEyre
, leaveMultiEyre
, multiEyre
)
where
import Urbit.Prelude hiding (Builder)
import Urbit.Arvo hiding (ServerId, reqUrl, secure)
import Urbit.Vere.Eyre.Serv
import Urbit.Vere.Eyre.Wai
import Network.TLS (Credential)
-- Types -----------------------------------------------------------------------
data WhichServer = Secure | Insecure | Loopback
deriving (Eq)
data MultiEyreConf = MultiEyreConf
{ mecHttpsPort :: Maybe Port
, mecHttpPort :: Maybe Port
, mecLocalhostOnly :: Bool
}
deriving (Show)
type OnMultiReq = WhichServer -> Ship -> Word64 -> ReqInfo -> STM ()
type OnMultiKil = Ship -> Word64 -> STM ()
data MultiEyreApi = MultiEyreApi
{ meaConf :: MultiEyreConf
, meaLive :: TVar LiveReqs
, meaPlan :: TVar (Map Ship OnMultiReq)
, meaCanc :: TVar (Map Ship OnMultiKil)
, meaTlsC :: TVar (Map Ship (TlsConfig, Credential))
, meaKill :: STM ()
}
-- Multi-Tenet HTTP ------------------------------------------------------------
joinMultiEyre
:: MultiEyreApi
-> Ship
-> Maybe (TlsConfig, Credential)
-> OnMultiReq
-> OnMultiKil
-> STM ()
joinMultiEyre api who mTls onReq onKil = do
modifyTVar' (meaPlan api) (insertMap who onReq)
modifyTVar' (meaCanc api) (insertMap who onKil)
for_ mTls $ \creds -> do
modifyTVar' (meaTlsC api) (insertMap who creds)
leaveMultiEyre :: MultiEyreApi -> Ship -> STM ()
leaveMultiEyre MultiEyreApi {..} who = do
modifyTVar' meaCanc (deleteMap who)
modifyTVar' meaPlan (deleteMap who)
modifyTVar' meaTlsC (deleteMap who)
multiEyre :: HasLogFunc e => MultiEyreConf -> RIO e MultiEyreApi
multiEyre conf@MultiEyreConf {..} = do
logDebug (displayShow ("EYRE", "MULTI", conf))
vLive <- io emptyLiveReqs >>= newTVarIO
vPlan <- newTVarIO mempty
vCanc <- newTVarIO (mempty :: Map Ship (Ship -> Word64 -> STM ()))
vTlsC <- newTVarIO mempty
let host = if mecLocalhostOnly then SHLocalhost else SHAnyHostOk
let onReq :: WhichServer -> Ship -> Word64 -> ReqInfo -> STM ()
onReq which who reqId reqInfo = do
plan <- readTVar vPlan
lookup who plan & \case
Nothing -> pure ()
Just cb -> cb which who reqId reqInfo
let onKil :: Ship -> Word64 -> STM ()
onKil who reqId = do
canc <- readTVar vCanc
lookup who canc & \case
Nothing -> pure ()
Just cb -> cb who reqId
mIns <- for mecHttpPort $ \por -> do
logDebug (displayShow ("EYRE", "MULTI", "HTTP", por))
serv vLive $ ServConf
{ scHost = host
, scPort = SPChoices $ singleton $ fromIntegral por
, scRedi = Nothing -- TODO
, scFake = False
, scType = STMultiHttp $ ReqApi
{ rcReq = onReq Insecure
, rcKil = onKil
}
}
mSec <- for mecHttpsPort $ \por -> do
logDebug (displayShow ("EYRE", "MULTI", "HTTPS", por))
serv vLive $ ServConf
{ scHost = host
, scPort = SPChoices $ singleton $ fromIntegral por
, scRedi = Nothing
, scFake = False
, scType = STMultiHttps (MTC vTlsC) $ ReqApi
{ rcReq = onReq Secure
, rcKil = onKil
}
}
pure $ MultiEyreApi
{ meaLive = vLive
, meaPlan = vPlan
, meaCanc = vCanc
, meaTlsC = vTlsC
, meaConf = conf
, meaKill = traverse_ saKil (toList mIns <> toList mSec)
}

View File

@ -0,0 +1,44 @@
{-|
Eyre: Http Server Driver
-}
module Urbit.Vere.Eyre.PortsFile
( Ports(..)
, writePortsFile
, removePortsFile
)
where
import Urbit.Prelude
import System.Directory (doesFileExist, removeFile)
import Urbit.Arvo (Port(unPort))
-- Types -----------------------------------------------------------------------
data Ports = Ports
{ pHttps :: Maybe Port
, pHttp :: Port
, pLoop :: Port
}
deriving (Eq, Ord, Show)
-- Creating and Deleting `.http.ports` files. ----------------------------------
portsFileText :: Ports -> Text
portsFileText Ports {..} = unlines $ catMaybes
[ pHttps <&> \p -> (tshow p <> " secure public")
, Just (tshow (unPort pHttp) <> " insecure public")
, Just (tshow (unPort pLoop) <> " insecure loopback")
]
removePortsFile :: FilePath -> IO ()
removePortsFile pax = do
doesFileExist pax >>= \case
True -> removeFile pax
False -> pure ()
writePortsFile :: FilePath -> Ports -> IO ()
writePortsFile f = writeFile f . encodeUtf8 . portsFileText

View File

@ -0,0 +1,356 @@
{-|
Runs a single HTTP (or HTTPS) server for the eyre driver.
A server is given:
- A port, or a range or ports.
- Opens a socket on one of those ports.
- If this fails, try again repeatedly.
- Once a socket is opened, runs an HTTP server on the specified port.
- Once the server is up, calls a callback with the port that was opened.
- Once we have chosen a port, we commit to that port (ignoring the
original range).
- If the socket ever goes down, keep trying to reopen that port forever.
- When the server is shutdown, make sure the socket is closed.
TODO How to detect socket closed during server run?
-}
{-# OPTIONS_GHC -Wno-deprecations #-}
module Urbit.Vere.Eyre.Serv
( ServApi(..)
, TlsConfig(..)
, MultiTlsConfig(..)
, ReqApi(..)
, ServType(..)
, ServPort(..)
, ServHost(..)
, ServConf(..)
, configCreds
, serv
, fakeServ
)
where
import Urbit.Prelude hiding (Builder)
import Data.Default (def)
import Data.List.NonEmpty (NonEmpty((:|)))
import Network.TLS (Credential, Credentials(..), ServerHooks(..))
import Network.TLS (credentialLoadX509ChainFromMemory)
import RIO.Prelude (decodeUtf8Lenient)
import qualified Control.Monad.STM as STM
import qualified Data.Char as C
import qualified Network.Socket as Net
import qualified Network.Wai as W
import qualified Network.Wai.Handler.Warp as W
import qualified Network.Wai.Handler.WarpTLS as W
import qualified Urbit.Ob as Ob
import qualified Urbit.Vere.Eyre.Wai as E
-- Internal Types --------------------------------------------------------------
data ServApi = ServApi
{ saKil :: STM ()
, saPor :: STM W.Port
}
data TlsConfig = TlsConfig
{ tcPrKey :: ByteString
, tcCerti :: ByteString
, tcChain :: [ByteString]
}
deriving (Show)
newtype MultiTlsConfig = MTC (TVar (Map Ship (TlsConfig, Credential)))
instance Show MultiTlsConfig where
show = const "MultiTlsConfig"
data ReqApi = ReqApi
{ rcReq :: Ship -> Word64 -> E.ReqInfo -> STM ()
, rcKil :: Ship -> Word64 -> STM ()
}
instance Show ReqApi where
show = const "ReqApi"
data ServType
= STHttp Ship ReqApi
| STHttps Ship TlsConfig ReqApi
| STMultiHttp ReqApi
| STMultiHttps MultiTlsConfig ReqApi
deriving (Show)
data ServPort
= SPAnyPort
| SPChoices (NonEmpty W.Port)
deriving (Show)
data ServHost
= SHLocalhost
| SHAnyHostOk
deriving (Show)
data ServConf = ServConf
{ scType :: ServType
, scHost :: ServHost
, scPort :: ServPort
, scRedi :: Maybe W.Port
, scFake :: Bool
}
deriving (Show)
-- Opening Sockets -------------------------------------------------------------
getBindAddr :: String -> W.Port -> IO Net.SockAddr
getBindAddr hos por =
Net.getAddrInfo Nothing (Just hos) (Just (show por)) >>= \case
[] -> error "this should never happen."
x : _ -> pure (Net.addrAddress x)
bindListenPort :: String -> W.Port -> Net.Socket -> IO Net.PortNumber
bindListenPort hos por sok = do
Net.bind sok =<< getBindAddr hos por
Net.listen sok 1
Net.socketPort sok
tcpSocket :: IO (Either IOError Net.Socket)
tcpSocket =
tryIOError (Net.socket Net.AF_INET Net.Stream Net.defaultProtocol)
tryOpen :: String -> W.Port -> IO (Either IOError (W.Port, Net.Socket))
tryOpen hos por =
tcpSocket >>= \case
Left exn -> pure (Left exn)
Right sok -> tryIOError (bindListenPort hos por sok) >>= \case
Left exn -> Net.close sok $> Left exn
Right por -> pure (Right (fromIntegral por, sok))
openFreePort :: String -> IO (Either IOError (W.Port, Net.Socket))
openFreePort hos = do
tcpSocket >>= \case
Left exn -> pure (Left exn)
Right sok -> tryIOError (doBind sok) >>= \case
Left exn -> Net.close sok $> Left exn
Right ps -> pure (Right ps)
where
doBind sok = do
adr <- Net.inet_addr hos
Net.bind sok (Net.SockAddrInet Net.defaultPort adr)
Net.listen sok 1
port <- Net.socketPort sok
pure (fromIntegral port, sok)
retry :: HasLogFunc e => RIO e (Either IOError a) -> RIO e a
retry act = act >>= \case
Right res -> pure res
Left exn -> do
logDbg ctx ("Failed to open ports. Waiting 5s, then trying again.", exn)
threadDelay 5_000_000
retry act
where
ctx = ["EYRE", "SERV", "retry"]
tryOpenChoices
:: HasLogFunc e
=> String
-> NonEmpty W.Port
-> RIO e (Either IOError (W.Port, Net.Socket))
tryOpenChoices hos = go
where
go (p :| ps) = do
logDebug (displayShow ("EYRE", "Trying to open port.", p))
io (tryOpen hos p) >>= \case
Left err -> do
logError (displayShow ("EYRE", "Failed to open port.", p))
case ps of
[] -> pure (Left err)
q : qs -> go (q :| qs)
Right (p, s) -> do
pure (Right (p, s))
tryOpenAny
:: HasLogFunc e => String -> RIO e (Either IOError (W.Port, Net.Socket))
tryOpenAny hos = do
let ctx = ["EYRE", "SERV", "tryOpenAny"]
logDbg ctx "Asking the OS for any free port."
io (openFreePort hos) >>= \case
Left exn -> pure (Left exn)
Right (p, s) -> do
pure (Right (p, s))
logDbg :: (HasLogFunc e, Show a) => [Text] -> a -> RIO e ()
logDbg ctx msg = logDebug (prefix <> suffix)
where
prefix = display (concat $ fmap (<> ": ") ctx)
suffix = displayShow msg
forceOpenSocket
:: forall e
. HasLogFunc e
=> ServHost
-> ServPort
-> RAcquire e (W.Port, Net.Socket)
forceOpenSocket hos por = mkRAcquire opn kil
where
kil = io . Net.close . snd
opn = do
let ctx = ["EYRE", "SERV", "forceOpenSocket"]
logDbg ctx (hos, por)
(p, s) <- retry $ case por of
SPAnyPort -> tryOpenAny bind
SPChoices ps -> tryOpenChoices bind ps
logDbg ctx ("Opened port.", p)
pure (p, s)
bind = case hos of
SHLocalhost -> "127.0.0.1"
SHAnyHostOk -> "0.0.0.0"
-- Starting WAI ----------------------------------------------------------------
hostShip :: Maybe ByteString -> IO Ship
hostShip Nothing = error "Request must contain HOST header."
hostShip (Just bs) = byteShip (hedLabel bs) & \case
Left err -> error ("Bad host prefix. Must be a ship name: " <> unpack err)
Right sp -> pure sp
where
byteShip = fmap (fromIntegral . Ob.fromPatp) . bytePatp
bytePatp = Ob.parsePatp . decodeUtf8Lenient
hedLabel = fst . break (== fromIntegral (C.ord '.'))
onSniHdr
:: HasLogFunc e => e -> MultiTlsConfig -> Maybe String -> IO Credentials
onSniHdr env (MTC mtls) mHos = do
tabl <- atomically (readTVar mtls)
runRIO env $ logDbg ctx (tabl, mHos)
ship <- hostShip (encodeUtf8 . pack <$> mHos)
runRIO env $ logDbg ctx ship
tcfg <- lookup ship tabl & maybe (notRunning ship) (pure . snd)
runRIO env $ logDbg ctx tcfg
pure (Credentials [tcfg])
where
notRunning ship = error ("Ship not running: ~" <> show ship)
ctx = ["EYRE", "HTTPS", "SNI"]
startServer
:: HasLogFunc e
=> ServType
-> ServHost
-> W.Port
-> Net.Socket
-> Maybe W.Port
-> TVar E.LiveReqs
-> RIO e ()
startServer typ hos por sok red vLive = do
envir <- ask
let host = case hos of
SHLocalhost -> "127.0.0.1"
SHAnyHostOk -> "*"
let opts =
W.defaultSettings
& W.setHost host
& W.setPort (fromIntegral por)
& W.setTimeout (5 * 60)
let runAppl who = E.app envir who vLive
reqShip = hostShip . W.requestHeaderHost
case typ of
STHttp who api -> do
let app = runAppl who (rcReq api who) (rcKil api who)
io (W.runSettingsSocket opts sok app)
STHttps who TlsConfig {..} api -> do
let tls = W.tlsSettingsChainMemory tcCerti tcChain tcPrKey
let app = runAppl who (rcReq api who) (rcKil api who)
io (W.runTLSSocket tls opts sok app)
STMultiHttp api -> do
let app req resp = do
who <- reqShip req
runAppl who (rcReq api who) (rcKil api who) req resp
io (W.runSettingsSocket opts sok app)
STMultiHttps mtls api -> do
TlsConfig {..} <- atomically (getFirstTlsConfig mtls)
let sni = def { onServerNameIndication = onSniHdr envir mtls }
let tlsSing = (W.tlsSettingsChainMemory tcCerti tcChain tcPrKey)
let tlsMany = tlsSing { W.tlsServerHooks = sni }
let ctx = ["EYRE", "HTTPS", "REQ"]
let
app = \req resp -> do
runRIO envir $ logDbg ctx "Got request"
who <- reqShip req
runRIO envir $ logDbg ctx ("Parsed HOST", who)
runAppl who (rcReq api who) (rcKil api who) req resp
io (W.runTLSSocket tlsMany opts sok app)
--------------------------------------------------------------------------------
configCreds :: TlsConfig -> Either Text Credential
configCreds TlsConfig {..} =
credentialLoadX509ChainFromMemory tcCerti tcChain tcPrKey & \case
Left str -> Left (pack str)
Right rs -> Right rs
fakeServ :: HasLogFunc e => ServConf -> RIO e ServApi
fakeServ conf = do
let por = fakePort (scPort conf)
logDebug (displayShow ("EYRE", "SERV", "Running Fake Server", por))
pure $ ServApi
{ saKil = pure ()
, saPor = pure por
}
where
fakePort :: ServPort -> W.Port
fakePort SPAnyPort = 55555
fakePort (SPChoices (x :| _)) = x
getFirstTlsConfig :: MultiTlsConfig -> STM TlsConfig
getFirstTlsConfig (MTC var) = do
map <- readTVar var
case toList map of
[] -> STM.retry
x:_ -> pure (fst x)
realServ :: HasLogFunc e => TVar E.LiveReqs -> ServConf -> RIO e ServApi
realServ vLive conf@ServConf {..} = do
logDebug (displayShow ("EYRE", "SERV", "Running Real Server"))
kil <- newEmptyTMVarIO
por <- newEmptyTMVarIO
tid <- async (runServ por)
_ <- async (atomically (takeTMVar kil) >> cancel tid)
pure $ ServApi
{ saKil = void (tryPutTMVar kil ())
, saPor = readTMVar por
}
where
runServ vPort = do
logDebug (displayShow ("EYRE", "SERV", "runServ"))
rwith (forceOpenSocket scHost scPort) $ \(por, sok) -> do
atomically (putTMVar vPort por)
startServer scType scHost por sok scRedi vLive
serv :: HasLogFunc e => TVar E.LiveReqs -> ServConf -> RIO e ServApi
serv vLive conf = do
if scFake conf
then fakeServ conf
else realServ vLive conf

View File

@ -0,0 +1,67 @@
{-|
Eyre: Http Server Driver
-}
module Urbit.Vere.Eyre.Service
( restartService
, stopService
)
where
import Urbit.Prelude
-- Generic Service Stop/Restart -- Using an MVar for Atomicity -----------------
{-|
Restart a running service.
This can probably be made simpler, but it
- Sets the MVar to Nothing if there was an exception while starting
or stopping the service.
- Keeps the MVar lock until the restart process finishes.
-}
restartService
:: forall e s
. HasLogFunc e
=> MVar (Maybe s)
-> RIO e s
-> (s -> RIO e ())
-> RIO e (Either SomeException s)
restartService vServ sstart kkill = do
logDebug "restartService"
modifyMVar vServ $ \case
Nothing -> doStart
Just sv -> doRestart sv
where
doRestart :: s -> RIO e (Maybe s, Either SomeException s)
doRestart serv = do
logDebug "doStart"
try (kkill serv) >>= \case
Left exn -> pure (Nothing, Left exn)
Right () -> doStart
doStart :: RIO e (Maybe s, Either SomeException s)
doStart = do
logDebug "doStart"
try sstart <&> \case
Right s -> (Just s, Right s)
Left exn -> (Nothing, Left exn)
{-|
Stop a running service. Do nothing if it's already stopped.
-}
stopService
:: HasLogFunc e
=> MVar (Maybe s)
-> (s -> RIO e ())
-> RIO e (Either SomeException ())
stopService vServ kkill = do
logDebug "stopService"
modifyMVar vServ $ \case
Nothing -> pure (Nothing, Right ())
Just sv -> do
res <- try (kkill sv)
pure (Nothing, res)

View File

@ -0,0 +1,229 @@
{-|
WAI Application for `eyre` driver.
# Request Lifecycles
- Requests come in, are given an identifier and are passed to a callback.
- When requests timeout, the identifier is passed to anothing callback.
- The server pulls response actions, and passes them to the associated
request.
-}
module Urbit.Vere.Eyre.Wai
( RespAct(..)
, RespApi(..)
, LiveReqs(..)
, ReqInfo(..)
, emptyLiveReqs
, routeRespAct
, rmLiveReq
, newLiveReq
, app
)
where
import Urbit.Prelude hiding (Builder)
import Data.Binary.Builder (Builder, fromByteString)
import Data.Bits (shiftL, (.|.))
import Data.Conduit (ConduitT, Flush(Chunk, Flush), yield)
import Network.Socket (SockAddr(..))
import System.Random (newStdGen, randoms)
import Urbit.Arvo (Address(..), Ipv4(..), Ipv6(..), Method)
import qualified Network.HTTP.Types as H
import qualified Network.Wai as W
import qualified Network.Wai.Conduit as W
-- Types -----------------------------------------------------------------------
data RespAct
= RAFull H.Status [H.Header] ByteString
| RAHead H.Status [H.Header] ByteString
| RABloc ByteString
| RADone
deriving (Eq, Ord, Show)
data RespApi = RespApi
{ raAct :: RespAct -> STM Bool
, raKil :: STM ()
}
data LiveReqs = LiveReqs
{ reqIdSuply :: [Word64]
, activeReqs :: Map Word64 (Ship, RespApi)
}
data ReqInfo = ReqInfo
{ riAdr :: Address
, riMet :: H.StdMethod
, riUrl :: ByteString
, riHdr :: [H.Header]
, riBod :: ByteString
}
-- Live Requests Table -- All Requests Still Waiting for Responses -------------
emptyLiveReqs :: IO LiveReqs
emptyLiveReqs = io $ do
gen <- newStdGen
pure (LiveReqs (randoms gen) mempty)
routeRespAct :: Ship -> TVar LiveReqs -> Word64 -> RespAct -> STM Bool
routeRespAct who vLiv reqId act =
(lookup reqId . activeReqs <$> readTVar vLiv) >>= \case
Nothing -> pure False
Just (own, tv) -> do
if (who == own)
then raAct tv act
else pure False
rmLiveReq :: TVar LiveReqs -> Word64 -> STM ()
rmLiveReq var reqId = modifyTVar' var
$ \liv -> liv { activeReqs = deleteMap reqId (activeReqs liv) }
allocateReqId :: TVar LiveReqs -> STM Word64
allocateReqId var = do
LiveReqs supply tbl <- readTVar var
let loop :: [Word64] -> (Word64, [Word64])
loop [] = error "impossible"
loop (x:xs) | member x tbl = loop xs
loop (x:xs) | otherwise = (x, xs)
let (fresh, supply') = loop supply
writeTVar var (LiveReqs supply' tbl)
pure fresh
newLiveReq :: Ship -> TVar LiveReqs -> STM (Word64, STM RespAct)
newLiveReq who var = do
tmv <- newTQueue
kil <- newEmptyTMVar
nex <- allocateReqId var
LiveReqs sup tbl <- readTVar var
let waitAct = (<|>) (readTMVar kil $> RADone) (readTQueue tmv)
respApi = RespApi
{ raKil = putTMVar kil ()
, raAct = \act -> tryReadTMVar kil >>= \case
Nothing -> writeTQueue tmv act $> True
Just () -> pure False
}
writeTVar var (LiveReqs sup (insertMap nex (who, respApi) tbl))
pure (nex, waitAct)
-- Random Helpers --------------------------------------------------------------
cookMeth :: W.Request -> Maybe Method
cookMeth = H.parseMethod . W.requestMethod >>> \case
Left _ -> Nothing
Right m -> Just m
reqAddr :: W.Request -> Address
reqAddr = W.remoteHost >>> \case
SockAddrInet _ a -> AIpv4 (Ipv4 a)
SockAddrInet6 _ _ a _ -> AIpv6 (mkIpv6 a)
_ -> error "invalid sock addr"
mkIpv6 :: (Word32, Word32, Word32, Word32) -> Ipv6
mkIpv6 (p, q, r, s) = Ipv6 (pBits .|. qBits .|. rBits .|. sBits)
where
pBits = shiftL (fromIntegral p) 0
qBits = shiftL (fromIntegral q) 32
rBits = shiftL (fromIntegral r) 64
sBits = shiftL (fromIntegral s) 96
reqUrl :: W.Request -> ByteString
reqUrl r = W.rawPathInfo r <> W.rawQueryString r
-- Responses -------------------------------------------------------------------
noHeader :: HasLogFunc e => RIO e a
noHeader = do
logError "Response block with no response header."
error "Bad HttpEvent: Response block with no response header."
dupHead :: HasLogFunc e => RIO e a
dupHead = do
logError "Multiple %head actions on one request"
error "Bad HttpEvent: Multiple header actions per on one request."
{-|
- Immediately yield all of the initial chunks
- Yield the data from %bloc action.
- Close the stream when we hit a %done action.
-}
streamBlocks
:: HasLogFunc e
=> e
-> ByteString
-> STM RespAct
-> ConduitT () (Flush Builder) IO ()
streamBlocks env init getAct = send init >> loop
where
loop = atomically getAct >>= \case
RAHead _ _ _ -> runRIO env dupHead
RAFull _ _ _ -> runRIO env dupHead
RADone -> pure ()
RABloc c -> send c >> loop
send "" = pure ()
send c = do
runRIO env (logTrace (display ("sending chunk " <> tshow c)))
yield $ Chunk $ fromByteString c
yield Flush
sendResponse
:: HasLogFunc e
=> (W.Response -> IO W.ResponseReceived)
-> STM RespAct
-> RIO e W.ResponseReceived
sendResponse cb waitAct = do
env <- ask
atomically waitAct >>= \case
RADone -> io $ cb $ W.responseLBS (H.mkStatus 444 "No Response") [] ""
RAFull s h b -> io $ cb $ W.responseLBS s h $ fromStrict b
RAHead s h b -> io $ cb $ W.responseSource s h $ streamBlocks env b waitAct
RABloc _ -> noHeader
liveReq :: Ship -> TVar LiveReqs -> RAcquire e (Word64, STM RespAct)
liveReq who vLiv = mkRAcquire ins del
where
ins = atomically (newLiveReq who vLiv)
del = atomically . rmLiveReq vLiv . fst
app
:: HasLogFunc e
=> e
-> Ship
-> TVar LiveReqs
-> (Word64 -> ReqInfo -> STM ())
-> (Word64 -> STM ())
-> W.Application
app env who liv inform cancel req respond =
runRIO env $ rwith (liveReq who liv) $ \(reqId, respApi) -> do
bod <- io (toStrict <$> W.strictRequestBody req)
met <- maybe (error "bad method") pure (cookMeth req)
let adr = reqAddr req
hdr = W.requestHeaders req
url = reqUrl req
atomically $ inform reqId $ ReqInfo adr met url hdr bod
try (sendResponse respond respApi) >>= \case
Right rr -> pure rr
Left exn -> do
atomically (cancel reqId)
logError $ display ("Exception during request" <> tshow exn)
throwIO (exn :: SomeException)

View File

@ -7,19 +7,22 @@
module Urbit.Vere.Http.Client where module Urbit.Vere.Http.Client where
import Urbit.Arvo (BlipEv(..), Ev(..), HttpClientEf(..), import Urbit.Prelude hiding (Builder)
HttpClientEv(..), HttpClientReq(..),
HttpEvent(..), KingId, ResponseHeader(..))
import Urbit.Prelude hiding (Builder)
import Urbit.Vere.Pier.Types
import Urbit.Vere.Http import Urbit.Vere.Http
import Urbit.Vere.Pier.Types
import Urbit.King.App
import Urbit.Arvo (BlipEv(..), Ev(..), HttpClientEf(..), HttpClientEv(..),
HttpClientReq(..), HttpEvent(..), KingId, ResponseHeader(..))
import qualified Data.Map as M import qualified Data.Map as M
import qualified Network.HTTP.Client as H import qualified Network.HTTP.Client as H
import qualified Network.HTTP.Client.TLS as TLS import qualified Network.HTTP.Client.TLS as TLS
import qualified Network.HTTP.Types as HT import qualified Network.HTTP.Types as HT
-- Types ----------------------------------------------------------------------- -- Types -----------------------------------------------------------------------
type ReqId = Word type ReqId = Word
@ -54,14 +57,54 @@ bornEv king =
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
client :: forall e. HasLogFunc e _bornFailed :: e -> WorkError -> IO ()
=> KingId -> QueueEv -> ([Ev], RAcquire e (EffCb e HttpClientEf)) _bornFailed env _ = runRIO env $ do
client kingId enqueueEv = (initialEvents, runHttpClient) pure () -- TODO What to do in this case?
client'
:: HasPierEnv e
=> RIO e ([Ev], RAcquire e (DriverApi HttpClientEf))
client' = do
ventQ :: TQueue EvErr <- newTQueueIO
env <- ask
let (bornEvs, startDriver) = client env (writeTQueue ventQ)
let runDriver = do
diOnEffect <- startDriver
let diEventSource = fmap RRWork <$> tryReadTQueue ventQ
pure (DriverApi {..})
pure (bornEvs, runDriver)
{-|
Iris -- HTTP Client Driver
Until born events succeeds, ignore effects.
Wait until born event callbacks invoked.
If success, signal success.
If failure, try again several times.
If still failure, bring down ship.
Once born event succeeds, hold on to effects.
Once all other drivers have booted:
- Execute stashed effects.
- Begin normal operation (start accepting requests)
-}
client
:: forall e
. (HasLogFunc e, HasKingId e)
=> e
-> (EvErr -> STM ())
-> ([Ev], RAcquire e (HttpClientEf -> IO ()))
client env plan = (initialEvents, runHttpClient)
where where
kingId = view (kingIdL . to fromIntegral) env
initialEvents :: [Ev] initialEvents :: [Ev]
initialEvents = [bornEv kingId] initialEvents = [bornEv kingId]
runHttpClient :: RAcquire e (EffCb e HttpClientEf) runHttpClient :: RAcquire e (HttpClientEf -> IO ())
runHttpClient = handleEffect <$> mkRAcquire start stop runHttpClient = handleEffect <$> mkRAcquire start stop
start :: RIO e (HttpClientDrv) start :: RIO e (HttpClientDrv)
@ -75,10 +118,10 @@ client kingId enqueueEv = (initialEvents, runHttpClient)
liveThreads <- atomically $ readTVar hcdLive liveThreads <- atomically $ readTVar hcdLive
mapM_ cancel liveThreads mapM_ cancel liveThreads
handleEffect :: HttpClientDrv -> HttpClientEf -> RIO e () handleEffect :: HttpClientDrv -> HttpClientEf -> IO ()
handleEffect drv = \case handleEffect drv = \case
HCERequest _ id req -> newReq drv id req HCERequest _ id req -> runRIO env (newReq drv id req)
HCECancelRequest _ id -> cancelReq drv id HCECancelRequest _ id -> runRIO env (cancelReq drv id)
newReq :: HttpClientDrv -> ReqId -> HttpClientReq -> RIO e () newReq :: HttpClientDrv -> ReqId -> HttpClientReq -> RIO e ()
newReq drv id req = do newReq drv id req = do
@ -124,8 +167,14 @@ client kingId enqueueEv = (initialEvents, runHttpClient)
planEvent :: ReqId -> HttpEvent -> RIO e () planEvent :: ReqId -> HttpEvent -> RIO e ()
planEvent id ev = do planEvent id ev = do
logDebug $ displayShow ("(http client response)", id, (describe ev)) logDebug $ displayShow ("(http client response)", id, (describe ev))
atomically $ enqueueEv $ EvBlip $ BlipEvHttpClient $
HttpClientEvReceive (kingId, ()) (fromIntegral id) ev let recvEv = EvBlip
$ BlipEvHttpClient
$ HttpClientEvReceive (kingId, ()) (fromIntegral id) ev
let recvFailed _ = pure ()
atomically $ plan (EvErr recvEv recvFailed)
-- show an HttpEvent with byte count instead of raw data -- show an HttpEvent with byte count instead of raw data
describe :: HttpEvent -> String describe :: HttpEvent -> String

View File

@ -1,635 +0,0 @@
{-|
Http Server Driver
TODO Make sure that HTTP sockets get closed on shutdown.
TODO What is this about?
// if we don't explicitly set this field, h2o will send with
// transfer-encoding: chunked
//
if ( 1 == has_len_i ) {
rec_u->res.content_length = ( 0 == gen_u->bod_u ) ?
0 : gen_u->bod_u->len_w;
}
TODO Does this matter, is is using WAI's default behavior ok?
rec_u->res.reason = (status < 200) ? "weird" :
(status < 300) ? "ok" :
(status < 400) ? "moved" :
(status < 500) ? "missing" :
"hosed";
-}
module Urbit.Vere.Http.Server where
import Data.Conduit
import Urbit.Arvo hiding (ServerId, reqBody, reqUrl, secure)
import Urbit.King.Config
import Urbit.Noun
import Urbit.Prelude hiding (Builder)
import Urbit.Vere.Pier.Types
import Data.Binary.Builder (Builder, fromByteString)
import Data.Bits (shiftL, (.|.))
import Data.PEM (pemParseBS, pemWriteBS)
import Network.Socket (SockAddr(..))
import System.Directory (doesFileExist, removeFile)
import System.Random (randomIO)
import Urbit.Vere.Http (convertHeaders, unconvertHeaders)
import qualified Network.HTTP.Types as H
import qualified Network.Socket as Net
import qualified Network.Wai as W
import qualified Network.Wai.Conduit as W
import qualified Network.Wai.Handler.Warp as W
import qualified Network.Wai.Handler.WarpTLS as W
-- Internal Types --------------------------------------------------------------
type HasShipEnv e = (HasLogFunc e, HasNetworkConfig e, HasPierConfig e)
type ReqId = UD
type SeqId = UD -- Unused, always 1
{-|
The sequence of actions on a given request *should* be:
[%head .] [%bloc .]* %done
But we will actually accept anything, and mostly do the right
thing. There are two situations where we ignore ignore the data from
some actions.
- If you send something *after* a %done action, it will be ignored.
- If you send a %done before a %head, we will produce "444 No
Response" with an empty response body.
-}
data RespAction
= RAHead ResponseHeader File
| RAFull ResponseHeader File
| RABloc File
| RADone
deriving (Eq, Ord, Show)
data LiveReqs = LiveReqs
{ nextReqId :: ReqId
, activeReqs :: Map ReqId (TQueue RespAction)
}
data Ports = Ports
{ pHttps :: Maybe Port
, pHttp :: Port
, pLoop :: Port
}
deriving (Eq, Ord, Show)
newtype Drv = Drv { unDrv :: MVar (Maybe Serv) }
data Serv = Serv
{ sServId :: ServId
, sConfig :: HttpServerConf
, sLoopTid :: Async ()
, sHttpTid :: Async ()
, sHttpsTid :: Maybe (Async ())
, sLoopSock :: Net.Socket
, sHttpSock :: Net.Socket
, sHttpsSock :: Net.Socket
, sPorts :: Ports
, sPortsFile :: FilePath
, sLiveReqs :: TVar LiveReqs
}
-- RespAction -- Reorganized HttpEvent for Cleaner Processing ------------------
reorgHttpEvent :: HttpEvent -> [RespAction]
reorgHttpEvent = \case
Start head mBlk True -> [RAFull head (fromMaybe "" mBlk)]
Start head mBlk False -> [RAHead head (fromMaybe "" mBlk)]
Cancel () -> [RADone]
Continue mBlk isDone -> toList (RABloc <$> mBlk)
<> if isDone then [RADone] else []
-- Generic Service Stop/Restart -- Using an MVar for Atomicity -----------------
{-|
Restart a running service.
This can probably be made simpler, but it
- Sets the MVar to Nothing if there was an exception whil starting
or stopping the service.
- Keeps the MVar lock until the restart process finishes.
-}
restartService :: e s. HasLogFunc e
=> MVar (Maybe s)
-> RIO e s
-> (s -> RIO e ())
-> RIO e (Either SomeException s)
restartService vServ sstart kkill = do
logDebug "restartService"
modifyMVar vServ $ \case
Nothing -> doStart
Just sv -> doRestart sv
where
doRestart :: s -> RIO e (Maybe s, Either SomeException s)
doRestart serv = do
logDebug "doStart"
try (kkill serv) >>= \case
Left exn -> pure (Nothing, Left exn)
Right () -> doStart
doStart :: RIO e (Maybe s, Either SomeException s)
doStart = do
logDebug "doStart"
try sstart <&> \case
Right s -> (Just s, Right s)
Left exn -> (Nothing, Left exn)
stopService :: HasLogFunc e
=> MVar (Maybe s)
-> (s -> RIO e ())
-> RIO e (Either SomeException ())
stopService vServ kkill = do
logDebug "stopService"
modifyMVar vServ $ \case
Nothing -> pure (Nothing, Right ())
Just sv -> do res <- try (kkill sv)
pure (Nothing, res)
-- Live Requests Table -- All Requests Still Waiting for Responses -------------
emptyLiveReqs :: LiveReqs
emptyLiveReqs = LiveReqs 1 mempty
respondToLiveReq :: TVar LiveReqs -> ReqId -> RespAction -> STM ()
respondToLiveReq var req ev = do
mVar <- lookup req . activeReqs <$> readTVar var
case mVar of
Nothing -> pure ()
Just tv -> writeTQueue tv ev
rmLiveReq :: TVar LiveReqs -> ReqId -> STM ()
rmLiveReq var reqId = do
liv <- readTVar var
writeTVar var (liv { activeReqs = deleteMap reqId (activeReqs liv) })
newLiveReq :: TVar LiveReqs -> STM (ReqId, TQueue RespAction)
newLiveReq var = do
liv <- readTVar var
tmv <- newTQueue
let (nex, act) = (nextReqId liv, activeReqs liv)
writeTVar var (LiveReqs (nex+1) (insertMap nex tmv act))
pure (nex, tmv)
-- Ports File ------------------------------------------------------------------
removePortsFile :: FilePath -> RIO e ()
removePortsFile pax =
io (doesFileExist pax) >>= \case
True -> io $ removeFile pax
False -> pure ()
portsFileText :: Ports -> Text
portsFileText Ports{..} =
unlines $ catMaybes
[ pHttps <&> \p -> (tshow p <> " secure public")
, Just (tshow (unPort pHttp) <> " insecure public")
, Just (tshow (unPort pLoop) <> " insecure loopback")
]
writePortsFile :: FilePath -> Ports -> RIO e ()
writePortsFile f = writeFile f . encodeUtf8 . portsFileText
-- Random Helpers --------------------------------------------------------------
cordBytes :: Cord -> ByteString
cordBytes = encodeUtf8 . unCord
wainBytes :: Wain -> ByteString
wainBytes = encodeUtf8 . unWain
pass :: Monad m => m ()
pass = pure ()
whenJust :: Monad m => Maybe a -> (a -> m ()) -> m ()
whenJust Nothing act = pure ()
whenJust (Just a) act = act a
cookMeth :: W.Request -> Maybe Method
cookMeth = H.parseMethod . W.requestMethod >>> \case
Left _ -> Nothing
Right m -> Just m
reqIdCord :: ReqId -> Cord
reqIdCord = Cord . tshow
reqBody :: W.Request -> RIO e (Maybe File)
reqBody req = do
bodyLbs <- io $ W.strictRequestBody req
pure $ if length bodyLbs == 0
then Nothing
else Just $ File $ Octs (toStrict bodyLbs)
reqAddr :: W.Request -> Address
reqAddr = W.remoteHost >>> \case
SockAddrInet _ a -> AIpv4 (Ipv4 a)
SockAddrInet6 _ _ a _ -> AIpv6 (mkIpv6 a)
_ -> error "invalid sock addr"
mkIpv6 :: (Word32, Word32, Word32, Word32) -> Ipv6
mkIpv6 (p, q, r, s) = Ipv6 (pBits .|. qBits .|. rBits .|. sBits)
where
pBits = shiftL (fromIntegral p) 0
qBits = shiftL (fromIntegral q) 32
rBits = shiftL (fromIntegral r) 64
sBits = shiftL (fromIntegral s) 96
reqUrl :: W.Request -> Cord
reqUrl r = Cord $ decodeUtf8 $ W.rawPathInfo r <> W.rawQueryString r
-- Utilities for Constructing Events -------------------------------------------
data WhichServer = Secure | Insecure | Loopback
deriving (Eq)
servEv :: HttpServerEv -> Ev
servEv = EvBlip . BlipEvHttpServer
bornEv :: KingId -> Ev
bornEv king =
servEv $ HttpServerEvBorn (king, ()) ()
liveEv :: ServId -> Ports -> Ev
liveEv sId Ports{..} =
servEv $ HttpServerEvLive (sId, ()) pHttp pHttps
cancelEv :: ServId -> ReqId -> Ev
cancelEv sId reqId =
servEv $ HttpServerEvCancelRequest (sId, reqId, 1, ()) ()
reqEv :: ServId -> ReqId -> WhichServer -> Address -> HttpRequest -> Ev
reqEv sId reqId which addr req =
case which of
Loopback ->
servEv $ HttpServerEvRequestLocal (sId, reqId, 1, ())
$ HttpServerReq False addr req
_ ->
servEv $ HttpServerEvRequest (sId, reqId, 1, ())
$ HttpServerReq (which == Secure) addr req
-- Http Server Flows -----------------------------------------------------------
data Resp
= RHead ResponseHeader [File]
| RFull ResponseHeader [File]
| RNone
deriving (Show)
{-|
This accepts all action orderings so that there are no edge-cases
to be handled:
- If %bloc before %head, collect it and wait for %head.
- If %done before %head, ignore all chunks and produce Nothing.
TODO Be strict about this instead. Ignore invalid request streams.
-}
getResp :: TQueue RespAction -> RIO e Resp
getResp tmv = go []
where
go çunks = atomically (readTQueue tmv) >>= \case
RAHead head ç -> pure $ RHead head $ reverse (ç : çunks)
RAFull head ç -> pure $ RFull head $ reverse (ç : çunks)
RABloc ç -> go (ç : çunks)
RADone -> pure RNone
{-|
- Immediatly yield all of the initial chunks
- Yield the data from %bloc action.
- Close the stream when we hit a %done action.
-}
streamBlocks :: HasLogFunc e
=> e -> [File] -> TQueue RespAction
-> ConduitT () (Flush Builder) IO ()
streamBlocks env init tmv =
for_ init yieldÇunk >> go
where
yieldFlush = \x -> yield (Chunk x) >> yield Flush
logDupHead = runRIO env (logError "Multiple %head actions on one request")
yieldÇunk = \case
"" -> runRIO env (logTrace "sending empty chunk")
c -> do runRIO env (logTrace (display ("sending chunk " <> tshow c)))
(yieldFlush . fromByteString . unOcts . unFile) c
go = atomically (readTQueue tmv) >>= \case
RAHead head c -> logDupHead >> yieldÇunk c >> go
RAFull head c -> logDupHead >> yieldÇunk c >> go
RABloc c -> yieldÇunk c >> go
RADone -> pure ()
sendResponse :: HasLogFunc e
=> (W.Response -> IO W.ResponseReceived)
-> TQueue RespAction
-> RIO e W.ResponseReceived
sendResponse cb tmv = do
env <- ask
getResp tmv >>= \case
RNone -> io $ cb $ W.responseLBS (H.mkStatus 444 "No Response") []
$ ""
RFull h f -> io $ cb $ W.responseLBS (hdrStatus h) (hdrHeaders h)
$ fromStrict $ concat $ unOcts . unFile <$> f
RHead h i -> io $ cb $ W.responseSource (hdrStatus h) (hdrHeaders h)
$ streamBlocks env i tmv
where
hdrHeaders :: ResponseHeader -> [H.Header]
hdrHeaders = unconvertHeaders . headers
hdrStatus :: ResponseHeader -> H.Status
hdrStatus = toEnum . fromIntegral . statusCode
liveReq :: TVar LiveReqs -> RAcquire e (ReqId, TQueue RespAction)
liveReq vLiv = mkRAcquire ins del
where
ins = atomically (newLiveReq vLiv)
del = atomically . rmLiveReq vLiv . fst
app :: HasLogFunc e
=> e -> ServId -> TVar LiveReqs -> (Ev -> STM ()) -> WhichServer
-> W.Application
app env sId liv plan which req respond =
runRIO env $
rwith (liveReq liv) $ \(reqId, respVar) -> do
body <- reqBody req
meth <- maybe (error "bad method") pure (cookMeth req)
let addr = reqAddr req
hdrs = convertHeaders $ W.requestHeaders req
evReq = HttpRequest meth (reqUrl req) hdrs body
atomically $ plan (reqEv sId reqId which addr evReq)
try (sendResponse respond respVar) >>= \case
Right rr -> pure rr
Left exn -> do
io $ atomically $ plan (cancelEv sId reqId)
logError $ display ("Exception during request" <> tshow exn)
throwIO (exn :: SomeException)
-- Top-Level Driver Interface --------------------------------------------------
data CantOpenPort = CantOpenPort W.Port
deriving (Eq, Ord, Show, Exception)
data WhichPort
= WPSpecific W.Port
| WPChoices [W.Port]
data SockOpts = SockOpts
{ soLocalhost :: Bool
, soWhich :: WhichPort
}
data PortsToTry = PortsToTry
{ pttSec :: SockOpts
, pttIns :: SockOpts
, pttLop :: SockOpts
}
{-|
Opens a socket on some port, accepting connections from `127.0.0.1`
if fake and `0.0.0.0` if real.
It will attempt to open a socket on each of the supplied ports in
order. If they all fail, it will ask the operating system to give
us an open socket on *any* open port. If that fails, it will throw
an exception.
-}
openPort :: forall e . HasLogFunc e => SockOpts -> RIO e (W.Port, Net.Socket)
openPort SockOpts {..} = case soWhich of
WPSpecific x -> insist (fromIntegral x)
WPChoices xs -> loop (fromIntegral <$> xs)
where
loop :: [W.Port] -> RIO e (W.Port, Net.Socket)
loop = \case
[] -> do
logTrace "Fallback: asking the OS to give us some free port."
ps <- io W.openFreePort
logTrace (display ("Opened port " <> tshow (fst ps)))
pure ps
x : xs -> do
logTrace (display ("Trying to open port " <> tshow x))
io (tryOpen x) >>= \case
Left (err :: IOError) -> do
logWarn (display ("Failed to open port " <> tshow x))
logWarn (display (tshow err))
loop xs
Right ps -> do
logTrace (display ("Opened port " <> tshow (fst ps)))
pure ps
insist :: W.Port -> RIO e (W.Port, Net.Socket)
insist p = do
logTrace (display ("Opening configured port " <> tshow p))
io (tryOpen p) >>= \case
Left (err :: IOError) -> do
logWarn (display ("Failed to open port " <> tshow p))
logWarn (display (tshow err))
throwIO (CantOpenPort p)
Right ps -> do
logTrace (display ("Opened port " <> tshow (fst ps)))
pure ps
bindTo = if soLocalhost then "127.0.0.1" else "0.0.0.0"
getBindAddr :: W.Port -> IO SockAddr
getBindAddr por =
Net.getAddrInfo Nothing (Just bindTo) (Just (show por)) >>= \case
[] -> error "this should never happen."
x : _ -> pure (Net.addrAddress x)
bindListenPort :: W.Port -> Net.Socket -> IO Net.PortNumber
bindListenPort por sok = do
Net.bind sok =<< getBindAddr por
Net.listen sok 1
Net.socketPort sok
-- `inet_addr`, `bind`, and `listen` all throw `IOError` if they fail.
tryOpen :: W.Port -> IO (Either IOError (W.Port, Net.Socket))
tryOpen por = do
sok <- Net.socket Net.AF_INET Net.Stream Net.defaultProtocol
try (bindListenPort por sok) >>= \case
Left exn -> Net.close sok $> Left exn
Right por -> pure (Right (fromIntegral por, sok))
httpServerPorts :: HasShipEnv e => Bool -> RIO e PortsToTry
httpServerPorts fak = do
ins <- view (networkConfigL . ncHttpPort . to (fmap fromIntegral))
sec <- view (networkConfigL . ncHttpsPort . to (fmap fromIntegral))
lop <- view (networkConfigL . ncLocalPort . to (fmap fromIntegral))
localMode <- view (networkConfigL . ncNetMode . to (== NMLocalhost))
let local = localMode || fak
let pttSec = case (sec, fak) of
(Just p , _ ) -> SockOpts local (WPSpecific p)
(Nothing, False) -> SockOpts local (WPChoices (443 : [8443 .. 8448]))
(Nothing, True ) -> SockOpts local (WPChoices ([8443 .. 8448]))
let pttIns = case (ins, fak) of
(Just p , _ ) -> SockOpts local (WPSpecific p)
(Nothing, False) -> SockOpts local (WPChoices (80 : [8080 .. 8085]))
(Nothing, True ) -> SockOpts local (WPChoices [8080 .. 8085])
let pttLop = case (lop, fak) of
(Just p , _) -> SockOpts local (WPSpecific p)
(Nothing, _) -> SockOpts local (WPChoices [12321 .. 12326])
pure (PortsToTry { .. })
parseCerts :: ByteString -> Maybe (ByteString, [ByteString])
parseCerts bs = do
pems <- pemParseBS bs & either (const Nothing) Just
case pems of
[] -> Nothing
p:ps -> pure (pemWriteBS p, pemWriteBS <$> ps)
startServ :: (HasPierConfig e, HasLogFunc e, HasNetworkConfig e)
=> Bool -> HttpServerConf -> (Ev -> STM ())
-> RIO e Serv
startServ isFake conf plan = do
logDebug "startServ"
let tls = do (PEM key, PEM certs) <- hscSecure conf
(cert, chain) <- parseCerts (wainBytes certs)
pure $ W.tlsSettingsChainMemory cert chain $ wainBytes key
sId <- io $ ServId . UV . fromIntegral <$> (randomIO :: IO Word32)
liv <- newTVarIO emptyLiveReqs
ptt <- httpServerPorts isFake
(httpPortInt, httpSock) <- openPort (pttIns ptt)
(httpsPortInt, httpsSock) <- openPort (pttSec ptt)
(loopPortInt, loopSock) <- openPort (pttLop ptt)
let httpPort = Port (fromIntegral httpPortInt)
httpsPort = Port (fromIntegral httpsPortInt)
loopPort = Port (fromIntegral loopPortInt)
let loopOpts = W.defaultSettings & W.setPort (fromIntegral loopPort)
& W.setHost "127.0.0.1"
& W.setTimeout (5 * 60)
httpOpts = W.defaultSettings & W.setHost "*"
& W.setPort (fromIntegral httpPort)
httpsOpts = W.defaultSettings & W.setHost "*"
& W.setPort (fromIntegral httpsPort)
env <- ask
logDebug "Starting loopback server"
loopTid <- async $ io
$ W.runSettingsSocket loopOpts loopSock
$ app env sId liv plan Loopback
logDebug "Starting HTTP server"
httpTid <- async $ io
$ W.runSettingsSocket httpOpts httpSock
$ app env sId liv plan Insecure
logDebug "Starting HTTPS server"
httpsTid <- for tls $ \tlsOpts ->
async $ io
$ W.runTLSSocket tlsOpts httpsOpts httpsSock
$ app env sId liv plan Secure
pierPath <- view pierPathL
let por = Ports (tls <&> const httpsPort) httpPort loopPort
fil = pierPath <> "/.http.ports"
logDebug $ displayShow (sId, por, fil)
logDebug "Finished started HTTP Servers"
pure $ Serv sId conf
loopTid httpTid httpsTid
httpSock httpsSock loopSock
por fil liv
killServ :: HasLogFunc e => Serv -> RIO e ()
killServ Serv{..} = do
cancel sLoopTid
cancel sHttpTid
traverse_ cancel sHttpsTid
io $ Net.close sHttpSock
io $ Net.close sHttpsSock
io $ Net.close sLoopSock
removePortsFile sPortsFile
(void . waitCatch) sLoopTid
(void . waitCatch) sHttpTid
traverse_ (void . waitCatch) sHttpsTid
kill :: HasLogFunc e => Drv -> RIO e ()
kill (Drv v) = stopService v killServ >>= fromEither
respond :: HasLogFunc e
=> Drv -> ReqId -> HttpEvent -> RIO e ()
respond (Drv v) reqId ev = do
readMVar v >>= \case
Nothing -> logWarn "Got a response to a request that does not exist."
Just sv -> do logDebug $ displayShow $ reorgHttpEvent ev
for_ (reorgHttpEvent ev) $
atomically . respondToLiveReq (sLiveReqs sv) reqId
serv :: e. HasShipEnv e
=> KingId -> QueueEv -> Bool
-> ([Ev], RAcquire e (EffCb e HttpServerEf))
serv king plan isFake =
(initialEvents, runHttpServer)
where
initialEvents :: [Ev]
initialEvents = [bornEv king]
runHttpServer :: RAcquire e (EffCb e HttpServerEf)
runHttpServer = handleEf <$> mkRAcquire (Drv <$> newMVar Nothing) kill
restart :: Drv -> HttpServerConf -> RIO e Serv
restart (Drv var) conf = do
logDebug "Restarting http server"
res <- fromEither =<<
restartService var (startServ isFake conf plan) killServ
logDebug "Done restating http server"
pure res
handleEf :: Drv -> HttpServerEf -> RIO e ()
handleEf drv = \case
HSESetConfig (i, ()) conf -> do
-- print (i, king)
-- when (i == fromIntegral king) $ do
logDebug "restarting"
Serv{..} <- restart drv conf
logDebug "Enqueue %live"
atomically $ plan (liveEv sServId sPorts)
logDebug "Write ports file"
writePortsFile sPortsFile sPorts
HSEResponse (i, req, _seq, ()) ev -> do
-- print (i, king)
-- when (i == fromIntegral king) $ do
logDebug "respond"
respond drv (fromIntegral req) ev

View File

@ -55,24 +55,24 @@ wsConn :: (FromNoun i, ToNoun o, Show i, Show o, HasLogFunc e)
-> WS.Connection -> WS.Connection
-> RIO e () -> RIO e ()
wsConn pre inp out wsc = do wsConn pre inp out wsc = do
logWarn (pre <> "(wcConn) Connected!") logDebug (pre <> "(wcConn) Connected!")
writer <- withRIOThread $ forever $ do writer <- withRIOThread $ forever $ do
logWarn (pre <> "(wsConn) Waiting for data.") logDebug (pre <> "(wsConn) Waiting for data.")
byt <- io $ toStrict <$> WS.receiveData wsc byt <- io $ toStrict <$> WS.receiveData wsc
logWarn (pre <> "Got data") logDebug (pre <> "Got data")
dat <- cueBSExn byt >>= fromNounExn dat <- cueBSExn byt >>= fromNounExn
logWarn (pre <> "(wsConn) Decoded data, writing to chan") logDebug (pre <> "(wsConn) Decoded data, writing to chan")
atomically $ writeTBMChan inp dat atomically $ writeTBMChan inp dat
reader <- withRIOThread $ forever $ do reader <- withRIOThread $ forever $ do
logWarn (pre <> "Waiting for data from chan") logDebug (pre <> "Waiting for data from chan")
atomically (readTBMChan out) >>= \case atomically (readTBMChan out) >>= \case
Nothing -> do Nothing -> do
logWarn (pre <> "(wsConn) Connection closed") logDebug (pre <> "(wsConn) Connection closed")
error "dead-conn" error "dead-conn"
Just msg -> do Just msg -> do
logWarn (pre <> "(wsConn) Got message! " <> displayShow msg) logDebug (pre <> "(wsConn) Got message! " <> displayShow msg)
io $ WS.sendBinaryData wsc $ fromStrict $ jamBS $ toNoun msg io $ WS.sendBinaryData wsc $ fromStrict $ jamBS $ toNoun msg
let cleanup = do let cleanup = do
@ -82,7 +82,7 @@ wsConn pre inp out wsc = do
flip finally cleanup $ do flip finally cleanup $ do
res <- atomically (waitCatchSTM writer <|> waitCatchSTM reader) res <- atomically (waitCatchSTM writer <|> waitCatchSTM reader)
logWarn $ displayShow (res :: Either SomeException ()) logDebug $ displayShow (res :: Either SomeException ())
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
@ -111,7 +111,7 @@ wsServApp :: (HasLogFunc e, ToNoun o, FromNoun i, Show i, Show o)
-> WS.PendingConnection -> WS.PendingConnection
-> RIO e () -> RIO e ()
wsServApp cb pen = do wsServApp cb pen = do
logError "NOUNSERV (wsServer) Got connection!" logDebug "NOUNSERV (wsServer) Got connection!"
wsc <- io $ WS.acceptRequest pen wsc <- io $ WS.acceptRequest pen
inp <- io $ newTBMChanIO 5 inp <- io $ newTBMChanIO 5
out <- io $ newTBMChanIO 5 out <- io $ newTBMChanIO 5
@ -125,10 +125,10 @@ wsServer = do
tid <- async $ do tid <- async $ do
env <- ask env <- ask
logError "NOUNSERV (wsServer) Starting server" logDebug "NOUNSERV (wsServer) Starting server"
io $ WS.runServer "127.0.0.1" 9999 io $ WS.runServer "127.0.0.1" 9999
$ runRIO env . wsServApp (writeTBMChan con) $ runRIO env . wsServApp (writeTBMChan con)
logError "NOUNSERV (wsServer) Server died" logDebug "NOUNSERV (wsServer) Server died"
atomically $ closeTBMChan con atomically $ closeTBMChan con
pure $ Server (readTBMChan con) tid 9999 pure $ Server (readTBMChan con) tid 9999
@ -147,34 +147,34 @@ example = Just (99, (), 44)
testIt :: HasLogFunc e => RIO e () testIt :: HasLogFunc e => RIO e ()
testIt = do testIt = do
logTrace "(testIt) Starting Server" logDebug "(testIt) Starting Server"
Server{..} <- wsServer @Example @Example Server{..} <- wsServer @Example @Example
logTrace "(testIt) Connecting" logDebug "(testIt) Connecting"
Client{..} <- wsClient @Example @Example "/" sData Client{..} <- wsClient @Example @Example "/" sData
logTrace "(testIt) Accepting connection" logDebug "(testIt) Accepting connection"
sConn <- fromJust "accept" =<< atomically sAccept sConn <- fromJust "accept" =<< atomically sAccept
let let
clientSend = do clientSend = do
logTrace "(testIt) Sending from client" logDebug "(testIt) Sending from client"
atomically (cSend cConn example) atomically (cSend cConn example)
logTrace "(testIt) Waiting for response" logDebug "(testIt) Waiting for response"
res <- atomically (cRecv sConn) res <- atomically (cRecv sConn)
print ("clientSend", res, example) print ("clientSend", res, example)
unless (res == Just example) $ do unless (res == Just example) $ do
error "Bad data" error "Bad data"
logInfo "(testIt) Success" logDebug "(testIt) Success"
serverSend = do serverSend = do
logTrace "(testIt) Sending from server" logDebug "(testIt) Sending from server"
atomically (cSend sConn example) atomically (cSend sConn example)
logTrace "(testIt) Waiting for response" logDebug "(testIt) Waiting for response"
res <- atomically (cRecv cConn) res <- atomically (cRecv cConn)
print ("serverSend", res, example) print ("serverSend", res, example)
unless (res == Just example) $ do unless (res == Just example) $ do
error "Bad data" error "Bad data"
logInfo "(testIt) Success" logDebug "(testIt) Success"
clientSend clientSend
clientSend clientSend

View File

@ -1,413 +1,553 @@
{-| {-|
Top-Level Pier Management Top-Level Pier Management
This is the code that starts the IO drivers and deals with This is the code that starts the IO drivers and deals with communication
communication between the serf, the log, and the IO drivers. between the serf, the event log, and the IO drivers.
-} -}
module Urbit.Vere.Pier module Urbit.Vere.Pier
( booted, resumed, getSnapshot, pier, runPersist, runCompute, generateBootSeq ( booted
) where , runSerf
, resumed
, getSnapshot
, pier
, runPersist
, runCompute
, genBootSeq
)
where
import Urbit.Prelude 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 Control.Monad.Trans.Maybe
import RIO.Directory
import Urbit.Arvo
import Urbit.King.App
import Urbit.Vere.Pier.Types
import Data.Text (append) import Control.Monad.STM (retry)
import System.Posix.Files (ownerModes, setFileMode) import System.Posix.Files (ownerModes, setFileMode)
import Urbit.King.App (HasConfigDir(..), HasStderrLogFunc(..)) import Urbit.EventLog.LMDB (EventLog)
import Urbit.Vere.Ames (ames) import Urbit.King.API (TermConn)
import Urbit.Vere.Behn (behn) import Urbit.Noun.Time (Wen)
import Urbit.Vere.Clay (clay) import Urbit.TermSize (TermSize(..))
import Urbit.Vere.Http.Client (client) import Urbit.Vere.Eyre.Multi (MultiEyreApi)
import Urbit.Vere.Http.Server (serv) import Urbit.Vere.Serf (Serf)
import Urbit.Vere.Log (EventLog)
import Urbit.Vere.Serf (Serf, SerfState(..), doJob, sStderr)
import qualified Data.Text as T
import qualified System.Entropy as Ent import qualified System.Entropy as Ent
import qualified Urbit.EventLog.LMDB as Log
import qualified Urbit.King.API as King import qualified Urbit.King.API as King
import qualified Urbit.Time as Time import qualified Urbit.Noun.Time as Time
import qualified Urbit.Vere.Log as Log import qualified Urbit.Vere.Ames as Ames
import qualified Urbit.Vere.Behn as Behn
import qualified Urbit.Vere.Clay as Clay
import qualified Urbit.Vere.Eyre as Eyre
import qualified Urbit.Vere.Http.Client as Iris
import qualified Urbit.Vere.Serf as Serf import qualified Urbit.Vere.Serf as Serf
import qualified Urbit.Vere.Term as Term import qualified Urbit.Vere.Term as Term
import qualified Urbit.Vere.Term.API as Term import qualified Urbit.Vere.Term.API as Term
import qualified Urbit.Vere.Term.Demux as Term import qualified Urbit.Vere.Term.Demux as Term
import qualified Urbit.Vere.Term.Render as Term
-------------------------------------------------------------------------------- -- Initialize pier directory. --------------------------------------------------
_ioDrivers = [] :: [IODriver] data PierDirectoryAlreadyExists = PierDirectoryAlreadyExists
deriving (Show, Exception)
setupPierDirectory :: FilePath -> RIO e () setupPierDirectory :: FilePath -> RIO e ()
setupPierDirectory shipPath = do setupPierDirectory shipPath = do
for_ ["put", "get", "log", "chk"] $ \seg -> do -- shipPath will already exist because we put a lock file there.
let pax = shipPath <> "/.urb/" <> seg alreadyExists <- doesPathExist (shipPath </> ".urb")
createDirectoryIfMissing True pax when alreadyExists $ do
io $ setFileMode pax ownerModes throwIO PierDirectoryAlreadyExists
for_ ["put", "get", "log", "chk"] $ \seg -> do
let pax = shipPath </> ".urb" </> seg
createDirectoryIfMissing True pax
io $ setFileMode pax ownerModes
-- Load pill into boot sequence. ----------------------------------------------- -- Load pill into boot sequence. -----------------------------------------------
genEntropy :: RIO e Word512 genEntropy :: MonadIO m => m Entropy
genEntropy = fromIntegral . bytesAtom <$> io (Ent.getEntropy 64) genEntropy = Entropy . fromIntegral . bytesAtom <$> io (Ent.getEntropy 64)
generateBootSeq :: Ship -> Pill -> Bool -> LegacyBootEvent -> RIO e BootSeq genBootSeq :: MonadIO m => Ship -> Pill -> Bool -> LegacyBootEvent -> m BootSeq
generateBootSeq ship Pill{..} lite boot = do genBootSeq ship Pill {..} lite boot = io $ do
ent <- genEntropy ent <- genEntropy
let ovums = preKern ent <> pKernelOvums <> postKern <> pUserspaceOvums let ovums = preKern ent <> pKernelOvums <> postKern <> pUserspaceOvums
pure $ BootSeq ident pBootFormulas ovums pure $ BootSeq ident pBootFormulas ovums
where where
ident = LogIdentity ship isFake (fromIntegral $ length pBootFormulas) ident = LogIdentity ship isFake (fromIntegral $ length pBootFormulas)
preKern ent = [ EvBlip $ BlipEvArvo $ ArvoEvWhom () ship preKern ent =
, EvBlip $ BlipEvArvo $ ArvoEvWack () ent [ EvBlip $ BlipEvArvo $ ArvoEvWhom () ship
] , EvBlip $ BlipEvArvo $ ArvoEvWack () ent
postKern = [ EvBlip $ BlipEvTerm $ TermEvBoot (1,()) lite boot ] ]
isFake = case boot of postKern = [EvBlip $ BlipEvTerm $ TermEvBoot (1, ()) lite boot]
Fake _ -> True isFake = case boot of
_ -> False Fake _ -> True
_ -> False
-- Write a batch of jobs into the event log ------------------------------------ -- Write to the log. -----------------------------------------------------------
-- | Write a batch of jobs to the event log.
writeJobs :: EventLog -> Vector Job -> RIO e () writeJobs :: EventLog -> Vector Job -> RIO e ()
writeJobs log !jobs = do writeJobs log !jobs = do
expect <- Log.nextEv log expect <- atomically (Log.nextEv log)
events <- fmap fromList $ traverse fromJob (zip [expect..] $ toList jobs) events <- fmap fromList $ traverse fromJob (zip [expect ..] $ toList jobs)
Log.appendEvents log events Log.appendEvents log events
where where
fromJob :: (EventId, Job) -> RIO e ByteString fromJob :: (EventId, Job) -> RIO e ByteString
fromJob (expectedId, job) = do fromJob (expectedId, job) = do
unless (expectedId == jobId job) $ unless (expectedId == jobId job) $ error $ show
error $ show ("bad job id!", expectedId, jobId job) ("bad job id!", expectedId, jobId job)
pure $ jamBS $ jobPayload job pure $ jamBS $ jobPayload job
jobPayload :: Job -> Noun jobPayload :: Job -> Noun
jobPayload (RunNok (LifeCyc _ m n)) = toNoun (m, n) jobPayload (RunNok (LifeCyc _ m n)) = toNoun (m, n)
jobPayload (DoWork (Work _ m d o)) = toNoun (m, d, o) jobPayload (DoWork (Work _ m d o )) = toNoun (m, d, o)
-- Acquire a running serf. -----------------------------------------------------
printTank :: (Text -> IO ()) -> Atom -> Tank -> IO ()
printTank f _priority = f . unlines . fmap unTape . wash (WashCfg 0 80) . tankTree
where
tankTree (Tank t) = t
runSerf
:: HasPierEnv e
=> TVar (Text -> IO ())
-> FilePath
-> RAcquire e Serf
runSerf vSlog pax = do
env <- ask
Serf.withSerf (config env)
where
slog txt = atomically (readTVar vSlog) >>= (\f -> f txt)
config env = Serf.Config
{ scSerf = env ^. pierConfigL . pcSerfExe . to unpack
, scPier = pax
, scFlag = env ^. pierConfigL . pcSerfFlags
, scSlog = \(pri, tank) -> printTank slog pri tank
, scStdr = \txt -> slog (txt <> "\r\n")
, scDead = pure () -- TODO: What can be done?
}
-- Boot a new ship. ------------------------------------------------------------ -- Boot a new ship. ------------------------------------------------------------
booted :: (HasPierConfig e, HasStderrLogFunc e, HasLogFunc e) booted
=> Pill -> Bool -> Serf.Flags -> Ship -> LegacyBootEvent :: TVar (Text -> IO ())
-> RAcquire e (Serf e, EventLog, SerfState) -> Pill
booted pill lite flags ship boot = do -> Bool
seq@(BootSeq ident x y) <- rio $ generateBootSeq ship pill lite boot -> Ship
-> LegacyBootEvent
-> RAcquire PierEnv (Serf, EventLog)
booted vSlog pill lite ship boot = do
rio $ bootNewShip pill lite ship boot
resumed vSlog Nothing
rio $ logTrace "BootSeq Computed" 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 nockJob nocks <> fmap ovumJob ovums
where
nockJob nok eId = RunNok $ LifeCyc eId 0 nok
ovumJob ov eId = DoWork $ Work eId 0 (wen eId) ov
bootNewShip
:: HasPierEnv e
=> Pill
-> Bool
-> Ship
-> LegacyBootEvent
-> RIO e ()
bootNewShip pill lite ship bootEv = do
seq@(BootSeq ident x y) <- genBootSeq ship pill lite bootEv
logDebug "BootSeq Computed"
pierPath <- view pierPathL pierPath <- view pierPathL
liftRIO (setupPierDirectory pierPath) rio (setupPierDirectory pierPath)
logDebug "Directory setup."
rio $ logTrace "Directory Setup" let logPath = (pierPath </> ".urb/log")
log <- Log.new (pierPath <> "/.urb/log") ident rwith (Log.new logPath ident) $ \log -> do
logDebug "Event log onitialized."
jobs <- (\now -> bootSeqJobs now seq) <$> io Time.now
writeJobs log (fromList jobs)
rio $ logTrace "Event Log Initialized" logDebug "Finsihed populating event log with boot sequence"
serf <- Serf.run (Serf.Config pierPath flags)
rio $ logTrace "Serf Started"
rio $ do
(events, serfSt) <- Serf.bootFromSeq serf seq
logTrace "Boot Sequence completed"
Serf.snapshot serf serfSt
logTrace "Snapshot taken"
writeJobs log (fromList events)
logTrace "Events written"
pure (serf, log, serfSt)
-- Resume an existing ship. ---------------------------------------------------- -- Resume an existing ship. ----------------------------------------------------
resumed :: (HasStderrLogFunc e, HasPierConfig e, HasLogFunc e) resumed
=> Maybe Word64 -> Serf.Flags :: TVar (Text -> IO ())
-> RAcquire e (Serf e, EventLog, SerfState) -> Maybe Word64
resumed event flags = do -> RAcquire PierEnv (Serf, EventLog)
rio $ logTrace "Resuming ship" resumed vSlog replayUntil = do
top <- view pierPathL rio $ logTrace "Resuming ship"
tap <- fmap (fromMaybe top) $ rio $ runMaybeT $ do top <- view pierPathL
ev <- MaybeT (pure event) tap <- fmap (fromMaybe top) $ rio $ runMaybeT $ do
MaybeT (getSnapshot top ev) ev <- MaybeT (pure replayUntil)
MaybeT (getSnapshot top ev)
rio $ logTrace $ display @Text ("pier: " <> pack top) rio $ do
rio $ logTrace $ display @Text ("running serf in: " <> pack tap) logTrace $ display @Text ("pier: " <> pack top)
logTrace $ display @Text ("running serf in: " <> pack tap)
log <- Log.existing (top <> "/.urb/log") log <- Log.existing (top </> ".urb/log")
serf <- runSerf vSlog tap
serf <- Serf.run (Serf.Config tap flags) rio $ do
logDebug "Replaying events"
Serf.execReplay serf log replayUntil >>= \case
Left err -> error (show err)
Right 0 -> do
logDebug "No work during replay so no snapshot"
pure ()
Right _ -> do
logDebug "Taking snapshot"
io (Serf.snapshot serf)
logDebug "SNAPSHOT TAKEN"
serfSt <- rio $ Serf.replay serf log event pure (serf, log)
rio $ Serf.snapshot serf serfSt -- | Get a fake pier directory for partial snapshots.
getSnapshot :: forall e . FilePath -> Word64 -> RIO e (Maybe FilePath)
pure (serf, log, serfSt)
getSnapshot :: forall e. FilePath -> Word64 -> RIO e (Maybe FilePath)
getSnapshot top last = do getSnapshot top last = do
lastSnapshot <- lastMay <$> listReplays lastSnapshot <- lastMay <$> listReplays
pure (replayToPath <$> lastSnapshot) pure (replayToPath <$> lastSnapshot)
where where
replayDir = top </> ".partial-replay" replayDir = top </> ".partial-replay"
replayToPath eId = replayDir </> show eId 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)
-- Utils for Spawning Worker Threads -------------------------------------------
acquireWorker :: HasLogFunc e => Text -> RIO e () -> RAcquire e (Async ())
acquireWorker nam act = mkRAcquire (async act) kill
where
kill tid = do
logDebug ("Killing worker thread: " <> display nam)
cancel tid
acquireWorkerBound :: HasLogFunc e => Text -> RIO e () -> RAcquire e (Async ())
acquireWorkerBound nam act = mkRAcquire (asyncBound act) kill
where
kill tid = do
logDebug ("Killing worker thread: " <> display nam)
cancel tid
listReplays :: RIO e [Word64]
listReplays = do
createDirectoryIfMissing True replayDir
snapshotNums <- mapMaybe readMay <$> listDirectory replayDir
pure $ sort (filter (<= fromIntegral last) snapshotNums)
-- Run Pier -------------------------------------------------------------------- -- Run Pier --------------------------------------------------------------------
acquireWorker :: RIO e () -> RAcquire e (Async ()) pier
acquireWorker act = mkRAcquire (async act) cancel :: (Serf, EventLog)
-> TVar (Text -> IO ())
-> MVar ()
-> MultiEyreApi
-> RAcquire PierEnv ()
pier (serf, log) vSlog startedSig multi = do
let logId = Log.identity log :: LogIdentity
let ship = who logId :: Ship
pier :: e. (HasConfigDir e, HasLogFunc e, HasNetworkConfig e, HasPierConfig e) -- TODO Instead of using a TMVar, pull directly from the IO driver
=> (Serf e, EventLog, SerfState) -- event sources.
-> MVar () computeQ :: TMVar RunReq <- newEmptyTMVarIO
-> RAcquire e ()
pier (serf, log, ss) mStart = do
computeQ <- newTQueueIO
persistQ <- newTQueueIO
executeQ <- newTQueueIO
saveM <- newEmptyTMVarIO
shutdownM <- newEmptyTMVarIO
kapi King.kingAPI persistQ :: TQueue (Fact, FX) <- newTQueueIO
executeQ :: TQueue FX <- newTQueueIO
saveSig :: TMVar () <- newEmptyTMVarIO
kingApi :: King.King <- King.kingAPI
termApiQ <- atomically $ do termApiQ :: TQueue TermConn <- atomically $ do
q <- newTQueue q <- newTQueue
writeTVar (King.kTermConn kapi) (Just $ writeTQueue q) writeTVar (King.kTermConn kingApi) (Just $ writeTQueue q)
pure q pure q
let shutdownEvent = putTMVar shutdownM () (demux :: Term.Demux, muxed :: Term.Client) <- atomically $ do
res <- Term.mkDemux
pure (res, Term.useDemux res)
inst <- io (KingId . UV . fromIntegral <$> randomIO @Word16) void $ acquireWorker "TERMSERV Listener" $ forever $ do
logDebug "TERMSERV Waiting for external terminal."
atomically $ do
ext <- Term.connClient <$> readTQueue termApiQ
Term.addDemux ext demux
logDebug "TERMSERV External terminal connected."
-- (sz, local) <- Term.localClient -- Slogs go to both stderr and to the terminal.
env <- ask
atomically $ writeTVar vSlog $ \txt -> runRIO env $ do
atomically $ Term.trace muxed txt
logOther "serf" (display $ T.strip txt)
-- (waitExternalTerm, termServPort) <- Term.termServer -- 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 compute = putTMVar computeQ
let execute = writeTQueue executeQ
let persist = writeTQueue persistQ
let sigint = Serf.sendSIGINT serf
(demux, muxed) <- atomically $ do (bootEvents, startDrivers) <- do
res <- Term.mkDemux env <- ask
-- Term.addDemux local res let err = atomically . Term.trace muxed . (<> "\r\n")
pure (res, Term.useDemux res) let siz = TermSize { tsWide = 80, tsTall = 24 }
let fak = isFake logId
drivers env multi ship fak compute (siz, muxed) err sigint
-- rio $ logInfo $ display $ scrySig <- newEmptyTMVarIO
-- "TERMSERV Terminal Server running on port: " <> tshow termServPort onKill <- view onKillPierSigL
acquireWorker $ forever $ do let computeConfig = ComputeConfig { ccOnWork = takeTMVar computeQ
logTrace "TERMSERV Waiting for external terminal." , ccOnKill = onKill
atomically $ do , ccOnSave = takeTMVar saveSig
ext <- Term.connClient <$> readTQueue termApiQ , ccOnScry = takeTMVar scrySig
Term.addDemux ext demux , ccPutResult = persist
logTrace "TERMSERV External terminal connected." , ccShowSpinner = Term.spin muxed
, ccHideSpinner = Term.stopSpin muxed
, ccLastEvInLog = Log.lastEv log
}
swapMVar (sStderr serf) (atomically . Term.trace muxed) tSerf <- acquireWorker "Serf" (runCompute serf computeConfig)
let logId = Log.identity log -- Run all born events and retry them until they succeed.
let ship = who logId wackEv <- EvBlip . BlipEvArvo . ArvoEvWack () <$> genEntropy
rio $ for_ (wackEv : bootEvents) $ \ev -> do
okaySig <- newEmptyMVar
-- Our call above to set the logging function which echos errors from the let inject n = atomically $ compute $ RRWork $ EvErr ev $ cb n
-- 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) -- TODO Make sure this dies cleanly.
cb :: Int -> WorkError -> IO ()
cb n | n >= 3 = error ("boot event failed: " <> show ev)
cb n = \case
RunOkay _ -> putMVar okaySig ()
RunSwap _ _ _ _ _ -> putMVar okaySig ()
RunBail _ -> inject (n + 1)
tExe <- startDrivers >>= router (readTQueue executeQ) -- logTrace ("[BOOT EVENT]: " <> display (summarizeEvent ev))
tDisk <- runPersist log persistQ (writeTQueue executeQ) io (inject 0)
tCpu <- runCompute serf ss
(readTQueue computeQ)
(takeTMVar saveM)
(takeTMVar shutdownM)
(Term.spin muxed)
(Term.stopSpin muxed)
(writeTQueue persistQ)
tSaveSignal <- saveSignalThread saveM let slog :: Text -> IO ()
slog txt = do
fn <- atomically (readTVar vSlog)
fn txt
putMVar mStart () drivz <- startDrivers
tExec <- acquireWorker "Effects" (router slog (readTQueue executeQ) drivz)
tDisk <- acquireWorkerBound "Persist" (runPersist log persistQ execute)
-- Wait for something to die. let snapshotEverySecs = 120
let ded = asum [ death "effect thread" tExe void $ acquireWorker "Save" $ forever $ do
, death "persist thread" tDisk threadDelay (snapshotEverySecs * 1_000_000)
, death "compute thread" tCpu void $ atomically $ tryPutTMVar saveSig ()
]
atomically ded >>= \case -- TODO bullshit scry tester
Left (txt, exn) -> logError $ displayShow ("Somthing died", txt, exn) when False $ do
Right tag -> logError $ displayShow ("something simply exited", tag) void $ acquireWorker "bullshit scry tester" $ do
env <- ask
forever $ do
threadDelay 15_000_000
wen <- io Time.now
let kal = \mTermNoun -> runRIO env $ do
logDebug $ displayShow ("scry result: ", mTermNoun)
let nkt = MkKnot $ tshow $ Time.MkDate wen
let pax = Path ["j", "~zod", "life", nkt, "~zod"]
atomically $ putTMVar scrySig (wen, Nothing, pax, kal)
atomically $ (Term.spin muxed) (Just "shutdown") putMVar startedSig ()
-- Wait for something to die.
let ded = asum
[ death "effects thread" tExec
, death "persist thread" tDisk
, death "compute thread" tSerf
]
atomically ded >>= \case
Left (tag, exn) -> logError $ displayShow (tag, "crashed", exn)
Right "compute thread" -> pure ()
Right tag -> logError $ displayShow (tag, "exited unexpectly")
atomically $ (Term.spin muxed) (Just "shutdown")
death :: Text -> Async () -> STM (Either (Text, SomeException) Text) death :: Text -> Async () -> STM (Either (Text, SomeException) Text)
death tag tid = do death tag tid = do
waitCatchSTM tid <&> \case waitCatchSTM tid <&> \case
Left exn -> Left (tag, exn) Left exn -> Left (tag, exn)
Right () -> Right tag 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 ----------------------------------------------------------- -- Start All Drivers -----------------------------------------------------------
data Drivers e = Drivers data Drivers = Drivers
{ dAmes :: EffCb e AmesEf { dBehn :: BehnEf -> IO ()
, dBehn :: EffCb e BehnEf , dIris :: HttpClientEf -> IO ()
, dHttpClient :: EffCb e HttpClientEf , dEyre :: HttpServerEf -> IO ()
, dHttpServer :: EffCb e HttpServerEf , dNewt :: NewtEf -> IO ()
, dNewt :: EffCb e NewtEf , dSync :: SyncEf -> IO ()
, dSync :: EffCb e SyncEf , dTerm :: TermEf -> IO ()
, dTerm :: EffCb e TermEf }
}
drivers :: (HasLogFunc e, HasNetworkConfig e, HasPierConfig e) drivers
=> KingId -> Ship -> Bool -> (Ev -> STM ()) :: HasPierEnv e
-> STM() => e
-> (Term.TSize, Term.Client) -> MultiEyreApi
-> (Text -> RIO e ()) -> Ship
-> ([Ev], RAcquire e (Drivers e)) -> Bool
drivers inst who isFake plan shutdownSTM termSys stderr = -> (RunReq -> STM ())
(initialEvents, runDrivers) -> (TermSize, Term.Client)
where -> (Text -> RIO e ())
(behnBorn, runBehn) = behn inst plan -> IO ()
(amesBorn, runAmes) = ames inst who isFake plan stderr -> RAcquire e ([Ev], RAcquire e Drivers)
(httpBorn, runHttp) = serv inst plan isFake drivers env multi who isFake plan termSys stderr serfSIGINT = do
(clayBorn, runClay) = clay inst plan (behnBorn, runBehn) <- rio Behn.behn'
(irisBorn, runIris) = client inst plan (termBorn, runTerm) <- rio (Term.term' termSys serfSIGINT)
(termBorn, runTerm) = Term.term termSys shutdownSTM inst plan (amesBorn, runAmes) <- rio (Ames.ames' who isFake stderr)
initialEvents = mconcat [behnBorn, clayBorn, amesBorn, httpBorn, (httpBorn, runEyre) <- rio (Eyre.eyre' multi who isFake)
termBorn, irisBorn] (clayBorn, runClay) <- rio Clay.clay'
runDrivers = do (irisBorn, runIris) <- rio Iris.client'
dNewt <- runAmes
dBehn <- liftAcquire $ runBehn let initialEvents = mconcat [behnBorn,clayBorn,amesBorn,httpBorn,irisBorn,termBorn]
dAmes <- pure $ const $ pure ()
dHttpClient <- runIris let runDrivers = do
dHttpServer <- runHttp behn <- runBehn
dSync <- runClay term <- runTerm
dTerm <- runTerm ames <- runAmes
pure (Drivers{..}) iris <- runIris
eyre <- runEyre
clay <- runClay
-- Sources lower in the list are starved until sources above them
-- have no events to offer.
acquireWorker "Event Prioritization" $ forever $ atomically $ do
let x = diEventSource
let eventSources = [x term, x clay, x behn, x iris, x eyre, x ames]
pullEvent eventSources >>= \case
Nothing -> retry
Just rr -> plan rr
pure $ Drivers
{ dTerm = diOnEffect term
, dBehn = diOnEffect behn
, dNewt = diOnEffect ames
, dIris = diOnEffect iris
, dEyre = diOnEffect eyre
, dSync = diOnEffect clay
}
pure (initialEvents, runDrivers)
where
pullEvent :: [STM (Maybe a)] -> STM (Maybe a)
pullEvent [] = pure Nothing
pullEvent (d:ds) = d >>= \case
Just r -> pure (Just r)
Nothing -> pullEvent ds
-- Route Effects to Drivers ---------------------------------------------------- -- Route Effects to Drivers ----------------------------------------------------
router :: HasLogFunc e => STM FX -> Drivers e -> RAcquire e (Async ()) router :: HasPierEnv e => (Text -> IO ()) -> STM FX -> Drivers -> RIO e ()
router waitFx Drivers{..} = router slog waitFx Drivers {..} = do
mkRAcquire start cancel kill <- view killPierActionL
where let exit = io (slog "<<<shutdown>>>\r\n") >> atomically kill
start = async $ forever $ do let vega = io (slog "<<<reset>>>\r\n")
fx <- atomically waitFx forever $ do
for_ fx $ \ef -> do fx <- atomically waitFx
logEffect ef for_ fx $ \ef -> do
case ef of logEffect ef
GoodParse (EfVega _ _) -> error "TODO" case ef of
GoodParse (EfExit _ _) -> error "TODO" GoodParse (EfVega _ _ ) -> vega
GoodParse (EfVane (VEAmes ef)) -> dAmes ef GoodParse (EfExit _ _ ) -> exit
GoodParse (EfVane (VEBehn ef)) -> dBehn ef GoodParse (EfVane (VEBehn ef)) -> io (dBehn ef)
GoodParse (EfVane (VEBoat ef)) -> dSync ef GoodParse (EfVane (VEBoat ef)) -> io (dSync ef)
GoodParse (EfVane (VEClay ef)) -> dSync ef GoodParse (EfVane (VEClay ef)) -> io (dSync ef)
GoodParse (EfVane (VEHttpClient ef)) -> dHttpClient ef GoodParse (EfVane (VEHttpClient ef)) -> io (dIris ef)
GoodParse (EfVane (VEHttpServer ef)) -> dHttpServer ef GoodParse (EfVane (VEHttpServer ef)) -> io (dEyre ef)
GoodParse (EfVane (VENewt ef)) -> dNewt ef GoodParse (EfVane (VENewt ef)) -> io (dNewt ef)
GoodParse (EfVane (VESync ef)) -> dSync ef GoodParse (EfVane (VESync ef)) -> io (dSync ef)
GoodParse (EfVane (VETerm ef)) -> dTerm ef GoodParse (EfVane (VETerm ef)) -> io (dTerm ef)
FailParse n -> logError FailParse n -> logError $ display $ pack @Text (ppShow n)
$ display
$ pack @Text (ppShow n)
-- Compute Thread -------------------------------------------------------------- -- Compute (Serf) Thread -------------------------------------------------------
data ComputeRequest
= CREvent Ev
| CRSave ()
| CRShutdown ()
deriving (Eq, Show)
logEvent :: HasLogFunc e => Ev -> RIO e () logEvent :: HasLogFunc e => Ev -> RIO e ()
logEvent ev = logEvent ev = do
logDebug $ display $ "[EVENT]\n" <> pretty logTrace $ "<- " <> display (summarizeEvent ev)
where logDebug $ "[EVENT]\n" <> display pretty
pretty :: Text where
pretty = pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow ev pretty :: Text
pretty = pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow ev
logEffect :: HasLogFunc e => Lenient Ef -> RIO e () logEffect :: HasLogFunc e => Lenient Ef -> RIO e ()
logEffect ef = logEffect ef = do
logDebug $ display $ "[EFFECT]\n" <> pretty ef logTrace $ " -> " <> display (summarizeEffect ef)
where logDebug $ display $ "[EFFECT]\n" <> pretty ef
pretty :: Lenient Ef -> Text where
pretty = \case pretty :: Lenient Ef -> Text
GoodParse e -> pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow e pretty = \case
FailParse n -> pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow n GoodParse e -> pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow e
FailParse n -> pack $ unlines $ fmap ("\t" <>) $ lines $ ppShow n
runCompute :: e. HasLogFunc e data ComputeConfig = ComputeConfig
=> Serf e { ccOnWork :: STM RunReq
-> SerfState , ccOnKill :: STM ()
-> STM Ev , ccOnSave :: STM ()
-> STM () , ccOnScry :: STM (Wen, Gang, Path, Maybe (Term, Noun) -> IO ())
-> STM () , ccPutResult :: (Fact, FX) -> STM ()
-> (Maybe Text -> STM ()) , ccShowSpinner :: Maybe Text -> STM ()
-> STM () , ccHideSpinner :: STM ()
-> ((Job, FX) -> STM ()) , ccLastEvInLog :: STM EventId
-> RAcquire e (Async ()) }
runCompute serf ss getEvent getSaveSignal getShutdownSignal
showSpinner hideSpinner putResult =
mkRAcquire (async (go ss)) cancel
where
go :: SerfState -> RIO e ()
go ss = do
cr <- atomically $
CRShutdown <$> getShutdownSignal <|>
CRSave <$> getSaveSignal <|>
CREvent <$> getEvent
case cr of
CREvent ev -> do
logEvent ev
wen <- io Time.now
eId <- pure (ssNextEv ss)
mug <- pure (ssLastMug ss)
atomically $ showSpinner (getSpinnerNameForEvent ev) runCompute :: forall e . HasKingEnv e => Serf.Serf -> ComputeConfig -> RIO e ()
(job', ss', fx) <- doJob serf $ DoWork $ Work eId mug wen ev runCompute serf ComputeConfig {..} = do
atomically $ hideSpinner logDebug "runCompute"
atomically (putResult (job', fx))
go ss' let onRR = asum [ ccOnKill <&> Serf.RRKill
CRSave () -> do , ccOnSave <&> Serf.RRSave
logDebug $ "Taking periodic snapshot" , ccOnWork
Serf.snapshot serf ss , ccOnScry <&> \(w,g,p,k) -> Serf.RRScry w g p k
go ss ]
CRShutdown () -> do
-- When shutting down, we first request a snapshot, and then we vEvProcessing :: TMVar Ev <- newEmptyTMVarIO
-- just exit this recursive processing, which will cause the serf
-- to exit from its RAcquire. void $ async $ forever (atomically (takeTMVar vEvProcessing) >>= logEvent)
logDebug $ "Shutting down compute system..."
Serf.snapshot serf ss let onSpin :: Maybe Ev -> STM ()
pure () onSpin = \case
Nothing -> ccHideSpinner
Just ev -> do
ccShowSpinner (getSpinnerNameForEvent ev)
putTMVar vEvProcessing ev
let maxBatchSize = 10
io (Serf.run serf maxBatchSize ccLastEvInLog onRR ccPutResult onSpin)
-- Persist Thread -------------------------------------------------------------- -- Event-Log Persistence Thread ------------------------------------------------
data PersistExn = BadEventId EventId EventId data PersistExn = BadEventId EventId EventId
deriving Show deriving Show
@ -418,43 +558,36 @@ instance Exception PersistExn where
, "\tExpected " <> show expected <> " but got " <> show got , "\tExpected " <> show expected <> " but got " <> show got
] ]
runPersist :: e. (HasPierConfig e, HasLogFunc e) runPersist
=> EventLog :: forall e
-> TQueue (Job, FX) . HasPierEnv e
-> (FX -> STM ()) => EventLog
-> RAcquire e (Async ()) -> TQueue (Fact, FX)
runPersist log inpQ out = -> (FX -> STM ())
mkRAcquire runThread cancel -> RIO e ()
where runPersist log inpQ out = do
runThread :: RIO e (Async ()) dryRun <- view dryRunL
runThread = asyncBound $ do forever $ do
dryRun <- view dryRunL writs <- atomically getBatchFromQueue
forever $ do events <- validateFactsAndGetBytes (fst <$> toNullable writs)
writs <- atomically getBatchFromQueue unless dryRun (Log.appendEvents log events)
unless dryRun $ do atomically $ for_ writs $ \(_, fx) -> do
events <- validateJobsAndGetBytes (toNullable writs) out fx
Log.appendEvents log events
atomically $ for_ writs $ \(_,fx) -> out fx
validateJobsAndGetBytes :: [(Job, FX)] -> RIO e (Vector ByteString) where
validateJobsAndGetBytes writs = do validateFactsAndGetBytes :: [Fact] -> RIO e (Vector ByteString)
expect <- Log.nextEv log validateFactsAndGetBytes facts = do
fmap fromList expect <- atomically (Log.nextEv log)
$ for (zip [expect..] writs) lis <- for (zip [expect ..] facts) $ \(expectedId, Fact eve mug wen non) ->
$ \(expectedId, (j, fx)) -> do do
unless (expectedId == jobId j) $ unless (expectedId == eve) $ do
throwIO (BadEventId expectedId (jobId j)) throwIO (BadEventId expectedId eve)
case j of pure $ jamBS $ toNoun (mug, wen, non)
RunNok _ -> pure (fromList lis)
error "This shouldn't happen here!"
DoWork (Work eId mug wen ev) ->
pure $ jamBS $ toNoun (mug, wen, ev)
getBatchFromQueue :: STM (NonNull [(Job, FX)]) getBatchFromQueue :: STM (NonNull [(Fact, FX)])
getBatchFromQueue = getBatchFromQueue = readTQueue inpQ >>= go . singleton
readTQueue inpQ >>= go . singleton where
where go acc = tryReadTQueue inpQ >>= \case
go acc = Nothing -> pure (reverse acc)
tryReadTQueue inpQ >>= \case Just item -> go (item <| acc)
Nothing -> pure (reverse acc)
Just item -> go (item <| acc)

View File

@ -3,12 +3,27 @@
TODO Most of these could probably find better homes. TODO Most of these could probably find better homes.
-} -}
module Urbit.Vere.Pier.Types where module Urbit.Vere.Pier.Types
( module Urbit.Vere.Serf.Types
, LogIdentity(..)
, Pill(..)
, Job(..)
, LifeCyc(..)
, BootSeq(..)
, Work(..)
, jobId
, jobMug
, DriverApi(..)
)
where
import Urbit.Prelude hiding (Term) import Urbit.Prelude hiding (Term)
import Urbit.Arvo import Urbit.Arvo
import Urbit.Time import Urbit.Noun.Time
import Urbit.Vere.Serf.Types
import Urbit.EventLog.LMDB (LogIdentity(..))
-- Avoid touching Nock values. ------------------------------------------------- -- Avoid touching Nock values. -------------------------------------------------
@ -29,25 +44,16 @@ instance Show Nock where
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
type EventId = Word64
data Pill = Pill data Pill = Pill
{ pBootFormulas :: [Nock] { pBootFormulas :: [Nock]
, pKernelOvums :: [Ev] , pKernelOvums :: [Ev]
, pUserspaceOvums :: [Ev] , pUserspaceOvums :: [Ev]
} }
deriving (Eq, Show) deriving (Eq, Show)
data LogIdentity = LogIdentity
{ who :: Ship
, isFake :: Bool
, lifecycleLen :: Word
} deriving (Eq, Ord, Show)
data BootSeq = BootSeq LogIdentity [Nock] [Ev] data BootSeq = BootSeq LogIdentity [Nock] [Ev]
deriving (Eq, Show) deriving (Eq, Show)
deriveNoun ''LogIdentity
deriveNoun ''Pill deriveNoun ''Pill
@ -60,40 +66,25 @@ data LifeCyc = LifeCyc EventId Mug Nock
deriving (Eq, Show) deriving (Eq, Show)
data Job data Job
= DoWork Work = DoWork Work
| RunNok LifeCyc | RunNok LifeCyc
deriving (Eq, Show) deriving (Eq, Show)
jobId :: Job -> EventId jobId :: Job -> EventId
jobId (RunNok (LifeCyc eId _ _)) = eId jobId (RunNok (LifeCyc eId _ _)) = eId
jobId (DoWork (Work eId _ _ _)) = eId jobId (DoWork (Work eId _ _ _ )) = eId
jobMug :: Job -> Mug jobMug :: Job -> Mug
jobMug (RunNok (LifeCyc _ mug _)) = mug jobMug (RunNok (LifeCyc _ mug _)) = mug
jobMug (DoWork (Work _ mug _ _)) = mug jobMug (DoWork (Work _ mug _ _ )) = mug
-------------------------------------------------------------------------------- -- API To IO Drivers -----------------------------------------------------------
data Order data DriverApi ef = DriverApi
= OBoot Word -- lifecycle length { diEventSource :: STM (Maybe RunReq)
| OExit Word8 , diOnEffect :: ef -> IO ()
| OSave EventId }
| OWork Job
deriving (Eq, Show)
deriveToNoun ''Order
type QueueEv = Ev -> STM ()
type EffCb e a = a -> RIO e ()
type Perform = Ef -> IO ()
data IODriver = IODriver
{ bornEvent :: IO Ev
, startDriver :: (Ev -> STM ()) -> IO (Async (), Perform)
}
-- Instances ------------------------------------------------------------------- -- Instances -------------------------------------------------------------------
@ -102,17 +93,17 @@ instance ToNoun Work where
toNoun (Work eid m d o) = toNoun (eid, Jammed (m, d, o)) toNoun (Work eid m d o) = toNoun (eid, Jammed (m, d, o))
instance FromNoun Work where instance FromNoun Work where
parseNoun n = named "Work" $ do parseNoun n = named "Work" $ do
(eid, Jammed (m, d, o)) <- parseNoun n (eid, Jammed (m, d, o)) <- parseNoun n
pure (Work eid m d o) pure (Work eid m d o)
instance ToNoun LifeCyc where instance ToNoun LifeCyc where
toNoun (LifeCyc eid m n) = toNoun (eid, Jammed (m, n)) toNoun (LifeCyc eid m n) = toNoun (eid, Jammed (m, n))
instance FromNoun LifeCyc where instance FromNoun LifeCyc where
parseNoun n = named "LifeCyc" $ do parseNoun n = named "LifeCyc" $ do
(eid, Jammed (m, n)) <- parseNoun n (eid, Jammed (m, n)) <- parseNoun n
pure (LifeCyc eid m n) pure (LifeCyc eid m n)
-- | No FromNoun instance, because it depends on context (lifecycle length) -- | No FromNoun instance, because it depends on context (lifecycle length)
instance ToNoun Job where instance ToNoun Job where

View File

@ -1,547 +1,162 @@
{-| {-|
Serf Interface High-Level Serf Interface
TODO: `recvLen` is not big-endian safe.
-} -}
module Urbit.Vere.Serf ( Serf, sStderr, SerfState(..), doJob module Urbit.Vere.Serf
, run, shutdown, kill ( withSerf
, replay, bootFromSeq, snapshot , execReplay
, collectFX , collectFX
, Config(..), Flags, Flag(..) , module X
) where )
where
import Urbit.Prelude import Urbit.Prelude
import Data.Conduit import Data.Conduit
import System.Process
import System.ProgressBar
import Urbit.Arvo
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Urbit.Vere.Serf.IPC
import Data.Bits (setBit) import Control.Monad.Trans.Resource (runResourceT)
import Data.ByteString (hGet) import Urbit.Arvo (FX)
import Data.ByteString.Unsafe (unsafeUseAsCString) import Urbit.King.App.Class (HasStderrLogFunc(..))
import Foreign.Marshal.Alloc (alloca)
import Foreign.Ptr (castPtr)
import Foreign.Storable (peek, poke)
import System.Exit (ExitCode)
import Urbit.King.App (HasStderrLogFunc(..))
import qualified Data.ByteString.Unsafe as BS
import qualified Data.Conduit.Combinators as CC import qualified Data.Conduit.Combinators as CC
import qualified Data.Text as T import qualified System.ProgressBar as PB
import qualified System.IO as IO import qualified Urbit.EventLog.LMDB as Log
import qualified System.IO.Error as IO
import qualified Urbit.Ob as Ob
import qualified Urbit.Time as Time
import qualified Urbit.Vere.Log as Log
import qualified Urbit.Vere.Serf.IPC as X (Config (..), EvErr (..), Flag (..),
-- Serf Config ----------------------------------------------------------------- RunReq (..), Serf, WorkError (..),
run, sendSIGINT, snapshot, start,
type Flags = [Flag] stop)
data Flag
= DebugRam
| DebugCpu
| CheckCorrupt
| CheckFatal
| Verbose
| DryRun
| Quiet
| Hashless
| Trace
deriving (Eq, Ord, Show, Enum, Bounded)
compileFlags :: [Flag] -> Word
compileFlags = foldl' (\acc flag -> setBit acc (fromEnum flag)) 0
data Config = Config FilePath [Flag]
deriving (Show)
serf :: HasLogFunc e => Text -> RIO e ()
serf msg = logInfo $ display ("SERF: " <> msg)
-- Types -----------------------------------------------------------------------
data SerfState = SerfState
{ ssNextEv :: EventId
, ssLastMug :: Mug
}
deriving (Eq, Ord, Show)
ssLastEv :: SerfState -> EventId
ssLastEv = pred . ssNextEv
data Serf e = Serf
{ sendHandle :: Handle
, recvHandle :: Handle
, process :: ProcessHandle
, sStderr :: MVar (Text -> RIO e ())
}
data ShipId = ShipId Ship Bool
deriving (Eq, Ord, Show)
data Plea
= PPlay EventId Mug
| PWork Work
| PDone EventId Mug FX
| PStdr EventId Cord
| PSlog EventId Word32 Tank
deriving (Eq, Show)
type ReplacementEv = Job
type WorkResult = (SerfState, FX)
type SerfResp = Either ReplacementEv WorkResult
data SerfExn
= BadComputeId EventId WorkResult
| BadReplacementId EventId ReplacementEv
| UnexpectedPlay EventId (EventId, Mug)
| BadPleaAtom Atom
| BadPleaNoun Noun [Text] Text
| ReplacedEventDuringReplay EventId ReplacementEv
| ReplacedEventDuringBoot EventId ReplacementEv
| EffectsDuringBoot EventId FX
| SerfConnectionClosed
| UnexpectedPleaOnNewShip Plea
| InvalidInitialPlea Plea
deriving (Show)
-- Instances -------------------------------------------------------------------
instance Exception SerfExn
deriveNoun ''ShipId
deriveNoun ''Plea
-- Utils -----------------------------------------------------------------------
printTank :: HasLogFunc e
=> MVar (Text -> RIO e ()) -> Word32 -> Tank
-> RIO e ()
printTank log _pri = printErr log . unlines . fmap unTape . wash (WashCfg 0 80)
guardExn :: (Exception e, MonadIO m) => Bool -> e -> m ()
guardExn ok = io . unless ok . throwIO
fromRightExn :: (Exception e, MonadIO m) => Either a b -> (a -> e) -> m b
fromRightExn (Left m) exn = throwIO (exn m)
fromRightExn (Right x) _ = pure x
printErr :: MVar (Text -> RIO e ()) -> Text -> RIO e ()
printErr m txt = do
f <- readMVar m
f txt
-- Process Management ----------------------------------------------------------
run :: HasLogFunc e => Config -> RAcquire e (Serf e)
run config = mkRAcquire (startUp config) tearDown
startUp :: HasLogFunc e => Config -> RIO e (Serf e)
startUp conf@(Config pierPath flags) = do
logTrace "STARTING SERF"
logTrace (displayShow conf)
(i, o, e, p) <- io $ do
(Just i, Just o, Just e, p) <- createProcess pSpec
pure (i, o, e, p)
stderr <- newMVar serf
async (readStdErr e stderr)
pure (Serf i o p stderr)
where
diskKey = ""
config = show (compileFlags flags)
args = [pierPath, diskKey, config]
pSpec = (proc "urbit-worker" args)
{ std_in = CreatePipe
, std_out = CreatePipe
, std_err = CreatePipe
}
readStdErr :: e. HasLogFunc e => Handle -> MVar (Text -> RIO e ()) -> RIO e ()
readStdErr h print =
untilEOFExn $ do
raw <- io $ IO.hGetLine h
let ln = T.strip (pack raw)
printErr print ln
serf ("[stderr] " <> ln)
where
eofMsg = "[Serf.readStdErr] serf stderr closed"
untilEOFExn :: RIO e () -> RIO e ()
untilEOFExn act = loop
where
loop :: RIO e ()
loop = do
env <- ask
res <- io $ IO.tryIOError $ runRIO env act
case res of
Left exn | IO.isEOFError exn -> logDebug eofMsg
Left exn -> io (IO.ioError exn)
Right () -> loop
tearDown :: HasLogFunc e => Serf e -> RIO e ()
tearDown serf = do
io $ terminateProcess (process serf)
void $ waitForExit serf
-- race_ waitThenKill (shutdownAndWait serf 0)
where
-- killedMsg =
-- "[Serf.tearDown]: Serf didn't die when asked, killing it"
-- waitThenKill = do
-- threadDelay 1000000
-- debug killedMsg
-- terminateProcess (process serf)
waitForExit :: HasLogFunc e => Serf e -> RIO e ExitCode
waitForExit = io . waitForProcess . process
kill :: HasLogFunc e => Serf e -> RIO e ExitCode
kill serf = io (terminateProcess $ process serf) >> waitForExit serf
_shutdownAndWait :: HasLogFunc e => Serf e -> Word8 -> RIO e ExitCode
_shutdownAndWait serf code = do
shutdown serf code
waitForExit serf
-- Basic Send and Receive Operations -------------------------------------------
withWord64AsByteString :: Word64 -> (ByteString -> RIO e a) -> RIO e a
withWord64AsByteString w k = do
env <- ask
io $ alloca $ \wp -> do
poke wp w
bs <- BS.unsafePackCStringLen (castPtr wp, 8)
runRIO env (k bs)
sendLen :: HasLogFunc e => Serf e -> Int -> RIO e ()
sendLen s i = do
w <- evaluate (fromIntegral i :: Word64)
withWord64AsByteString (fromIntegral i) (hPut (sendHandle s))
sendOrder :: HasLogFunc e => Serf e -> Order -> RIO e ()
sendOrder w o = do
-- logDebug $ display ("(sendOrder) " <> tshow o)
sendBytes w $ jamBS $ toNoun o
-- logDebug "(sendOrder) Done"
sendBytes :: HasLogFunc e => Serf e -> ByteString -> RIO e ()
sendBytes s bs = handle ioErr $ do
sendLen s (length bs)
hPut (sendHandle s) bs
hFlush (sendHandle s)
where
ioErr :: IOError -> RIO e ()
ioErr _ = throwIO SerfConnectionClosed
recvLen :: (MonadIO m, HasLogFunc e) => Serf e -> m Word64
recvLen w = io $ do
bs <- hGet (recvHandle w) 8
case length bs of
8 -> unsafeUseAsCString bs (peek . castPtr)
_ -> throwIO SerfConnectionClosed
recvBytes :: HasLogFunc e => Serf e -> Word64 -> RIO e ByteString
recvBytes serf =
io . hGet (recvHandle serf) . fromIntegral
recvAtom :: HasLogFunc e => Serf e -> RIO e Atom
recvAtom w = do
len <- recvLen w
bytesAtom <$> recvBytes w len
cordText :: Cord -> Text
cordText = T.strip . unCord
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
snapshot :: HasLogFunc e => Serf e -> SerfState -> RIO e () parseLogRow :: MonadIO m => ByteString -> m (Mug, Noun)
snapshot serf ss = do parseLogRow = cueBSExn >=> fromNounExn
logTrace $ display ("Taking snapshot at event " <> tshow (ssLastEv ss))
sendOrder serf $ OSave $ ssLastEv ss
shutdown :: HasLogFunc e => Serf e -> Word8 -> RIO e () withSerf :: HasLogFunc e => Config -> RAcquire e Serf
shutdown serf code = sendOrder serf (OExit code) withSerf config = mkRAcquire startup kill
where
startup = do
(serf, st) <- io $ start config
logDebug (displayShow ("serf state", st))
pure serf
kill serf = do
void $ rio $ stop serf
{-| execReplay
TODO Find a cleaner way to handle `PStdr` Pleas. :: forall e
-} . (HasLogFunc e, HasStderrLogFunc e)
recvPlea :: HasLogFunc e => Serf e -> RIO e Plea => Serf
recvPlea w = do -> Log.EventLog
logDebug "(recvPlea) Waiting" -> Maybe Word64
a <- recvAtom w -> RIO e (Either PlayBail Word)
logDebug "(recvPlea) Got atom" execReplay serf log last = do
n <- fromRightExn (cue a) (const $ BadPleaAtom a) lastEventInSnap <- io (serfLastEventBlocking serf)
p <- fromRightExn (fromNounErr n) (\(p,m) -> BadPleaNoun n p m) if lastEventInSnap == 0 then doBoot else doReplay
where
doBoot :: RIO e (Either PlayBail Word)
doBoot = do
logDebug "Beginning boot sequence"
case p of PStdr e msg -> do printErr (sStderr w) (cordText msg) let bootSeqLen = lifecycleLen (Log.identity log)
recvPlea w
PSlog _ pri t -> do printTank (sStderr w) pri t
recvPlea w
_ -> do logTrace "recvPlea got something else"
pure p
{-| evs <- runConduit $ Log.streamEvents log 1
Waits for initial plea, and then sends boot IPC if necessary. .| CC.take (fromIntegral bootSeqLen)
-} .| CC.mapM (fmap snd . parseLogRow)
handshake :: HasLogFunc e => Serf e -> LogIdentity -> RIO e SerfState .| CC.sinkList
handshake serf ident = do
logTrace "Serf Handshake"
ss@SerfState{..} <- recvPlea serf >>= \case let numEvs = fromIntegral (length evs)
PPlay e m -> pure $ SerfState e m
x -> throwIO (InvalidInitialPlea x)
logTrace $ display ("Handshake result: " <> tshow ss) when (numEvs /= bootSeqLen) $ do
throwIO (MissingBootEventsInEventLog numEvs bootSeqLen)
when (ssNextEv == 1) $ do logDebug $ display ("Sending " <> tshow numEvs <> " boot events to serf")
let ev = OBoot (lifecycleLen ident)
logTrace $ display ("No snapshot. Sending boot event: " <> tshow ev)
sendOrder serf ev
logTrace "Finished handshake" io (boot serf evs) >>= \case
Just err -> do
logDebug "Error on replay, exiting"
pure (Left err)
Nothing -> do
logDebug "Finished boot events, moving on to more events from log."
doReplay <&> \case
Left err -> Left err
Right num -> Right (num + numEvs)
pure ss doReplay :: RIO e (Either PlayBail Word)
doReplay = do
sendWork :: e. HasLogFunc e => Serf e -> Job -> RIO e SerfResp
sendWork w job =
do
sendOrder w (OWork job)
res <- loop
logTrace ("[sendWork] Got response")
pure res
where
eId = jobId job
produce :: WorkResult -> RIO e SerfResp
produce (ss@SerfState{..}, o) = do
guardExn (ssNextEv == (1+eId)) (BadComputeId eId (ss, o))
pure $ Right (ss, o)
replace :: ReplacementEv -> RIO e SerfResp
replace job = do
guardExn (jobId job == eId) (BadReplacementId eId job)
pure (Left job)
loop :: RIO e SerfResp
loop = recvPlea w >>= \case
PPlay e m -> throwIO (UnexpectedPlay eId (e, m))
PDone i m o -> produce (SerfState (i+1) m, o)
PWork work -> replace (DoWork work)
PStdr _ cord -> printErr (sStderr w) (cordText cord) >> loop
PSlog _ pri t -> printTank (sStderr w) pri t >> loop
--------------------------------------------------------------------------------
doJob :: HasLogFunc e => Serf e -> Job -> RIO e (Job, SerfState, FX)
doJob serf job = do
sendWork serf job >>= \case
Left replaced -> doJob serf replaced
Right (ss, fx) -> pure (job, ss, fx)
bootJob :: HasLogFunc e => Serf e -> Job -> RIO e (Job, SerfState)
bootJob serf job = do
doJob serf job >>= \case
(job, ss, _) -> pure (job, ss)
-- (job, ss, fx) -> throwIO (EffectsDuringBoot (jobId job) fx)
replayJob :: HasLogFunc e => Serf e -> Job -> RIO e SerfState
replayJob serf job = do
sendWork serf job >>= \case
Left replace -> throwIO (ReplacedEventDuringReplay (jobId job) replace)
Right (ss, _) -> pure ss
--------------------------------------------------------------------------------
updateProgressBar :: HasLogFunc e
=> Int -> Text -> Maybe (ProgressBar ())
-> RIO e (Maybe (ProgressBar ()))
updateProgressBar count startMsg = \case
Nothing -> do
-- We only construct the progress bar on the first time that we
-- process an event so that we don't display an empty progress
-- bar when the snapshot is caught up to the log.
let style = defStyle { stylePrefix = msg (fromStrict startMsg) }
pb <- newProgressBar style 10 (Progress 0 count ())
pure (Just pb)
Just pb -> do
incProgress pb 1
pure (Just pb)
--------------------------------------------------------------------------------
type BootSeqFn = EventId -> Mug -> Time.Wen -> Job
data BootExn = ShipAlreadyBooted
deriving stock (Eq, Ord, Show)
deriving anyclass (Exception)
logStderr :: HasStderrLogFunc e => RIO LogFunc a -> RIO e a
logStderr action = do
logFunc <- view stderrLogFuncL
runRIO logFunc action
bootFromSeq :: e. (HasStderrLogFunc e, HasLogFunc e)
=> Serf e -> BootSeq -> RIO e ([Job], SerfState)
bootFromSeq serf (BootSeq ident nocks ovums) = do
handshake serf ident >>= \case
ss@(SerfState 1 (Mug 0)) -> loop [] ss Nothing bootSeqFns
_ -> throwIO ShipAlreadyBooted
where
loop :: [Job] -> SerfState -> Maybe (ProgressBar ()) -> [BootSeqFn]
-> RIO e ([Job], SerfState)
loop acc ss pb = \case
[] -> do
pb <- logStderr (updateProgressBar 0 bootMsg pb)
pure (reverse acc, ss)
x:xs -> do
wen <- io Time.now
job <- pure $ x (ssNextEv ss) (ssLastMug ss) wen
pb <- logStderr (updateProgressBar (1 + length xs) bootMsg pb)
(job, ss) <- bootJob serf job
loop (job:acc) ss pb xs
bootSeqFns :: [BootSeqFn]
bootSeqFns = fmap muckNock nocks <> fmap muckOvum ovums
where
muckNock nok eId mug _ = RunNok $ LifeCyc eId mug nok
muckOvum ov eId mug wen = DoWork $ Work eId mug wen ov
bootMsg = "Booting " ++ (fakeStr (isFake ident)) ++
(Ob.renderPatp (Ob.patp (fromIntegral (who ident))))
fakeStr True = "fake "
fakeStr False = ""
{-|
The ship is booted, but it is behind. shove events to the worker
until it is caught up.
-}
replayJobs :: (HasStderrLogFunc e, HasLogFunc e)
=> Serf e -> Int -> SerfState -> ConduitT Job Void (RIO e) SerfState
replayJobs serf lastEv = go Nothing
where
go pb ss = do
await >>= \case
Nothing -> pure ss
Just job -> do
pb <- lift $ logStderr (updatePb ss pb)
played <- lift $ replayJob serf job
go pb played
updatePb ss = do
let start = lastEv - fromIntegral (ssNextEv ss)
let msg = pack ( "Replaying events #" ++ (show (ssNextEv ss))
<> " to #" ++ (show lastEv)
)
updateProgressBar start msg
replay :: (HasStderrLogFunc e, HasLogFunc e)
=> Serf e -> Log.EventLog -> Maybe Word64 -> RIO e SerfState
replay serf log last = do
logTrace "Beginning event log replay" logTrace "Beginning event log replay"
lastEventInSnap <- io (serfLastEventBlocking serf)
last & \case last & \case
Nothing -> pure () Nothing -> pure ()
Just lt -> logTrace $ display $ Just lt -> logTrace $ display $
"User requested to replay up to event #" <> tshow lt "User requested to replay up to event #" <> tshow lt
ss <- handshake serf (Log.identity log) logLastEv :: Word64 <- atomically $ fromIntegral <$> Log.lastEv log
logLastEv :: Word64 <- fromIntegral <$> Log.lastEv log
let serfNextEv = ssNextEv ss
lastEventInSnap = serfNextEv - 1
logTrace $ display $ "Last event in event log is #" <> tshow logLastEv logTrace $ display $ "Last event in event log is #" <> tshow logLastEv
let replayUpTo = fromMaybe logLastEv last let replayUpTo = min (fromMaybe logLastEv last) logLastEv
let numEvs :: Int = fromIntegral replayUpTo - fromIntegral lastEventInSnap let numEvs :: Int = fromIntegral replayUpTo - fromIntegral lastEventInSnap
when (numEvs < 0) $ do
throwIO (SnapshotAheadOfLog logLastEv lastEventInSnap)
incProgress <- logStderr (trackProgress (fromIntegral numEvs))
logTrace $ display $ "Replaying up to event #" <> tshow replayUpTo logTrace $ display $ "Replaying up to event #" <> tshow replayUpTo
logTrace $ display $ "Will replay " <> tshow numEvs <> " in total." logTrace $ display $ "Will replay " <> tshow numEvs <> " in total."
runConduit $ Log.streamEvents log serfNextEv env <- ask
.| CC.take (fromIntegral numEvs)
.| toJobs (Log.identity log) serfNextEv
.| replayJobs serf (fromIntegral replayUpTo) ss
toJobs :: HasLogFunc e res <- runResourceT
=> LogIdentity -> EventId -> ConduitT ByteString Job (RIO e) () $ runConduit
toJobs ident eId = $ Log.streamEvents log (lastEventInSnap + 1)
await >>= \case .| CC.take (fromIntegral numEvs)
Nothing -> lift $ logTrace "[toJobs] no more jobs" .| CC.mapM (fmap snd . parseLogRow)
Just at -> do yield =<< lift (fromAtom at) .| replay 5 incProgress serf
lift $ logTrace $ display ("[toJobs] " <> tshow eId)
toJobs ident (eId+1)
where
isNock = eId <= fromIntegral (lifecycleLen ident)
fromAtom :: ByteString -> RIO e Job res & \case
fromAtom bs | isNock = do Nothing -> pure (Right $ fromIntegral numEvs)
noun <- cueBSExn bs Just er -> pure (Left er)
(mug, nok) <- fromNounExn noun
pure $ RunNok (LifeCyc eId mug nok) logStderr :: HasStderrLogFunc e => RIO LogFunc a -> RIO e a
fromAtom bs = do logStderr action = do
noun <- cueBSExn bs logFunc <- view stderrLogFuncL
(mug, wen, ovm) <- fromNounExn noun runRIO logFunc action
pure $ DoWork (Work eId mug wen ovm)
trackProgress
:: HasLogFunc e
=> Word64
-> RIO e (Int -> IO ())
trackProgress = \case
0 -> pure $ const $ pure ()
num -> do
let style = PB.defStyle { PB.stylePostfix = PB.exact }
let refresh = 10
let init = PB.Progress 0 (fromIntegral num) ()
bar <- PB.newProgressBar style refresh init
env <- ask
let incr = PB.incProgress bar
pure (runRIO env . incr)
-- Collect Effects for Parsing ------------------------------------------------- -- Collect FX ------------------------------------------------------------------
collectFX :: HasLogFunc e => Serf e -> Log.EventLog -> RIO e () collectFX :: HasLogFunc e => Serf -> Log.EventLog -> RIO e ()
collectFX serf log = do collectFX serf log = do
ss <- handshake serf (Log.identity log) lastEv <- io (serfLastEventBlocking serf)
runResourceT
$ runConduit
$ Log.streamEvents log (lastEv + 1)
.| CC.mapM (parseLogRow >=> fromNounExn . snd)
.| swim serf
.| persistFX log
runConduit $ Log.streamEvents log (ssNextEv ss) persistFX :: MonadIO m => Log.EventLog -> ConduitT (EventId, FX) Void m ()
.| toJobs (Log.identity log) (ssNextEv ss) persistFX log = CC.mapM_ $ \(eId, fx) -> do
.| doCollectFX serf ss Log.writeEffectsRow log eId $ jamBS $ toNoun fx
.| persistFX log
persistFX :: Log.EventLog -> ConduitT (EventId, FX) Void (RIO e) ()
persistFX log = loop
where
loop = await >>= \case
Nothing -> pure ()
Just (eId, fx) -> do
lift $ Log.writeEffectsRow log eId (jamBS $ toNoun fx)
loop
doCollectFX :: e. HasLogFunc e
=> Serf e -> SerfState -> ConduitT Job (EventId, FX) (RIO e) ()
doCollectFX serf = go
where
go :: SerfState -> ConduitT Job (EventId, FX) (RIO e) ()
go ss = await >>= \case
Nothing -> pure ()
Just jb -> do
-- jb <- pure $ replaceMug jb (ssLastMug ss)
(_, ss, fx) <- lift $ doJob serf jb
when (0 == (jobId jb `mod` 10_000)) $ do
lift $ logTrace $ displayShow (jobId jb)
yield (jobId jb, fx)
go ss
_replaceMug :: Job -> Mug -> Job
_replaceMug jb mug =
case jb of
DoWork (Work eId _ w o) -> DoWork (Work eId mug w o)
RunNok (LifeCyc eId _ n) -> RunNok (LifeCyc eId mug n)

View File

@ -0,0 +1,704 @@
{-|
Low-Level IPC flows for interacting with the serf process.
- Serf process can be started and shutdown with `start` and `stop`.
- You can ask the serf what it's last event was with
`serfLastEventBlocking`.
- A running serf can be asked to compact it's heap or take a snapshot.
- You can scry into a running serf.
- A running serf can be asked to execute a boot sequence, replay from
existing events, and run a ship with `boot`, `replay`, and `run`.
The `run` and `replay` flows will do batching of events to keep the
IPC pipe full.
```
|%
:: +writ: from king to serf
::
+$ gang (unit (set ship))
+$ writ
$% $: %live
$% [%cram eve=@]
[%exit cod=@]
[%save eve=@]
[%pack ~]
== ==
[%peek mil=@ now=@da lyc=gang pat=path]
[%play eve=@ lit=(list ?((pair @da ovum) *))]
[%work mil=@ job=(pair @da ovum)]
==
:: +plea: from serf to king
::
+$ plea
$% [%live ~]
[%ripe [pro=%1 hon=@ nok=@] eve=@ mug=@]
[%slog pri=@ ?(cord tank)]
$: %peek
$% [%done dat=(unit (cask))]
[%bail dud=goof]
== ==
$: %play
$% [%done mug=@]
[%bail eve=@ mug=@ dud=goof]
== ==
$: %work
$% [%done eve=@ mug=@ fec=(list ovum)]
[%swap eve=@ mug=@ job=(pair @da ovum) fec=(list ovum)]
[%bail lud=(list goof)]
== ==
==
```
-}
module Urbit.Vere.Serf.IPC
( Serf
, start
, stop
, serfLastEventBlocking
, snapshot
, compact
, scry
, boot
, replay
, run
, swim
, sendSIGINT
, module Urbit.Vere.Serf.Types
)
where
import Urbit.Prelude hiding ((<|))
import Data.Bits
import Data.Conduit
import System.Process
import Urbit.Vere.Serf.Types
import Control.Monad.STM (retry)
import Control.Monad.Trans.Resource (MonadResource, allocate, runResourceT)
import Data.Sequence (Seq((:<|), (:|>)))
import Foreign.Marshal.Alloc (alloca)
import Foreign.Ptr (castPtr)
import Foreign.Storable (peek, poke)
import RIO.Prelude (decodeUtf8Lenient)
import System.Posix.Signals (sigINT, sigKILL, signalProcess)
import Urbit.Arvo (Ev, FX)
import Urbit.Noun.Time (Wen)
import qualified Data.ByteString as BS
import qualified Data.ByteString.Unsafe as BS
import qualified System.IO.Error as IO
import qualified Urbit.Noun.Time as Time
-- Serf API --------------------------------------------------------------------
data Serf = Serf
{ serfSend :: Handle
, serfRecv :: Handle
, serfProc :: ProcessHandle
, serfSlog :: Slog -> IO ()
, serfLock :: MVar (Maybe SerfState)
}
-- Internal Protocol Types -----------------------------------------------------
data Live
= LExit Atom -- exit status code
| LSave EventId
| LCram EventId
| LPack ()
deriving (Show)
data Play
= PDone Mug
| PBail PlayBail
deriving (Show)
data Scry
= SDone (Maybe (Term, Noun))
| SBail Goof
deriving (Show)
data Work
= WDone EventId Mug FX
| WSwap EventId Mug (Wen, Noun) FX
| WBail [Goof]
deriving (Show)
data Writ
= WLive Live
| WPeek Atom Wen Gang Path
| WPlay EventId [Noun]
| WWork Atom Wen Ev
deriving (Show)
data Plea
= PLive ()
| PRipe SerfInfo
| PSlog Slog
| PPeek Scry
| PPlay Play
| PWork Work
deriving (Show)
deriveNoun ''Live
deriveNoun ''Play
deriveNoun ''Scry
deriveNoun ''Work
deriveNoun ''Writ
deriveNoun ''Plea
-- Access Current Serf State ---------------------------------------------------
serfLastEventBlocking :: Serf -> IO EventId
serfLastEventBlocking Serf{serfLock} = readMVar serfLock >>= \case
Nothing -> throwIO SerfNotRunning
Just ss -> pure (ssLast ss)
-- Low Level IPC Functions -----------------------------------------------------
fromRightExn :: (Exception e, MonadIO m) => Either a b -> (a -> e) -> m b
fromRightExn (Left m) exn = throwIO (exn m)
fromRightExn (Right x) _ = pure x
-- TODO Support Big Endian
sendLen :: Serf -> Int -> IO ()
sendLen s i = do
w <- evaluate (fromIntegral i :: Word64)
withWord64AsByteString w (hPut (serfSend s))
where
withWord64AsByteString :: Word64 -> (ByteString -> IO a) -> IO a
withWord64AsByteString w k = alloca $ \wp -> do
poke wp w
bs <- BS.unsafePackCStringLen (castPtr wp, 8)
k bs
sendBytes :: Serf -> ByteString -> IO ()
sendBytes s bs = handle onIOError $ do
sendLen s (length bs)
hPut (serfSend s) bs
hFlush (serfSend s)
where
onIOError :: IOError -> IO ()
onIOError = const (throwIO SerfConnectionClosed)
recvBytes :: Serf -> Word64 -> IO ByteString
recvBytes serf = BS.hGet (serfRecv serf) . fromIntegral
recvLen :: Serf -> IO Word64
recvLen w = do
bs <- BS.hGet (serfRecv w) 8
case length bs of
8 -> BS.unsafeUseAsCString bs (peek @Word64 . castPtr)
_ -> throwIO SerfConnectionClosed
recvResp :: Serf -> IO ByteString
recvResp serf = do
len <- recvLen serf
recvBytes serf len
-- Send Writ / Recv Plea -------------------------------------------------------
sendWrit :: Serf -> Writ -> IO ()
sendWrit s = sendBytes s . jamBS . toNoun
recvPlea :: Serf -> IO Plea
recvPlea w = do
b <- recvResp w
n <- fromRightExn (cueBS b) (const $ BadPleaAtom $ bytesAtom b)
p <- fromRightExn (fromNounErr @Plea n) (\(p, m) -> BadPleaNoun n p m)
pure p
recvPleaHandlingSlog :: Serf -> IO Plea
recvPleaHandlingSlog serf = loop
where
loop = recvPlea serf >>= \case
PSlog info -> serfSlog serf info >> loop
other -> pure other
-- Higher-Level IPC Functions --------------------------------------------------
recvRipe :: Serf -> IO SerfInfo
recvRipe serf = recvPleaHandlingSlog serf >>= \case
PRipe ripe -> pure ripe
plea -> throwIO (UnexpectedPlea (toNoun plea) "expecting %play")
recvPlay :: Serf -> IO Play
recvPlay serf = recvPleaHandlingSlog serf >>= \case
PPlay play -> pure play
plea -> throwIO (UnexpectedPlea (toNoun plea) "expecting %play")
recvLive :: Serf -> IO ()
recvLive serf = recvPleaHandlingSlog serf >>= \case
PLive () -> pure ()
plea -> throwIO (UnexpectedPlea (toNoun plea) "expecting %live")
recvWork :: Serf -> IO Work
recvWork serf = do
recvPleaHandlingSlog serf >>= \case
PWork work -> pure work
plea -> throwIO (UnexpectedPlea (toNoun plea) "expecting %work")
recvPeek :: Serf -> IO (Maybe (Term, Noun))
recvPeek serf = do
recvPleaHandlingSlog serf >>= \case
PPeek (SDone peek) -> pure peek
-- XX produce error
PPeek (SBail dud) -> throwIO (PeekBail dud)
plea -> throwIO (UnexpectedPlea (toNoun plea) "expecting %peek")
-- Request-Response Points -- These don't touch the lock -----------------------
sendSnapshotRequest :: Serf -> EventId -> IO ()
sendSnapshotRequest serf eve = do
sendWrit serf (WLive $ LSave eve)
recvLive serf
sendCompactionRequest :: Serf -> IO ()
sendCompactionRequest serf = do
sendWrit serf (WLive $ LPack ())
recvLive serf
sendScryRequest :: Serf -> Wen -> Gang -> Path -> IO (Maybe (Term, Noun))
sendScryRequest serf w g p = do
sendWrit serf (WPeek 0 w g p)
recvPeek serf
sendShutdownRequest :: Serf -> Atom -> IO ()
sendShutdownRequest serf exitCode = do
sendWrit serf (WLive $ LExit exitCode)
pure ()
-- Starting the Serf -----------------------------------------------------------
compileFlags :: [Flag] -> Word
compileFlags = foldl' (\acc flag -> setBit acc (fromEnum flag)) 0
readStdErr :: Handle -> (Text -> IO ()) -> IO () -> IO ()
readStdErr h onLine onClose = loop
where
loop = do
IO.tryIOError (BS.hGetLine h >>= onLine . decodeUtf8Lenient) >>= \case
Left exn -> onClose
Right () -> loop
start :: Config -> IO (Serf, SerfInfo)
start (Config exePax pierPath flags onSlog onStdr onDead) = do
(Just i, Just o, Just e, p) <- createProcess pSpec
void $ async (readStdErr e onStdr onDead)
vLock <- newEmptyMVar
let serf = Serf i o p onSlog vLock
info <- recvRipe serf
putMVar vLock (Just $ siStat info)
pure (serf, info)
where
diskKey = ""
config = show (compileFlags flags)
rock = "0" -- XX support loading from rock
cache = "50000" -- XX support memo-cache size
args = ["serf", pierPath, diskKey, config, cache, rock]
pSpec = (proc exePax args) { std_in = CreatePipe
, std_out = CreatePipe
, std_err = CreatePipe
}
-- Taking the SerfState Lock ---------------------------------------------------
takeLock :: MonadIO m => Serf -> m SerfState
takeLock serf = io $ do
takeMVar (serfLock serf) >>= \case
Nothing -> putMVar (serfLock serf) Nothing >> throwIO SerfNotRunning
Just ss -> pure ss
serfLockTaken
:: MonadResource m => Serf -> m (IORef (Maybe SerfState), SerfState)
serfLockTaken serf = snd <$> allocate take release
where
take = (,) <$> newIORef Nothing <*> takeLock serf
release (rv, _) = do
mRes <- readIORef rv
when (mRes == Nothing) (forcefullyKillSerf serf)
putMVar (serfLock serf) mRes
withSerfLock
:: MonadResource m => Serf -> (SerfState -> m (SerfState, a)) -> m a
withSerfLock serf act = do
(vState , initialState) <- serfLockTaken serf
(newState, result ) <- act initialState
writeIORef vState (Just newState)
pure result
withSerfLockIO :: Serf -> (SerfState -> IO (SerfState, a)) -> IO a
withSerfLockIO s a = runResourceT (withSerfLock s (io . a))
-- SIGINT ----------------------------------------------------------------------
sendSIGINT :: Serf -> IO ()
sendSIGINT serf = do
getPid (serfProc serf) >>= \case
Nothing -> pure ()
Just pid -> do
io $ signalProcess sigINT pid
-- Killing the Serf ------------------------------------------------------------
{-|
Ask the serf to shutdown. If it takes more than 2s, kill it with
SIGKILL.
-}
stop :: HasLogFunc e => Serf -> RIO e ()
stop serf = do
race_ niceKill (wait2sec >> forceKill)
where
wait2sec = threadDelay 2_000_000
niceKill = do
logTrace "Asking serf to shut down"
io (gracefullyKillSerf serf)
logTrace "Serf went down when asked."
forceKill = do
logTrace "Serf taking too long to go down, kill with fire (SIGTERM)."
io (forcefullyKillSerf serf)
logTrace "Serf process killed with SIGTERM."
{-|
Kill the serf by taking the lock, then asking for it to exit.
-}
gracefullyKillSerf :: Serf -> IO ()
gracefullyKillSerf serf@Serf{..} = do
finalState <- takeMVar serfLock
sendShutdownRequest serf 0
waitForProcess serfProc
pure ()
{-|
Kill the serf by sending it a SIGKILL.
-}
forcefullyKillSerf :: Serf -> IO ()
forcefullyKillSerf serf = do
getPid (serfProc serf) >>= \case
Nothing -> pure ()
Just pid -> do
io $ signalProcess sigKILL pid
io $ void $ waitForProcess (serfProc serf)
-- Flows for Interacting with the Serf -----------------------------------------
{-|
Ask the serf to write a snapshot to disk.
-}
snapshot :: Serf -> IO ()
snapshot serf = withSerfLockIO serf $ \ss -> do
sendSnapshotRequest serf (ssLast ss)
pure (ss, ())
{-|
Ask the serf to de-duplicate and de-fragment it's heap.
-}
compact :: Serf -> IO ()
compact serf = withSerfLockIO serf $ \ss -> do
sendCompactionRequest serf
pure (ss, ())
{-|
Peek into the serf state.
-}
scry :: Serf -> Wen -> Gang -> Path -> IO (Maybe (Term, Noun))
scry serf w g p = withSerfLockIO serf $ \ss -> do
(ss,) <$> sendScryRequest serf w g p
{-|
Given a list of boot events, send them to to the serf in a single
%play message. They must all be sent in a single %play event so that
the serf can determine the length of the boot sequence.
-}
boot :: Serf -> [Noun] -> IO (Maybe PlayBail)
boot serf@Serf {..} seq = do
withSerfLockIO serf $ \ss -> do
sendWrit serf (WPlay 1 seq)
recvPlay serf >>= \case
PBail bail -> pure (ss, Just bail)
PDone mug -> pure (SerfState (fromIntegral $ length seq) mug, Nothing)
{-|
Given a stream of nouns (from the event log), feed them into the serf
in batches of size `batchSize`.
- On `%bail` response, return early.
- On IPC errors, kill the serf and rethrow.
- On success, return `Nothing`.
-}
replay
:: forall m
. (MonadResource m, MonadUnliftIO m, MonadIO m)
=> Int
-> (Int -> IO ())
-> Serf
-> ConduitT Noun Void m (Maybe PlayBail)
replay batchSize cb serf = do
withSerfLock serf $ \ss -> do
(r, ss') <- loop ss
pure (ss', r)
where
loop :: SerfState -> ConduitT Noun Void m (Maybe PlayBail, SerfState)
loop ss@(SerfState lastEve lastMug) = do
awaitBatch batchSize >>= \case
[] -> pure (Nothing, SerfState lastEve lastMug)
evs -> do
let nexEve = lastEve + 1
let newEve = lastEve + fromIntegral (length evs)
io $ sendWrit serf (WPlay nexEve evs)
io (recvPlay serf) >>= \case
PBail bail -> pure (Just bail, SerfState lastEve lastMug)
PDone newMug -> do
io (cb $ length evs)
loop (SerfState newEve newMug)
{-|
TODO If this is slow, use a mutable vector instead of reversing a list.
-}
awaitBatch :: Monad m => Int -> ConduitT i o m [i]
awaitBatch = go []
where
go acc 0 = pure (reverse acc)
go acc n = await >>= \case
Nothing -> pure (reverse acc)
Just x -> go (x:acc) (n-1)
-- Special Replay for Collecting FX --------------------------------------------
{-|
This does event-log replay using the running IPC flow so that we
can collect effects.
We don't tolerate replacement events or bails since we are actually
replaying the log, so we just throw exceptions in those cases.
-}
swim
:: forall m
. (MonadIO m, MonadUnliftIO m, MonadResource m)
=> Serf
-> ConduitT (Wen, Ev) (EventId, FX) m ()
swim serf = do
withSerfLock serf $ \SerfState {..} -> do
(, ()) <$> loop ssHash ssLast
where
loop
:: Mug
-> EventId
-> ConduitT (Wen, Ev) (EventId, FX) m SerfState
loop mug eve = await >>= \case
Nothing -> do
pure (SerfState eve mug)
Just (wen, evn) -> do
io (sendWrit serf (WWork 0 wen evn))
io (recvWork serf) >>= \case
WBail goofs -> do
throwIO (BailDuringReplay eve goofs)
WSwap eid hash (wen, noun) fx -> do
throwIO (SwapDuringReplay eid hash (wen, noun) fx)
WDone eid hash fx -> do
yield (eid, fx)
loop hash eid
-- Running Ship Flow -----------------------------------------------------------
{-|
TODO Don't take snapshot until event log has processed current event.
-}
run
:: Serf
-> Int
-> STM EventId
-> STM RunReq
-> ((Fact, FX) -> STM ())
-> (Maybe Ev -> STM ())
-> IO ()
run serf maxBatchSize getLastEvInLog onInput sendOn spin = topLoop
where
topLoop :: IO ()
topLoop = atomically onInput >>= \case
RRWork workErr -> doWork workErr
RRSave () -> doSave
RRKill () -> doKill
RRPack () -> doPack
RRScry w g p k -> doScry w g p k
doPack :: IO ()
doPack = compact serf >> topLoop
waitForLog :: IO ()
waitForLog = do
serfLast <- serfLastEventBlocking serf
atomically $ do
logLast <- getLastEvInLog
when (logLast < serfLast) retry
doSave :: IO ()
doSave = waitForLog >> snapshot serf >> topLoop
doKill :: IO ()
doKill = waitForLog >> snapshot serf >> pure ()
doScry :: Wen -> Gang -> Path -> (Maybe (Term, Noun) -> IO ()) -> IO ()
doScry w g p k = (scry serf w g p >>= k) >> topLoop
doWork :: EvErr -> IO ()
doWork firstWorkErr = do
que <- newTBMQueueIO 1
() <- atomically (writeTBMQueue que firstWorkErr)
tWork <- async (processWork serf maxBatchSize que onWorkResp spin)
flip onException (cancel tWork) $ do
nexSt <- workLoop que
wait tWork
nexSt
workLoop :: TBMQueue EvErr -> IO (IO ())
workLoop que = atomically onInput >>= \case
RRKill () -> atomically (closeTBMQueue que) >> pure doKill
RRSave () -> atomically (closeTBMQueue que) >> pure doSave
RRPack () -> atomically (closeTBMQueue que) >> pure doPack
RRScry w g p k -> atomically (closeTBMQueue que) >> pure (doScry w g p k)
RRWork workErr -> atomically (writeTBMQueue que workErr) >> workLoop que
onWorkResp :: Wen -> EvErr -> Work -> IO ()
onWorkResp wen (EvErr evn err) = \case
WDone eid hash fx -> do
io $ err (RunOkay eid)
atomically $ sendOn ((Fact eid hash wen (toNoun evn)), fx)
WSwap eid hash (wen, noun) fx -> do
io $ err (RunSwap eid hash wen noun fx)
atomically $ sendOn (Fact eid hash wen noun, fx)
WBail goofs -> do
io $ err (RunBail goofs)
{-|
Given:
- A stream of incoming requests
- A sequence of in-flight requests that haven't been responded to
- A maximum number of in-flight requests.
Wait until the number of in-fligh requests is smaller than the maximum,
and then take the next item from the stream of requests.
-}
pullFromQueueBounded :: Int -> TVar (Seq a) -> TBMQueue b -> STM (Maybe b)
pullFromQueueBounded maxSize vInFlight queue = do
inFlight <- length <$> readTVar vInFlight
if inFlight >= maxSize
then retry
else readTBMQueue queue
{-|
Given
- `maxSize`: The maximum number of jobs to send to the serf before
getting a response.
- `q`: A bounded queue (which can be closed)
- `onResp`: a callback to call for each response from the serf.
- `spin`: a callback to tell the terminal driver which event is
currently being processed.
Pull jobs from the queue and send them to the serf (eagerly, up to
`maxSize`) and call the callback with each response from the serf.
When the queue is closed, wait for the serf to respond to all pending
work, and then return.
Whenever the serf is idle, call `spin Nothing` and whenever the serf
is working on an event, call `spin (Just ev)`.
-}
processWork
:: Serf
-> Int
-> TBMQueue EvErr
-> (Wen -> EvErr -> Work -> IO ())
-> (Maybe Ev -> STM ())
-> IO ()
processWork serf maxSize q onResp spin = do
vDoneFlag <- newTVarIO False
vInFlightQueue <- newTVarIO empty
recvThread <- async (recvLoop serf vDoneFlag vInFlightQueue spin)
flip onException (print "KILLING: processWork" >> cancel recvThread) $ do
loop vInFlightQueue vDoneFlag
wait recvThread
where
loop :: TVar (Seq (Ev, Work -> IO ())) -> TVar Bool -> IO ()
loop vInFlight vDone = do
atomically (pullFromQueueBounded maxSize vInFlight q) >>= \case
Nothing -> do
atomically (writeTVar vDone True)
Just evErr@(EvErr ev _) -> do
now <- Time.now
let cb = onResp now evErr
atomically $ modifyTVar' vInFlight (:|> (ev, cb))
sendWrit serf (WWork 0 now ev)
loop vInFlight vDone
{-|
Given:
- `vDone`: A flag that no more work will be sent to the serf.
- `vWork`: A list of work requests that have been sent to the serf,
haven't been responded to yet.
If the serf has responded to all work requests, and no more work is
going to be sent to the serf, then return.
If we are going to send more work to the serf, but the queue is empty,
then wait.
If work requests have been sent to the serf, take the first one,
wait for a response from the serf, call the associated callback,
and repeat the whole process.
-}
recvLoop
:: Serf
-> TVar Bool
-> TVar (Seq (Ev, Work -> IO ()))
-> (Maybe Ev -> STM ())
-> IO ()
recvLoop serf vDone vWork spin = do
withSerfLockIO serf \SerfState {..} -> do
loop ssLast ssHash
where
loop eve mug = do
atomically $ do
whenM (null <$> readTVar vWork) $ do
spin Nothing
atomically takeCallback >>= \case
Nothing -> pure (SerfState eve mug, ())
Just (curEve, cb) -> do
atomically (spin (Just curEve))
recvWork serf >>= \case
work@(WDone eid hash _) -> cb work >> loop eid hash
work@(WSwap eid hash _ _) -> cb work >> loop eid hash
work@(WBail _) -> cb work >> loop eve mug
takeCallback :: STM (Maybe (Ev, Work -> IO ()))
takeCallback = do
((,) <$> readTVar vDone <*> readTVar vWork) >>= \case
(False, Empty ) -> retry
(True , Empty ) -> pure Nothing
(_ , (e, x) :<| xs) -> writeTVar vWork xs $> Just (e, x)
(_ , _ ) -> error "impossible"

View File

@ -0,0 +1,121 @@
module Urbit.Vere.Serf.Types where
import Urbit.Prelude
import Urbit.Arvo (Ev, FX)
import Urbit.Noun.Time (Wen)
-- Types -----------------------------------------------------------------------
type EventId = Word64
type PlayBail = (EventId, Mug, Goof)
type Slog = (Atom, Tank)
data SerfState = SerfState
{ ssLast :: EventId
, ssHash :: Mug
}
deriving (Show, Eq)
data RipeInfo = RipeInfo
{ riProt :: Atom
, riHoon :: Atom
, riNock :: Atom
}
deriving (Show)
data SerfInfo = SerfInfo
{ siRipe :: RipeInfo
, siStat :: SerfState
}
deriving (Show)
data Fact = Fact
{ factEve :: EventId
, factMug :: Mug
, factWen :: Wen
, factNon :: Noun
}
data Flag
= DebugRam
| DebugCpu
| CheckCorrupt
| CheckFatal
| Verbose
| DryRun
| Quiet
| Hashless
| Trace
deriving (Eq, Ord, Show, Enum, Bounded)
data Config = Config
{ scSerf :: FilePath -- Where is the urbit-worker executable?
, scPier :: FilePath -- Where is the pier directory?
, scFlag :: [Flag] -- Serf execution flags.
, scSlog :: Slog -> IO () -- What to do with slogs?
, scStdr :: Text -> IO () -- What to do with lines from stderr?
, scDead :: IO () -- What to do when the serf process goes down?
}
-- Serf Commands ---------------------------------------------------------------
type Gang = Maybe (HoonSet Ship)
type Goof = (Term, [Tank])
data EvErr = EvErr Ev (WorkError -> IO ())
{-|
Two types of serf failures.
- `RunSwap`: Event processing failed, but the serf replaced it with
another event which succeeded.
- `RunBail`: Event processing failed and all attempt to replace it
with a failure-notice event also caused crashes. We are really fucked.
-}
data WorkError -- TODO Rename type and constructors
= RunSwap EventId Mug Wen Noun FX -- TODO Maybe provide less info here?
| RunBail [Goof]
| RunOkay EventId
{-
- RRWork: Ask the serf to do work, will output (Fact, FX) if work
succeeded and call callback on failure.
- RRSave: Wait for the serf to finish all pending work
-}
data RunReq
= RRWork EvErr
| RRSave ()
| RRKill ()
| RRPack ()
| RRScry Wen Gang Path (Maybe (Term, Noun) -> IO ())
-- Exceptions ------------------------------------------------------------------
data SerfExn
= UnexpectedPlea Noun Text
| BadPleaAtom Atom
| BadPleaNoun Noun [Text] Text
| PeekBail Goof
| SerfConnectionClosed
| SerfHasShutdown
| BailDuringReplay EventId [Goof]
| SwapDuringReplay EventId Mug (Wen, Noun) FX
| SerfNotRunning
| MissingBootEventsInEventLog Word Word
| SnapshotAheadOfLog EventId EventId
deriving (Show, Exception)
-- Instances -------------------------------------------------------------------
deriveNoun ''RipeInfo
deriveNoun ''SerfInfo
deriveNoun ''SerfState

View File

@ -8,6 +8,7 @@ module Urbit.Vere.Term
, runTerminalClient , runTerminalClient
, connClient , connClient
, term , term
, term'
) where ) where
import Data.Char import Data.Char
@ -18,19 +19,21 @@ import RIO.FilePath
import System.Posix.IO import System.Posix.IO
import System.Posix.Terminal import System.Posix.Terminal
import Urbit.Arvo hiding (Term) import Urbit.Arvo hiding (Term)
import Urbit.King.Config import Urbit.King.App
import Urbit.Noun.Time
import Urbit.Prelude hiding (getCurrentTime) import Urbit.Prelude hiding (getCurrentTime)
import Urbit.Time
import Urbit.Vere.Pier.Types 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.King.App (HasConfigDir(..)) import Urbit.TermSize (TermSize(TermSize))
import Urbit.Vere.Term.API (Client(Client)) import Urbit.Vere.Term.API (Client(Client))
import qualified Data.ByteString.Internal as BS import qualified Data.ByteString.Internal as BS
import qualified Data.ByteString.UTF8 as BS import qualified Data.ByteString.UTF8 as BS
import qualified System.Console.ANSI as ANSI
import qualified Urbit.TermSize as T
import qualified Urbit.Vere.NounServ as Serv import qualified Urbit.Vere.NounServ as Serv
import qualified Urbit.Vere.Term.API as Term import qualified Urbit.Vere.Term.API as Term
import qualified Urbit.Vere.Term.Render as T import qualified Urbit.Vere.Term.Render as T
@ -74,20 +77,6 @@ initialHail = EvBlip $ BlipEvTerm $ TermEvHail (UD 1, ()) ()
-- Version one of this is punting on the ops_u.dem flag: whether we're running -- Version one of this is punting on the ops_u.dem flag: whether we're running
-- in daemon mode. -- in daemon mode.
spinners :: [Text]
spinners = ["|", "/", "-", "\\"]
leftBracket :: Text
leftBracket = "«"
rightBracket :: Text
rightBracket = "»"
_spin_cool_us = 500000
_spin_warm_us = 50000
_spin_rate_us = 250000
_spin_idle_us = 500000
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
rioAllocaBytes :: (MonadIO m, MonadUnliftIO m) rioAllocaBytes :: (MonadIO m, MonadUnliftIO m)
@ -138,7 +127,7 @@ connectToRemote port local = mkRAcquire start stop
data HackConfigDir = HCD { _hcdPax :: FilePath } data HackConfigDir = HCD { _hcdPax :: FilePath }
makeLenses ''HackConfigDir makeLenses ''HackConfigDir
instance HasConfigDir HackConfigDir where configDirL = hcdPax instance HasPierPath HackConfigDir where pierPathL = hcdPax
runTerminalClient :: e. HasLogFunc e => FilePath -> RIO e () runTerminalClient :: e. HasLogFunc e => FilePath -> RIO e ()
runTerminalClient pier = runRAcquire $ do runTerminalClient pier = runRAcquire $ do
@ -153,20 +142,46 @@ runTerminalClient pier = runRAcquire $ do
runRAcquire :: RAcquire e () -> RIO e () runRAcquire :: RAcquire e () -> RIO e ()
runRAcquire act = rwith act $ const $ pure () runRAcquire act = rwith act $ const $ pure ()
-- Spinner ---------------------------------------------------------------------
-- Call an STM action after delay of `first` microseconds and then every
-- `rest` microseconds after that.
repeatedly :: Int -> Int -> STM () -> IO ()
repeatedly first rest action = do
threadDelay first
forever $ do
atomically action
threadDelay rest
spinners :: [Text]
spinners = ["|", "/", "-", "\\"]
leftBracket, rightBracket :: Text
leftBracket = "«"
rightBracket = "»"
_spin_cool_us = 500000
_spin_warm_us = 50000
_spin_rate_us = 250000
_spin_idle_us = 500000
-- Client ----------------------------------------------------------------------
{-| {-|
Initializes the generalized input/output parts of the terminal. Initializes the generalized input/output parts of the terminal.
-} -}
localClient :: e. HasLogFunc e localClient :: e. HasLogFunc e
=> STM () => STM ()
-> RAcquire e (T.TSize, Client) -> RAcquire e (TermSize, Client)
localClient doneSignal = fst <$> mkRAcquire start stop localClient doneSignal = fst <$> mkRAcquire start stop
where where
start :: HasLogFunc e => RIO e ((T.TSize, Client), Private) start :: HasLogFunc e => RIO e ((TermSize, Client), Private)
start = do start = do
tsWriteQueue <- newTQueueIO tsWriteQueue <- newTQueueIO :: RIO e (TQueue [Term.Ev])
spinnerMVar <- newEmptyTMVarIO spinnerMVar <- newEmptyTMVarIO :: RIO e (TMVar ())
pWriterThread <- pWriterThread <- asyncBound (writeTerminal tsWriteQueue spinnerMVar)
asyncBound (writeTerminal tsWriteQueue spinnerMVar)
pPreviousConfiguration <- io $ getTerminalAttributes stdInput pPreviousConfiguration <- io $ getTerminalAttributes stdInput
@ -187,12 +202,12 @@ localClient doneSignal = fst <$> mkRAcquire start stop
, give = writeTQueue tsWriteQueue , give = writeTQueue tsWriteQueue
} }
tsize <- io $ T.tsize tsize <- io $ T.termSize
pure ((tsize, client), Private{..}) pure ((tsize, client), Private{..})
stop :: HasLogFunc e stop :: HasLogFunc e
=> ((T.TSize, Client), Private) -> RIO e () => ((TermSize, Client), Private) -> RIO e ()
stop ((_, Client{..}), Private{..}) = do stop ((_, Client{..}), Private{..}) = do
-- Note that we don't `cancel pReaderThread` here. This is a deliberate -- Note that we don't `cancel pReaderThread` here. This is a deliberate
-- decision because fdRead calls into a native function which the runtime -- decision because fdRead calls into a native function which the runtime
@ -226,17 +241,6 @@ localClient doneSignal = fst <$> mkRAcquire start stop
, ProcessOutput , ProcessOutput
] ]
-- An async which will put into an mvar after a delay. Used to spin the
-- spinner in writeTerminal.
spinnerHeartBeat :: Int -> Int -> TMVar () -> RIO e ()
spinnerHeartBeat first rest mvar = do
threadDelay first
loop
where
loop = do
atomically $ putTMVar mvar ()
threadDelay rest
loop
-- Writes data to the terminal. Both the terminal reading, normal logging, -- Writes data to the terminal. Both the terminal reading, normal logging,
-- and effect handling can all emit bytes which go to the terminal. -- and effect handling can all emit bytes which go to the terminal.
@ -246,9 +250,7 @@ localClient doneSignal = fst <$> mkRAcquire start stop
loop (LineState "" 0 Nothing Nothing True 0 currentTime) loop (LineState "" 0 Nothing Nothing True 0 currentTime)
where where
writeBlank :: LineState -> RIO e LineState writeBlank :: LineState -> RIO e LineState
writeBlank ls = do writeBlank ls = putStr "\r\n" $> ls
putStr "\r\n"
pure ls
writeTrace :: LineState -> Text -> RIO e LineState writeTrace :: LineState -> Text -> RIO e LineState
writeTrace ls p = do writeTrace ls p = do
@ -266,6 +268,8 @@ localClient doneSignal = fst <$> mkRAcquire start stop
-} -}
doSpin :: LineState -> Maybe Text -> RIO e LineState doSpin :: LineState -> Maybe Text -> RIO e LineState
doSpin ls@LineState{..} mTxt = do doSpin ls@LineState{..} mTxt = do
maybe (pure ()) cancel lsSpinTimer
current <- io $ now current <- io $ now
delay <- pure $ case mTxt of delay <- pure $ case mTxt of
Nothing -> 0 Nothing -> 0
@ -274,7 +278,10 @@ localClient doneSignal = fst <$> mkRAcquire start stop
then _spin_warm_us then _spin_warm_us
else _spin_cool_us else _spin_cool_us
spinTimer <- async $ spinnerHeartBeat delay _spin_rate_us spinner spinTimer <- io $ async
$ repeatedly delay _spin_rate_us
$ void
$ tryPutTMVar spinner ()
pure $ ls { lsSpinTimer = Just spinTimer pure $ ls { lsSpinTimer = Just spinTimer
, lsSpinCause = mTxt , lsSpinCause = mTxt
@ -291,7 +298,7 @@ localClient doneSignal = fst <$> mkRAcquire start stop
-- If we ever actually ran the spinner display callback, we need -- If we ever actually ran the spinner display callback, we need
-- to force a redisplay of the command prompt. -- to force a redisplay of the command prompt.
ls <- if not lsSpinFirstRender ls <- if not lsSpinFirstRender || True
then termRefreshLine ls then termRefreshLine ls
else pure ls else pure ls
@ -306,16 +313,16 @@ localClient doneSignal = fst <$> mkRAcquire start stop
Term.Spinr (Just txt) -> doSpin ls (unCord <$> txt) Term.Spinr (Just txt) -> doSpin ls (unCord <$> txt)
Term.Spinr Nothing -> unspin ls Term.Spinr Nothing -> unspin ls
-- TODO What does this do?
spin :: LineState -> RIO e LineState spin :: LineState -> RIO e LineState
spin ls@LineState{..} = do spin ls@LineState{..} = do
let spinner = (spinners !! lsSpinFrame) ++ case lsSpinCause of let spinner = (spinners !! lsSpinFrame) ++ case lsSpinCause of
Nothing -> "" Nothing -> ""
Just str -> leftBracket ++ str ++ rightBracket Just str -> leftBracket ++ str ++ rightBracket
putStr spinner putStr (spinner <> pack (ANSI.cursorBackwardCode (length spinner)))
termSpinnerMoveLeft (length spinner)
let newFrame = (lsSpinFrame + 1) `mod` (length spinners) let newFrame = (lsSpinFrame + 1) `mod` length spinners
pure $ ls { lsSpinFirstRender = False pure $ ls { lsSpinFirstRender = False
, lsSpinFrame = newFrame , lsSpinFrame = newFrame
@ -356,8 +363,8 @@ localClient doneSignal = fst <$> mkRAcquire start stop
-- Moves the cursor left without any mutation of the LineState. Used only -- Moves the cursor left without any mutation of the LineState. Used only
-- in cursor spinning. -- in cursor spinning.
termSpinnerMoveLeft :: Int RIO e () _termSpinnerMoveLeft :: Int RIO e ()
termSpinnerMoveLeft = T.cursorLeft _termSpinnerMoveLeft = T.cursorLeft
-- Displays and sets the current line -- Displays and sets the current line
termShowLine :: LineState -> Text -> RIO e LineState termShowLine :: LineState -> Text -> RIO e LineState
@ -489,28 +496,55 @@ localClient doneSignal = fst <$> mkRAcquire start stop
-- logDebug $ displayShow ("terminalBelt", b) -- logDebug $ displayShow ("terminalBelt", b)
atomically $ writeTQueue rq b atomically $ writeTQueue rq b
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
{-|
Terminal Driver
Until blew/hail events succeeds, ignore effects.
Wait until blew/hail event callbacks invoked.
If success, signal success.
If failure, try again several times.
If still failure, bring down ship.
Don't wait for other drivers to boot
Begin normal operation (start accepting requests)
-}
term'
:: HasPierEnv e
=> (TermSize, Client)
-> IO ()
-> RIO e ([Ev], RAcquire e (DriverApi TermEf))
term' (tsize, client) serfSIGINT = do
let TermSize wi hi = tsize
initEv = [initialBlew wi hi, initialHail]
pure (initEv, runDriver)
where
runDriver = do
env <- ask
ventQ :: TQueue EvErr <- newTQueueIO
diOnEffect <- term env (tsize, client) (writeTQueue ventQ) serfSIGINT
let diEventSource = fmap RRWork <$> tryReadTQueue ventQ
pure (DriverApi {..})
{-| {-|
Terminal Driver Terminal Driver
-} -}
term :: forall e. (HasPierConfig e, HasLogFunc e) term :: forall e. (HasPierEnv e)
=> (T.TSize, Client) => e
-> (STM ()) -> (TermSize, Client)
-> KingId -> (EvErr -> STM ())
-> QueueEv -> IO ()
-> ([Ev], RAcquire e (EffCb e TermEf)) -> RAcquire e (TermEf -> IO ())
term (tsize, Client{..}) shutdownSTM king enqueueEv = term env (tsize, Client{..}) plan serfSIGINT = runTerm
(initialEvents, runTerm)
where where
T.TSize wi hi = tsize runTerm :: RAcquire e (TermEf -> IO ())
initialEvents = [(initialBlew wi hi), initialHail]
runTerm :: RAcquire e (EffCb e TermEf)
runTerm = do runTerm = do
tim <- mkRAcquire (async readLoop) cancel tim <- mkRAcquire (async readLoop) cancel
pure handleEffect pure (runRIO env . handleEffect)
{- {-
Because our terminals are always `Demux`ed, we don't have to Because our terminals are always `Demux`ed, we don't have to
@ -521,14 +555,17 @@ term (tsize, Client{..}) shutdownSTM king enqueueEv =
atomically take >>= \case atomically take >>= \case
Nothing -> pure () Nothing -> pure ()
Just b -> do Just b -> do
let blip = EvBlip $ BlipEvTerm $ TermEvBelt (UD 1, ()) $ b when (b == Ctl (Cord "c")) $ do
atomically $ enqueueEv $ blip io serfSIGINT
let beltEv = EvBlip $ BlipEvTerm $ TermEvBelt (UD 1, ()) $ b
let beltFailed _ = pure ()
atomically $ plan (EvErr beltEv beltFailed)
handleEffect :: TermEf -> RIO e () handleEffect :: TermEf -> RIO e ()
handleEffect = \case handleEffect = \case
TermEfInit _ _ -> pure () TermEfInit _ _ -> pure ()
TermEfMass _ _ -> pure () TermEfMass _ _ -> pure ()
TermEfLogo _ _ -> atomically shutdownSTM TermEfLogo _ _ -> atomically =<< view killPierActionL
TermEfBlit _ blits -> do TermEfBlit _ blits -> do
let (termBlits, fsWrites) = partition isTerminalBlit blits let (termBlits, fsWrites) = partition isTerminalBlit blits
atomically $ give [Term.Blits termBlits] atomically $ give [Term.Blits termBlits]

View File

@ -2,9 +2,7 @@
Terminal Driver Terminal Driver
-} -}
module Urbit.Vere.Term.Render module Urbit.Vere.Term.Render
( TSize(..) ( clearScreen
, tsize
, clearScreen
, clearLine , clearLine
, cursorRight , cursorRight
, cursorLeft , cursorLeft
@ -13,29 +11,11 @@ module Urbit.Vere.Term.Render
import ClassyPrelude import ClassyPrelude
import qualified System.Console.Terminal.Size as TSize import qualified System.Console.ANSI as ANSI
import qualified System.Console.ANSI as ANSI
-- Types ----------------------------------------------------------------------- -- Types -----------------------------------------------------------------------
data TSize = TSize
{ tsWide Word
, tsTall Word
}
--------------------------------------------------------------------------------
{- |
Get terminal size. Produces 80x24 as a fallback if unable to figure
out terminal size.
-}
tsize IO TSize
tsize = do
TSize.Window wi hi <- TSize.size <&> fromMaybe (TSize.Window 80 24)
pure $ TSize { tsWide = wi, tsTall = hi }
clearScreen MonadIO m m () clearScreen MonadIO m m ()
clearScreen = liftIO $ ANSI.clearScreen clearScreen = liftIO $ ANSI.clearScreen

View File

@ -1,5 +1,5 @@
name: urbit-king name: urbit-king
version: 0.10.4 version: 0.10.8
license: MIT license: MIT
license-file: LICENSE license-file: LICENSE
@ -72,6 +72,7 @@ dependencies:
- primitive - primitive
- process - process
- QuickCheck - QuickCheck
- racquire
- random - random
- regex-tdfa - regex-tdfa
- regex-tdfa-text - regex-tdfa-text
@ -88,10 +89,10 @@ dependencies:
- tasty-th - tasty-th
- template-haskell - template-haskell
- terminal-progress-bar - terminal-progress-bar
- terminal-size
- text - text
- these - these
- time - time
- tls
- transformers - transformers
- unix - unix
- unliftio - unliftio
@ -99,7 +100,11 @@ dependencies:
- unordered-containers - unordered-containers
- urbit-atom - urbit-atom
- urbit-azimuth - urbit-azimuth
- urbit-eventlog-lmdb
- urbit-hob - urbit-hob
- urbit-noun
- urbit-noun-core
- urbit-termsize
- utf8-string - utf8-string
- vector - vector
- wai - wai

View File

@ -8,22 +8,27 @@ import Test.Tasty
import Test.Tasty.QuickCheck import Test.Tasty.QuickCheck
import Test.Tasty.TH import Test.Tasty.TH
import Urbit.Arvo import Urbit.Arvo
import Urbit.EventLog.LMDB
import Urbit.King.Config import Urbit.King.Config
import Urbit.Noun import Urbit.Noun
import Urbit.Noun.Time
import Urbit.Prelude import Urbit.Prelude
import Urbit.Time
import Urbit.Vere.Ames import Urbit.Vere.Ames
import Urbit.Vere.Log
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Control.Concurrent (runInBoundThread) import Control.Concurrent (runInBoundThread)
import Data.LargeWord (LargeKey(..)) import Data.LargeWord (LargeKey(..))
import GHC.Natural (Natural) import GHC.Natural (Natural)
import Network.Socket (tupleToHostAddress) import Network.Socket (tupleToHostAddress)
import Urbit.King.App (HasKingId(..))
import qualified Urbit.Vere.Log as Log import qualified Urbit.EventLog.LMDB as Log
--------------------------------------------------------------------------------
type HasAmes e = (HasLogFunc e, HasNetworkConfig e, HasKingId e)
-- Utils ----------------------------------------------------------------------- -- Utils -----------------------------------------------------------------------
pid :: KingId pid :: KingId
@ -38,6 +43,7 @@ sendEf g w bs = NewtEfSend (0, ()) (EachYes g) bs
data NetworkTestApp = NetworkTestApp data NetworkTestApp = NetworkTestApp
{ _ntaLogFunc :: !LogFunc { _ntaLogFunc :: !LogFunc
, _ntaNetworkConfig :: !NetworkConfig , _ntaNetworkConfig :: !NetworkConfig
, _ntaKingId :: !Word16
} }
makeLenses ''NetworkTestApp makeLenses ''NetworkTestApp
@ -48,67 +54,81 @@ instance HasLogFunc NetworkTestApp where
instance HasNetworkConfig NetworkTestApp where instance HasNetworkConfig NetworkTestApp where
networkConfigL = ntaNetworkConfig networkConfigL = ntaNetworkConfig
instance HasKingId NetworkTestApp where
kingIdL = ntaKingId
runNetworkApp :: RIO NetworkTestApp a -> IO a runNetworkApp :: RIO NetworkTestApp a -> IO a
runNetworkApp = runRIO NetworkTestApp runNetworkApp = runRIO NetworkTestApp
{ _ntaLogFunc = mkLogFunc l { _ntaLogFunc = mkLogFunc (\_ _ _ _ -> pure ())
, _ntaNetworkConfig = NetworkConfig NMNormal Nothing Nothing Nothing Nothing , _ntaKingId = 34
, _ntaNetworkConfig = NetworkConfig { _ncNetMode = NMNormal
, _ncAmesPort = Nothing
, _ncNoAmes = False
, _ncNoHttp = False
, _ncNoHttps = False
, _ncHttpPort = Nothing
, _ncHttpsPort = Nothing
, _ncLocalPort = Nothing
}
} }
where
l _ _ _ _ = pure ()
runGala :: forall e. (HasLogFunc e, HasNetworkConfig e) runGala
=> Word8 -> RAcquire e (TQueue Ev, EffCb e NewtEf) :: forall e
runGala point = . HasAmes e
do => Word8
q <- newTQueueIO -> RAcquire e (TQueue EvErr, NewtEf -> IO ())
let (_, runAmes) = runGala point = do
ames pid (fromIntegral point) True (writeTQueue q) noStderr env <- ask
cb runAmes que <- newTQueueIO
rio $ cb turfEf let enqueue = \p -> writeTQueue que p $> Intake
pure (q, cb) let (_, runAmes) = ames env (fromIntegral point) True enqueue noStderr
cb <- runAmes
io (cb turfEf)
pure (que, cb)
where where
noStderr _ = pure () noStderr _ = pure ()
waitForPacket :: TQueue Ev -> Bytes -> IO Bool waitForPacket :: TQueue EvErr -> Bytes -> IO Bool
waitForPacket q val = go waitForPacket q val = go
where where
go = go = atomically (readTQueue q) >>= \case
atomically (readTQueue q) >>= \case EvErr (EvBlip (BlipEvNewt (NewtEvBorn (_, ()) ()))) _ -> go
EvBlip (BlipEvNewt (NewtEvBorn (_, ()) ())) -> go EvErr (EvBlip (BlipEvAmes (AmesEvHear () _ bs))) _ -> pure (bs == val)
EvBlip (BlipEvAmes (AmesEvHear () _ bs)) -> pure (bs == val) _ -> pure False
_ -> pure False
runRAcquire :: RAcquire e a -> RIO e a runRAcquire :: RAcquire e a -> RIO e a
runRAcquire acq = rwith acq pure runRAcquire acq = rwith acq pure
sendThread :: EffCb e NewtEf -> (Galaxy, Bytes) -> RAcquire e () sendThread :: (NewtEf -> IO ()) -> (Galaxy, Bytes) -> RAcquire e ()
sendThread cb (to, val) = void $ mkRAcquire start cancel sendThread cb (to, val) = void $ mkRAcquire start cancel
where where
start = async $ forever $ do threadDelay 1_000 start = async $ forever $ do threadDelay 1_000
wen <- io $ now wen <- io $ now
cb (sendEf to wen val) io $ cb (sendEf to wen val)
threadDelay 10_000 threadDelay 10_000
zodSelfMsg :: Property zodSelfMsg :: Property
zodSelfMsg = forAll arbitrary (ioProperty . runNetworkApp . runTest) zodSelfMsg = forAll arbitrary (ioProperty . runNetworkApp . runTest)
where where
runTest :: (HasLogFunc e, HasNetworkConfig e) => Bytes -> RIO e Bool runTest
runTest val = runRAcquire $ do :: (HasLogFunc e, HasNetworkConfig e, HasKingId e) => Bytes -> RIO e Bool
(zodQ, zod) <- runGala 0 runTest val = runRAcquire $ do
() <- sendThread zod (0, val) env <- ask
liftIO (waitForPacket zodQ val) (zodQ, zod) <- runGala 0
() <- sendThread zod (0, val)
liftIO (waitForPacket zodQ val)
twoTalk :: Property twoTalk :: Property
twoTalk = forAll arbitrary (ioProperty . runNetworkApp . runTest) twoTalk = forAll arbitrary (ioProperty . runNetworkApp . runTest)
where where
runTest :: (HasLogFunc e, HasNetworkConfig e) runTest :: (HasLogFunc e, HasNetworkConfig e, HasKingId e)
=> (Word8, Word8, Bytes) -> RIO e Bool => (Word8, Word8, Bytes) -> RIO e Bool
runTest (aliceShip, bobShip, val) = runTest (aliceShip, bobShip, val) =
if aliceShip == bobShip if aliceShip == bobShip
then pure True then pure True
else go aliceShip bobShip val else go aliceShip bobShip val
go :: (HasLogFunc e, HasNetworkConfig e) go :: (HasLogFunc e, HasNetworkConfig e, HasKingId e)
=> Word8 -> Word8 -> Bytes -> RIO e Bool => Word8 -> Word8 -> Bytes -> RIO e Bool
go aliceShip bobShip val = runRAcquire $ do go aliceShip bobShip val = runRAcquire $ do
(aliceQ, alice) <- runGala aliceShip (aliceQ, alice) <- runGala aliceShip

View File

@ -10,9 +10,9 @@ import Test.Tasty
import Test.Tasty.QuickCheck import Test.Tasty.QuickCheck
import Test.Tasty.TH import Test.Tasty.TH
import Urbit.Arvo import Urbit.Arvo
import Urbit.EventLog.LMDB
import Urbit.Noun.Time
import Urbit.Prelude import Urbit.Prelude
import Urbit.Time
import Urbit.Vere.Log
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Control.Concurrent (runInBoundThread, threadDelay) import Control.Concurrent (runInBoundThread, threadDelay)
@ -20,7 +20,7 @@ import Data.LargeWord (LargeKey(..))
import GHC.Natural (Natural) import GHC.Natural (Natural)
import Network.Socket (tupleToHostAddress) import Network.Socket (tupleToHostAddress)
import qualified Urbit.Vere.Log as Log import qualified Urbit.EventLog.LMDB as Log
-- Utils ----------------------------------------------------------------------- -- Utils -----------------------------------------------------------------------

View File

@ -9,37 +9,36 @@ import Test.Tasty
import Test.Tasty.QuickCheck import Test.Tasty.QuickCheck
import Test.Tasty.TH import Test.Tasty.TH
import Urbit.Arvo import Urbit.Arvo
import Urbit.EventLog.LMDB
import Urbit.Noun import Urbit.Noun
import Urbit.Noun.Time
import Urbit.Prelude import Urbit.Prelude
import Urbit.Time
import Urbit.Vere.Behn import Urbit.Vere.Behn
import Urbit.Vere.Log
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Control.Concurrent (runInBoundThread, threadDelay) import Control.Concurrent (runInBoundThread, threadDelay)
import Data.LargeWord (LargeKey(..)) import Data.LargeWord (LargeKey(..))
import GHC.Natural (Natural) import GHC.Natural (Natural)
import Network.Socket (tupleToHostAddress) import Network.Socket (tupleToHostAddress)
import Urbit.King.App (runApp) import Urbit.King.App (runKingEnvNoLog, HasKingId(..))
import qualified Urbit.Time as Time import qualified Urbit.EventLog.LMDB as Log
import qualified Urbit.Vere.Log as Log import qualified Urbit.Noun.Time as Time
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
king :: KingId
king = KingId 0
-- TODO Timers always fire immediatly. Something is wrong! -- TODO Timers always fire immediatly. Something is wrong!
timerFires :: Property timerFires :: Property
timerFires = forAll arbitrary (ioProperty . runApp . runTest) timerFires = forAll arbitrary (ioProperty . runKingEnvNoLog . runTest)
where where
runTest :: () -> RIO e Bool runTest :: HasKingId e => () -> RIO e Bool
runTest () = do runTest () = do
envr <- ask
king <- fromIntegral <$> view kingIdL
q <- newTQueueIO q <- newTQueueIO
rwith (liftAcquire $ snd $ behn king (writeTQueue q)) $ \cb -> do rwith (liftAcquire $ behn envr (writeTQueue q)) $ \cb -> do
cb (BehnEfDoze (king, ()) (Just (2^20))) io $ cb (BehnEfDoze (king, ()) (Just (2^20)))
t <- atomically $ readTQueue q t <- atomically $ readTQueue q
pure True pure True

View File

@ -7,15 +7,15 @@ import Test.QuickCheck hiding ((.&.))
import Test.Tasty import Test.Tasty
import Test.Tasty.QuickCheck import Test.Tasty.QuickCheck
import Test.Tasty.TH import Test.Tasty.TH
import Urbit.EventLog.LMDB
import Urbit.Prelude import Urbit.Prelude
import Urbit.Vere.Log
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Control.Concurrent (runInBoundThread, threadDelay) import Control.Concurrent (runInBoundThread, threadDelay)
import Data.LargeWord (LargeKey(..)) import Data.LargeWord (LargeKey(..))
import GHC.Natural (Natural) import GHC.Natural (Natural)
import qualified Urbit.Vere.Log as Log import qualified Urbit.EventLog.LMDB as Log
-- Sum Types ------------------------------------------------------------------- -- Sum Types -------------------------------------------------------------------

View File

@ -7,16 +7,16 @@ import Test.QuickCheck hiding ((.&.))
import Test.Tasty import Test.Tasty
import Test.Tasty.QuickCheck import Test.Tasty.QuickCheck
import Test.Tasty.TH import Test.Tasty.TH
import Urbit.EventLog.LMDB
import Urbit.Prelude import Urbit.Prelude
import Urbit.Vere.Log
import Urbit.Vere.Pier.Types import Urbit.Vere.Pier.Types
import Control.Concurrent (runInBoundThread, threadDelay) import Control.Concurrent (runInBoundThread, threadDelay)
import Data.LargeWord (LargeKey(..)) import Data.LargeWord (LargeKey(..))
import GHC.Natural (Natural) import GHC.Natural (Natural)
import Urbit.King.App (App, runApp) import Urbit.King.App (KingEnv, runKingEnvNoLog)
import qualified Urbit.Vere.Log as Log import qualified Urbit.EventLog.LMDB as Log
-- Utils ----------------------------------------------------------------------- -- Utils -----------------------------------------------------------------------
@ -42,13 +42,13 @@ data Db = Db LogIdentity [ByteString] (Map Word64 ByteString)
addEvents :: Db -> [ByteString] -> Db addEvents :: Db -> [ByteString] -> Db
addEvents (Db id evs efs) new = Db id (evs <> new) efs addEvents (Db id evs efs) new = Db id (evs <> new) efs
readDb :: EventLog -> RIO App Db readDb :: EventLog -> RIO KingEnv Db
readDb log = do readDb log = do
events <- runConduit (streamEvents log 1 .| consume) events <- runConduit (streamEvents log 1 .| consume)
effects <- runConduit (streamEffectsRows log 0 .| consume) effects <- runConduit (streamEffectsRows log 0 .| consume)
pure $ Db (Log.identity log) events (mapFromList effects) pure $ Db (Log.identity log) events (mapFromList effects)
withDb :: FilePath -> Db -> (EventLog -> RIO App a) -> RIO App a withDb :: FilePath -> Db -> (EventLog -> RIO KingEnv a) -> RIO KingEnv a
withDb dir (Db dId dEvs dFx) act = do withDb dir (Db dId dEvs dFx) act = do
rwith (Log.new dir dId) $ \log -> do rwith (Log.new dir dId) $ \log -> do
Log.appendEvents log (fromList dEvs) Log.appendEvents log (fromList dEvs)
@ -58,10 +58,13 @@ withDb dir (Db dId dEvs dFx) act = do
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
runApp :: RIO KingEnv a -> IO a
runApp = runKingEnvNoLog
tryReadIdentity :: Property tryReadIdentity :: Property
tryReadIdentity = forAll arbitrary (ioProperty . runApp . runTest) tryReadIdentity = forAll arbitrary (ioProperty . runApp . runTest)
where where
runTest :: LogIdentity -> RIO App Bool runTest :: LogIdentity -> RIO KingEnv Bool
runTest ident = do runTest ident = do
env <- ask env <- ask
io $ runInBoundThread $ runRIO env $ io $ runInBoundThread $ runRIO env $
@ -77,7 +80,7 @@ tryReadIdentity = forAll arbitrary (ioProperty . runApp . runTest)
tryReadDatabase :: Property tryReadDatabase :: Property
tryReadDatabase = forAll arbitrary (ioProperty . runApp . runTest) tryReadDatabase = forAll arbitrary (ioProperty . runApp . runTest)
where where
runTest :: Db -> RIO App Bool runTest :: Db -> RIO KingEnv Bool
runTest db = do runTest db = do
env <- ask env <- ask
io $ runInBoundThread $ runRIO env $ io $ runInBoundThread $ runRIO env $
@ -89,7 +92,7 @@ tryReadDatabase = forAll arbitrary (ioProperty . runApp . runTest)
tryReadDatabaseFuzz :: Property tryReadDatabaseFuzz :: Property
tryReadDatabaseFuzz = forAll arbitrary (ioProperty . runApp . runTest) tryReadDatabaseFuzz = forAll arbitrary (ioProperty . runApp . runTest)
where where
runTest :: Db -> RIO App Bool runTest :: Db -> RIO KingEnv Bool
runTest db = do runTest db = do
env <- ask env <- ask
io $ runInBoundThread $ runRIO env $ io $ runInBoundThread $ runRIO env $
@ -106,7 +109,7 @@ tryReadDatabaseFuzz = forAll arbitrary (ioProperty . runApp . runTest)
tryAppend :: Property tryAppend :: Property
tryAppend = forAll arbitrary (ioProperty . runApp . runTest) tryAppend = forAll arbitrary (ioProperty . runApp . runTest)
where where
runTest :: ([ByteString], Db) -> RIO App Bool runTest :: ([ByteString], Db) -> RIO KingEnv Bool
runTest (extra, db) = do runTest (extra, db) = do
env <- ask env <- ask
io $ runInBoundThread $ runRIO env $ io $ runInBoundThread $ runRIO env $
@ -123,7 +126,7 @@ tryAppend = forAll arbitrary (ioProperty . runApp . runTest)
tryAppendHuge :: Property tryAppendHuge :: Property
tryAppendHuge = forAll arbitrary (ioProperty . runApp . runTest) tryAppendHuge = forAll arbitrary (ioProperty . runApp . runTest)
where where
runTest :: ([ByteString], Db) -> RIO App Bool runTest :: ([ByteString], Db) -> RIO KingEnv Bool
runTest (extra, db) = do runTest (extra, db) = do
env <- ask env <- ask
io $ runInBoundThread $ runRIO env $ do io $ runInBoundThread $ runRIO env $ do

3
pkg/hs/urbit-noun-core/.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
.stack-work
*.cabal
test/gold/*.writ

View File

@ -0,0 +1,21 @@
The MIT License (MIT)
Copyright (c) 2016 urbit
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

View File

@ -0,0 +1,71 @@
name: urbit-noun-core
version: 0.10.4
license: MIT
license-file: LICENSE
library:
source-dirs: lib
ghc-options:
- -fwarn-incomplete-patterns
- -fwarn-unused-binds
- -fwarn-unused-imports
- -Werror
- -O2
dependencies:
- base
- QuickCheck
- ghc-prim
- hashable
- urbit-atom
- classy-prelude
- bytestring
- hashtables
- vector
- integer-gmp
- template-haskell
default-extensions:
- ApplicativeDo
- BangPatterns
- BlockArguments
- ConstraintKinds
- DataKinds
- DefaultSignatures
- DeriveAnyClass
- DeriveDataTypeable
- DeriveFoldable
- DeriveGeneric
- DeriveTraversable
- DerivingStrategies
- EmptyCase
- EmptyDataDecls
- FlexibleContexts
- FlexibleInstances
- FunctionalDependencies
- GADTs
- GeneralizedNewtypeDeriving
- LambdaCase
- MagicHash
- MultiParamTypeClasses
- NamedFieldPuns
- NoImplicitPrelude
- NumericUnderscores
- OverloadedStrings
- PackageImports
- PartialTypeSignatures
- PatternSynonyms
- QuasiQuotes
- Rank2Types
- RankNTypes
- RecordWildCards
- ScopedTypeVariables
- StandaloneDeriving
- TemplateHaskell
- TupleSections
- TypeApplications
- TypeFamilies
- TypeOperators
- UnboxedTuples
- UnicodeSyntax
- ViewPatterns

3
pkg/hs/urbit-noun/.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
.stack-work
*.cabal
test/gold/*.writ

21
pkg/hs/urbit-noun/LICENSE Normal file
View File

@ -0,0 +1,21 @@
The MIT License (MIT)
Copyright (c) 2016 urbit
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

View File

@ -713,6 +713,7 @@ instance (FromNoun a, FromNoun b) => FromNoun (Each a b) where
1 -> named "|" (EachNo <$> parseNoun v) 1 -> named "|" (EachNo <$> parseNoun v)
n -> fail ("Each has invalid head-atom: " <> show n) n -> fail ("Each has invalid head-atom: " <> show n)
-- Tuple Conversions ----------------------------------------------------------- -- Tuple Conversions -----------------------------------------------------------
instance ToNoun () where instance ToNoun () where

View File

@ -7,18 +7,32 @@ module Urbit.Noun.Tank where
import ClassyPrelude import ClassyPrelude
import Urbit.Noun.Conversions import Urbit.Noun.Conversions
import Urbit.Noun.TH import Urbit.Noun.TH
import Urbit.Noun.Convert
import Urbit.Noun.Core
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
type Tang = [Tank] type Tang = [Tank]
data Tank data TankTree
= Leaf Tape = Leaf Tape
| Plum Plum | Plum Plum
| Palm (Tape, Tape, Tape, Tape) [Tank] | Palm (Tape, Tape, Tape, Tape) [TankTree]
| Rose (Tape, Tape, Tape) [Tank] | Rose (Tape, Tape, Tape) [TankTree]
deriving (Eq, Ord, Show) deriving (Eq, Ord, Show)
newtype Tank = Tank { tankTree :: TankTree }
deriving newtype (Eq, Ord, Show)
instance ToNoun Tank where
toNoun (Tank t) = toNoun t
instance FromNoun Tank where
parseNoun n@(Atom _) = do
Cord txt <- parseNoun n
pure $ Tank $ Leaf $ Tape txt
parseNoun n = Tank <$> parseNoun n
data WideFmt = WideFmt { delimit :: Cord, enclose :: Maybe (Cord, Cord) } data WideFmt = WideFmt { delimit :: Cord, enclose :: Maybe (Cord, Cord) }
deriving (Eq, Ord, Show) deriving (Eq, Ord, Show)
@ -39,7 +53,7 @@ data PlumTree
deriveNoun ''WideFmt deriveNoun ''WideFmt
deriveNoun ''TallFmt deriveNoun ''TallFmt
deriveNoun ''PlumFmt deriveNoun ''PlumFmt
deriveNoun ''Tank deriveNoun ''TankTree
deriveNoun ''PlumTree deriveNoun ''PlumTree
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
@ -51,7 +65,7 @@ data WashCfg = WashCfg
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
wash :: WashCfg -> Tank -> Wall wash :: WashCfg -> TankTree -> Wall
wash _cfg t = [ram t] wash _cfg t = [ram t]
-- win :: WashCfg -> Tank -> Wall -- win :: WashCfg -> Tank -> Wall
@ -60,7 +74,7 @@ wash _cfg t = [ram t]
flat :: Plum -> Tape flat :: Plum -> Tape
flat = Tape . tshow flat = Tape . tshow
ram :: Tank -> Tape ram :: TankTree -> Tape
ram = \case ram = \case
Leaf tape -> tape Leaf tape -> tape
Plum plum -> flat plum Plum plum -> flat plum

View File

@ -2,16 +2,21 @@
TODO This is slow. TODO This is slow.
-} -}
module Urbit.Time where module Urbit.Noun.Time where
import Control.Lens import Control.Lens
import Prelude import Prelude
import Data.Bits (shiftL, shiftR) import Data.Bits (shiftL, shiftR, (.&.))
import Data.Time.Clock (DiffTime, UTCTime) import Data.List (intercalate)
import Data.Time.Calendar (toGregorian)
import Data.Time.Clock (DiffTime, UTCTime(..))
import Data.Time.Clock (diffTimeToPicoseconds, picosecondsToDiffTime) import Data.Time.Clock (diffTimeToPicoseconds, picosecondsToDiffTime)
import Data.Time.Clock.System (SystemTime(..), getSystemTime) import Data.Time.Clock.System (SystemTime(..), getSystemTime)
import Data.Time.Clock.System (systemToUTCTime, utcToSystemTime) import Data.Time.Clock.System (systemToUTCTime, utcToSystemTime)
import Data.Time.LocalTime (TimeOfDay(..), timeToTimeOfDay)
import Data.Word (Word64)
import Text.Printf (printf)
import Urbit.Noun (FromNoun, ToNoun) import Urbit.Noun (FromNoun, ToNoun)
@ -26,12 +31,47 @@ newtype Unix = Unix { _sinceUnixEpoch :: Gap }
newtype Wen = Wen { _sinceUrbitEpoch :: Gap } newtype Wen = Wen { _sinceUrbitEpoch :: Gap }
deriving newtype (Eq, Ord, Show, Num, ToNoun, FromNoun) deriving newtype (Eq, Ord, Show, Num, ToNoun, FromNoun)
newtype Date = MkDate { _dateWen :: Wen }
deriving newtype (Eq, Ord, Num, ToNoun, FromNoun)
-- Lenses ----------------------------------------------------------------------
-- Record Lenses ---------------------------------------------------------------
makeLenses ''Gap makeLenses ''Gap
makeLenses ''Unix makeLenses ''Unix
makeLenses ''Wen makeLenses ''Wen
makeLenses ''Date
-- Instances -------------------------------------------------------------------
instance Show Date where
show (MkDate wen) = if fs == 0
then printf "~%i.%u.%u..%02u.%02u.%02u" y m d h min s
else printf "~%i.%u.%u..%02u.%02u.%02u..%s" y m d h min s (showGap fs)
where
utc = wen ^. systemTime . to systemToUTCTime
(y, m, d) = toGregorian (utctDay utc)
TimeOfDay h min (floor -> s::Int) = timeToTimeOfDay (utctDayTime utc)
fs = (wen ^. wenFracto . to (fromIntegral @Integer @Word64))
wenFracto :: Lens' Wen Integer
wenFracto = sinceUrbitEpoch . fractoSecs
showGap :: Word64 -> String
showGap gap = intercalate "." (printf "%04x" <$> bs)
where
bs = reverse $ dropWhile (== 0) [b4, b3, b2, b1]
b4 = takeBits 16 gap
b3 = takeBits 16 (shiftR gap 16)
b2 = takeBits 16 (shiftR gap 32)
b1 = takeBits 16 (shiftR gap 48)
takeBits :: Int -> Word64 -> Word64
takeBits wid wor = wor .&. (shiftL 1 wid - 1)
-- Conversion Lenses -----------------------------------------------------------
diffTime :: Iso' Gap DiffTime diffTime :: Iso' Gap DiffTime
diffTime = iso fromGap toGap diffTime = iso fromGap toGap

View File

@ -0,0 +1,73 @@
name: urbit-noun
version: 0.10.4
license: MIT
license-file: LICENSE
library:
source-dirs: lib
ghc-options:
- -fwarn-incomplete-patterns
- -fwarn-unused-binds
- -fwarn-unused-imports
- -Werror
- -O2
dependencies:
- base
- classy-prelude
- ghc-prim
- largeword
- lens
- murmur3
- regex-tdfa
- regex-tdfa-text
- rio
- text
- time
- urbit-atom
- urbit-noun-core
default-extensions:
- ApplicativeDo
- BangPatterns
- BlockArguments
- ConstraintKinds
- DataKinds
- DefaultSignatures
- DeriveAnyClass
- DeriveDataTypeable
- DeriveFoldable
- DeriveGeneric
- DeriveTraversable
- DerivingStrategies
- EmptyCase
- EmptyDataDecls
- FlexibleContexts
- FlexibleInstances
- FunctionalDependencies
- GADTs
- GeneralizedNewtypeDeriving
- LambdaCase
- MagicHash
- MultiParamTypeClasses
- NamedFieldPuns
- NoImplicitPrelude
- NumericUnderscores
- OverloadedStrings
- PackageImports
- PartialTypeSignatures
- PatternSynonyms
- QuasiQuotes
- Rank2Types
- RankNTypes
- RecordWildCards
- ScopedTypeVariables
- StandaloneDeriving
- TemplateHaskell
- TupleSections
- TypeApplications
- TypeFamilies
- TypeOperators
- UnboxedTuples
- UnicodeSyntax
- ViewPatterns

3
pkg/hs/urbit-termsize/.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
.stack-work/
urbit-termsize.cabal
*~

View File

@ -0,0 +1,21 @@
The MIT License (MIT)
Copyright (c) 2016 urbit
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

View File

@ -0,0 +1,13 @@
module Main where
import Prelude
import Urbit.TermSize (liveTermSize)
import System.IO (getLine)
main :: IO ()
main = do
init <- liveTermSize (putStrLn . ("New Size: " <>) . show)
putStrLn ("Initial Size: " <> show init)
_ <- getLine
pure ()

View File

@ -0,0 +1,40 @@
{-# LANGUAGE DeriveAnyClass #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE RecordWildCards #-}
module Urbit.TermSize
( TermSize(..)
, termSize
, liveTermSize
)
where
import Prelude
import Data.Functor ((<&>))
import System.Console.Terminal.Size (Window(..), size)
import qualified System.Posix.Signals as Sys
import qualified System.Posix.Signals.Exts as Sys
-- Types -----------------------------------------------------------------------
data TermSize = TermSize
{ tsWide :: !Word
, tsTall :: !Word
}
deriving (Eq, Ord, Show)
-- Utilities -------------------------------------------------------------------
termSize :: IO TermSize
termSize = size <&> \case
Nothing -> TermSize 80 24
Just (Window {..}) -> TermSize width height
liveTermSize :: (TermSize -> IO ()) -> IO TermSize
liveTermSize cb = do
Sys.installHandler Sys.sigWINCH (Sys.Catch (termSize >>= cb)) Nothing
termSize

View File

@ -0,0 +1,25 @@
name: urbit-termsize
version: 0.1.0
license: MIT
license-file: LICENSE
dependencies:
- base
- terminal-size
- unix
ghc-options:
- -fwarn-incomplete-patterns
- -fwarn-unused-binds
- -fwarn-unused-imports
- -O2
library:
source-dirs: lib
executables:
urbit-test-termsize-updates:
main: Main.hs
source-dirs: app
dependencies:
- urbit-termsize

View File

@ -2,7 +2,7 @@ include config.mk
jets = jets/tree.c $(wildcard jets/*/*.c) jets = jets/tree.c $(wildcard jets/*/*.c)
noun = $(wildcard noun/*.c) noun = $(wildcard noun/*.c)
vere = $(wildcard vere/*.c) vere = $(wildcard vere/*.c) $(wildcard vere/*/*.c)
daemon = $(wildcard daemon/*.c) daemon = $(wildcard daemon/*.c)
worker = $(wildcard worker/*.c) worker = $(wildcard worker/*.c)
tests = $(wildcard tests/*.c) tests = $(wildcard tests/*.c)

8
pkg/urbit/configure vendored
View File

@ -2,11 +2,11 @@
set -e set -e
URBIT_VERSION="0.10.7" URBIT_VERSION="0.10.8"
deps=" \ deps=" \
curl gmp sigsegv argon2 ed25519 ent h2o scrypt sni uv murmur3 secp256k1 \ curl gmp sigsegv argon2 ed25519 ent h2o scrypt uv murmur3 secp256k1 \
softfloat3 ncurses ssl crypto z lmdb ge-additions aes_siv \ softfloat3 ssl crypto z lmdb ge-additions aes_siv pthread \
" "
headers=" \ headers=" \

View File

@ -9,9 +9,7 @@
#include <uv.h> #include <uv.h>
#include <sigsegv.h> #include <sigsegv.h>
#include <stdlib.h> #include <stdlib.h>
#include <ncurses/curses.h>
#include <termios.h> #include <termios.h>
#include <ncurses/term.h>
#include <dirent.h> #include <dirent.h>
#include <openssl/conf.h> #include <openssl/conf.h>
#include <openssl/engine.h> #include <openssl/engine.h>
@ -20,6 +18,7 @@
#include <h2o.h> #include <h2o.h>
#include <curl/curl.h> #include <curl/curl.h>
#include <argon2.h> #include <argon2.h>
#include <lmdb.h>
#define U3_GLOBAL #define U3_GLOBAL
#define C3_GLOBAL #define C3_GLOBAL
@ -97,9 +96,17 @@ _main_getopt(c3_i argc, c3_c** argv)
u3_Host.ops_u.kno_w = DefaultKernel; u3_Host.ops_u.kno_w = DefaultKernel;
while ( -1 != (ch_i=getopt(argc, argv, while ( -1 != (ch_i=getopt(argc, argv,
"G:J:B:K:A:H:I:C:w:u:e:F:k:p:LljacdgqstvxPDRS")) ) "X:Y:G:J:B:K:A:H:I:C:w:u:e:F:k:n:p:r:LljacdgqstvxPDRS")) )
{ {
switch ( ch_i ) { switch ( ch_i ) {
case 'X': {
u3_Host.ops_u.pek_c = strdup(optarg);
break;
}
case 'Y': {
u3_Host.ops_u.puk_c = strdup(optarg);
break;
}
case 'J': { case 'J': {
u3_Host.ops_u.lit_c = strdup(optarg); u3_Host.ops_u.lit_c = strdup(optarg);
break; break;
@ -162,6 +169,10 @@ _main_getopt(c3_i argc, c3_c** argv)
u3_Host.ops_u.key_c = strdup(optarg); u3_Host.ops_u.key_c = strdup(optarg);
break; break;
} }
case 'n': {
u3_Host.ops_u.til_c = strdup(optarg);
break;
}
case 'p': { case 'p': {
if ( c3n == _main_readw(optarg, 65536, &arg_w) ) { if ( c3n == _main_readw(optarg, 65536, &arg_w) ) {
return c3n; return c3n;
@ -172,6 +183,10 @@ _main_getopt(c3_i argc, c3_c** argv)
u3_Host.ops_u.rep = c3y; u3_Host.ops_u.rep = c3y;
return c3y; return c3y;
} }
case 'r': {
u3_Host.ops_u.roc_c = strdup(optarg);
break;
}
case 'L': { u3_Host.ops_u.net = c3n; break; } case 'L': { u3_Host.ops_u.net = c3n; break; }
case 'l': { u3_Host.ops_u.lit = c3y; break; } case 'l': { u3_Host.ops_u.lit = c3y; break; }
case 'j': { u3_Host.ops_u.tra = c3y; break; } case 'j': { u3_Host.ops_u.tra = c3y; break; }
@ -398,7 +413,9 @@ u3_ve_usage(c3_i argc, c3_c** argv)
"-u url URL from which to download pill\n", "-u url URL from which to download pill\n",
"-v Verbose\n", "-v Verbose\n",
"-w name Boot as ~name\n", "-w name Boot as ~name\n",
"-X path Scry, jam to file, then exit\n"
"-x Exit immediately\n", "-x Exit immediately\n",
"-Y file Optional name of jamfile (for -X)\n"
"\n", "\n",
"Development Usage:\n", "Development Usage:\n",
" To create a development ship, use a fakezod:\n", " To create a development ship, use a fakezod:\n",
@ -449,7 +466,6 @@ report(void)
(libsigsegv_version >> 8) & 0xff, (libsigsegv_version >> 8) & 0xff,
libsigsegv_version & 0xff); libsigsegv_version & 0xff);
printf("openssl: %s\n", SSLeay_version(SSLEAY_VERSION)); printf("openssl: %s\n", SSLeay_version(SSLEAY_VERSION));
printf("curses: %s\n", curses_version());
printf("libuv: %s\n", uv_version_string()); printf("libuv: %s\n", uv_version_string());
printf("libh2o: %d.%d.%d\n", printf("libh2o: %d.%d.%d\n",
H2O_LIBRARY_VERSION_MAJOR, H2O_LIBRARY_VERSION_MAJOR,
@ -474,19 +490,7 @@ _stop_exit(c3_i int_i)
// explicit fprintf to avoid allocation in u3l_log // explicit fprintf to avoid allocation in u3l_log
// //
fprintf(stderr, "\r\n[received keyboard stop signal, exiting]\r\n"); fprintf(stderr, "\r\n[received keyboard stop signal, exiting]\r\n");
u3_daemon_bail(); u3_king_bail();
}
/* _stop_signal(): handle termination signal.
*/
static void
_stop_signal(c3_i int_i)
{
// if we have a pier, unmap the event log before dumping core
//
if ( 0 != u3K.len_w ) {
u3_pier_db_shutdown(u3_pier_stub());
}
} }
/* /*
@ -584,7 +588,7 @@ _fork_into_background_process()
static void static void
_stop_on_boot_completed_cb() _stop_on_boot_completed_cb()
{ {
u3_pier_exit(u3_pier_stub()); u3_king_exit();
} }
c3_i c3_i
@ -603,12 +607,6 @@ main(c3_i argc,
u3_Host.wrk_c = c3_malloc(worker_exe_len); u3_Host.wrk_c = c3_malloc(worker_exe_len);
snprintf(u3_Host.wrk_c, worker_exe_len, "%s-worker", argv[0]); snprintf(u3_Host.wrk_c, worker_exe_len, "%s-worker", argv[0]);
// Set TERMINFO_DIRS environment variable
c3_i terminfo_len = 1 + strlen(argv[0]) + strlen("-terminfo");
c3_c terminfo_dir[terminfo_len];
snprintf(terminfo_dir, terminfo_len, "%s-terminfo", argv[0]);
setenv("TERMINFO_DIRS", terminfo_dir, 1);
if ( c3y == u3_Host.ops_u.dem ) { if ( c3y == u3_Host.ops_u.dem ) {
_fork_into_background_process(); _fork_into_background_process();
} }
@ -651,10 +649,6 @@ main(c3_i argc,
// //
signal(SIGTSTP, _stop_exit); signal(SIGTSTP, _stop_exit);
// Cleanup on SIGABRT.
//
signal(SIGABRT, _stop_signal);
printf("~\n"); printf("~\n");
// printf("welcome.\n"); // printf("welcome.\n");
printf("urbit %s\n", URBIT_VERSION); printf("urbit %s\n", URBIT_VERSION);
@ -768,7 +762,7 @@ main(c3_i argc,
exit(1); exit(1);
} }
u3_daemon_commence(); u3_king_commence();
// uninitialize curl // uninitialize curl
// //

View File

@ -41,7 +41,7 @@
/* Stub. /* Stub.
*/ */
# define c3_stub (assert(!"stub"), 0) # define c3_stub c3_assert(!"stub")
/* Size in words. /* Size in words.
*/ */

View File

@ -248,6 +248,7 @@
# define c3__cow c3_s3('c','o','w') # define c3__cow c3_s3('c','o','w')
# define c3__cpu c3_s3('c','p','u') # define c3__cpu c3_s3('c','p','u')
# define c3__crad c3_s4('c','r','a','d') # define c3__crad c3_s4('c','r','a','d')
# define c3__cram c3_s4('c','r','a','m')
# define c3__crap c3_s4('c','r','a','p') # define c3__crap c3_s4('c','r','a','p')
# define c3__cret c3_s4('c','r','e','t') # define c3__cret c3_s4('c','r','e','t')
# define c3__crib c3_s4('c','r','i','b') # define c3__crib c3_s4('c','r','i','b')
@ -365,6 +366,7 @@
# define c3__dumb c3_s4('d','u','m','b') # define c3__dumb c3_s4('d','u','m','b')
# define c3__dump c3_s4('d','u','m','p') # define c3__dump c3_s4('d','u','m','p')
# define c3__dust c3_s4('d','u','s','t') # define c3__dust c3_s4('d','u','s','t')
# define c3__e c3_s1('e')
# define c3__earl c3_s4('e','a','r','l') # define c3__earl c3_s4('e','a','r','l')
# define c3__east c3_s4('e','a','s','t') # define c3__east c3_s4('e','a','s','t')
# define c3__echo c3_s4('e','c','h','o') # define c3__echo c3_s4('e','c','h','o')
@ -928,6 +930,7 @@
# define c3__revo c3_s4('r','e','v','o') # define c3__revo c3_s4('r','e','v','o')
# define c3__rin c3_s3('r','i','n') # define c3__rin c3_s3('r','i','n')
# define c3__ring c3_s4('r','i','n','g') # define c3__ring c3_s4('r','i','n','g')
# define c3__ripe c3_s4('r','i','p','e')
# define c3__rite c3_s4('r','i','t','e') # define c3__rite c3_s4('r','i','t','e')
# define c3__rock c3_s4('r','o','c','k') # define c3__rock c3_s4('r','o','c','k')
# define c3__roll c3_s4('r','o','l','l') # define c3__roll c3_s4('r','o','l','l')
@ -1062,6 +1065,7 @@
# define c3__sunt c3_s4('s','u','n','t') # define c3__sunt c3_s4('s','u','n','t')
# define c3__sure c3_s4('s','u','r','e') # define c3__sure c3_s4('s','u','r','e')
# define c3__susp c3_s4('s','u','s','p') # define c3__susp c3_s4('s','u','s','p')
# define c3__swap c3_s4('s','w','a','p')
# define c3__sym c3_s3('s','y','m') # define c3__sym c3_s3('s','y','m')
# define c3__sync c3_s4('s','y','n','c') # define c3__sync c3_s4('s','y','n','c')
# define c3__sys c3_s3('s','y','s') # define c3__sys c3_s3('s','y','s')
@ -1206,6 +1210,7 @@
# define c3__wack c3_s4('w','a','c','k') # define c3__wack c3_s4('w','a','c','k')
# define c3__wail c3_s4('w','a','i','l') # define c3__wail c3_s4('w','a','i','l')
# define c3__wake c3_s4('w','a','k','e') # define c3__wake c3_s4('w','a','k','e')
# define c3__walk c3_s4('w','a','l','k')
# define c3__wamp c3_s4('w','a','m','p') # define c3__wamp c3_s4('w','a','m','p')
# define c3__want c3_s4('w','a','n','t') # define c3__want c3_s4('w','a','n','t')
# define c3__warm c3_s4('w','a','r','m') # define c3__warm c3_s4('w','a','r','m')

Some files were not shown because too many files have changed in this diff Show More