2019-09-20 01:40:23 +03:00
|
|
|
{-# OPTIONS_GHC -Wwarn #-}
|
|
|
|
module Vere.Dawn where
|
|
|
|
|
|
|
|
import Arvo.Common
|
2019-09-20 20:41:58 +03:00
|
|
|
import Arvo.Event hiding (Address)
|
2019-09-20 01:40:23 +03:00
|
|
|
import Azimuth.Azimuth
|
2019-09-21 02:10:03 +03:00
|
|
|
import UrbitPrelude hiding (Call, rights, to)
|
2019-09-20 01:40:23 +03:00
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
import Data.Solidity.Abi.Codec (encode)
|
|
|
|
import Network.Ethereum.Account
|
2019-09-20 01:40:23 +03:00
|
|
|
import Network.Ethereum.Api.Eth
|
|
|
|
import Network.Ethereum.Api.Provider
|
|
|
|
import Network.Ethereum.Api.Types hiding (blockNumber)
|
|
|
|
import Network.Ethereum.Web3
|
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
|
2019-09-20 01:40:23 +03:00
|
|
|
import Data.Text (splitOn)
|
|
|
|
|
2019-09-25 00:01:39 +03:00
|
|
|
import qualified Crypto.ECC.Edwards25519 as Ed
|
|
|
|
import qualified Crypto.Error as Ed
|
|
|
|
import qualified Data.ByteArray as BA
|
|
|
|
import qualified Data.Map.Strict as M
|
|
|
|
import qualified Network.Ethereum.Ens as Ens
|
|
|
|
import qualified Urbit.Ob as Ob
|
2019-09-20 01:40:23 +03:00
|
|
|
|
|
|
|
{-TODOs:
|
|
|
|
|
|
|
|
- Dawn takes a NounMap instead of a Map. Need a conversion function.
|
|
|
|
|
2019-09-20 02:51:34 +03:00
|
|
|
- The Haskell Dawn structure as it exists right now isn't right? It can't
|
|
|
|
parse a real %dawn event in the event browser.
|
|
|
|
|
2019-09-20 01:40:23 +03:00
|
|
|
-}
|
|
|
|
|
|
|
|
|
|
|
|
-- During boot, use the infura provider
|
|
|
|
provider = HttpProvider
|
|
|
|
"https://mainnet.infura.io/v3/196a7f37c7d54211b4a07904ec73ad87"
|
|
|
|
|
2019-09-20 20:41:58 +03:00
|
|
|
|
|
|
|
--azimuthContract = "0x223c067F8CF28ae173EE5CafEa60cA44C335fecB"
|
2019-09-20 01:40:23 +03:00
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
-- Conversion Utilities --------------------------------------------------------
|
|
|
|
|
|
|
|
bsToAtom :: ByteString -> Atom
|
|
|
|
bsToAtom x = x ^. from atomBytes
|
|
|
|
|
|
|
|
-- Takes the web3's bytes representation and changes the endianness.
|
|
|
|
bytes32ToBS :: BytesN 32 -> ByteString
|
|
|
|
bytes32ToBS = reverse . BA.pack . BA.unpack
|
|
|
|
|
2019-09-20 02:51:34 +03:00
|
|
|
bytes32ToAtom :: BytesN 32 -> Atom
|
2019-09-21 02:10:03 +03:00
|
|
|
bytes32ToAtom = bsToAtom . bytes32ToBS
|
2019-09-20 02:51:34 +03:00
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
-- web3 doesn't export unAddress.
|
|
|
|
addressToBS :: Address -> ByteString
|
|
|
|
addressToBS = reverse . encode
|
2019-09-20 01:40:23 +03:00
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
addressToAtom = bsToAtom . addressToBS
|
2019-09-20 20:41:58 +03:00
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
-- A Pass is the encryptionKey and authenticationKey concatenated together.
|
|
|
|
passFromEth :: BytesN 32 -> BytesN 32 -> UIntN 32 -> Pass
|
2019-09-25 00:01:39 +03:00
|
|
|
passFromEth enc aut sut | sut /= 1 = error "Invalid crypto suite number"
|
|
|
|
passFromEth enc aut sut = Pass (decode enc) (decode aut)
|
|
|
|
where
|
|
|
|
decode = Ed.throwCryptoError . Ed.pointDecode . bytes32ToBS
|
2019-09-20 20:41:58 +03:00
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
clanFromShip :: Ship -> Ob.Class
|
|
|
|
clanFromShip = Ob.clan . Ob.patp . fromIntegral
|
|
|
|
|
2019-09-24 00:42:19 +03:00
|
|
|
shipSein :: Ship -> Ship
|
|
|
|
shipSein = Ship . fromIntegral . Ob.fromPatp . Ob.sein . Ob.patp . fromIntegral
|
|
|
|
|
2019-09-25 00:01:39 +03:00
|
|
|
|
|
|
|
|
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
-- Data Validation -------------------------------------------------------------
|
|
|
|
|
2019-09-25 00:01:39 +03:00
|
|
|
-- for =(who.seed `@`fix:ex:cub)
|
|
|
|
getFingerprintFromKey :: Ring -> Atom
|
|
|
|
getFingerprintFromKey = undefined
|
|
|
|
|
|
|
|
-- getPassFromKey :: Ring -> Pass
|
|
|
|
-- getPassFromKey (Ring crypt sign) = (Pass pubCrypt pubSign)
|
|
|
|
-- where
|
|
|
|
-- pubCrypt = decode crypt
|
|
|
|
-- pubSign = decode sign
|
|
|
|
-- decode = Ed.throwCryptoError . Ed.pointDecode
|
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
-- Validates the keys, life, discontinuity, etc. If everything is ok, return
|
|
|
|
-- the sponsoring ship for Seed.
|
|
|
|
validateAndGetSponsor :: Seed -> EthPoint -> Either Text Ship
|
|
|
|
validateAndGetSponsor (Seed ship life ring mo) EthPoint{..} = do
|
|
|
|
let clan = clanFromShip ship
|
|
|
|
|
|
|
|
case clan of
|
2019-09-24 00:42:19 +03:00
|
|
|
Ob.Comet -> do
|
|
|
|
-- A comet address is the fingerprint of the keypair
|
|
|
|
-- when (ship /= (x ring.seed)) (Left "todo: key mismatch")
|
|
|
|
-- A comet can never be breached
|
|
|
|
-- when live Left "comet already booted"
|
|
|
|
-- TODO: the parent must be launched check?
|
|
|
|
Right $ shipSein ship
|
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
-- When the ship is a moon, the only requirement is that the parent is
|
|
|
|
-- launched.
|
|
|
|
Ob.Moon -> do
|
|
|
|
Left "todo: moon's parent must be launched"
|
|
|
|
|
2019-09-24 00:42:19 +03:00
|
|
|
-- For Galaxies, Stars and Planets, we do the full checks.
|
2019-09-21 02:10:03 +03:00
|
|
|
_ -> case epNet of
|
|
|
|
Nothing -> Left "ship not keyed"
|
|
|
|
Just (life, pass, contNum, _, _) -> do
|
|
|
|
-- when ( /= pass) $ Left "key mismatch"
|
|
|
|
pure $ Ship 5
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
-- Azimuth Functions -----------------------------------------------------------
|
|
|
|
|
|
|
|
-- Perform a request to azimuth at a certain block number
|
|
|
|
withAzimuth :: Quantity
|
|
|
|
-> Address
|
|
|
|
-> DefaultAccount Web3 a
|
|
|
|
-> Web3 a
|
|
|
|
withAzimuth bloq azimuth action =
|
2019-09-20 01:40:23 +03:00
|
|
|
withAccount () $
|
2019-09-20 20:41:58 +03:00
|
|
|
withParam (to .~ azimuth) $
|
2019-09-21 02:10:03 +03:00
|
|
|
withParam (block .~ (BlockWithNumber bloq))
|
|
|
|
action
|
|
|
|
|
|
|
|
-- In the Hoon implementation, the EthPoint structure has space for the deed
|
|
|
|
-- information, but it immediately punts on this by bunting the deed structure
|
|
|
|
-- instead of making the correct calls. We just do the right thing.
|
|
|
|
--
|
|
|
|
retrievePoint :: Quantity -> Address -> Int -> Web3 (EthPoint)
|
|
|
|
retrievePoint bloq azimuth p =
|
|
|
|
withAzimuth bloq azimuth $ do
|
|
|
|
(owner, managementProxy, spawnProxy, votingProxy, transferProxy)
|
|
|
|
<- rights (fromIntegral p)
|
|
|
|
|
|
|
|
(encryptionKey,
|
|
|
|
authenticationKey,
|
|
|
|
hasSponsor,
|
|
|
|
active,
|
|
|
|
escapeRequested,
|
|
|
|
sponsor,
|
|
|
|
escapeTo,
|
|
|
|
cryptoSuite,
|
|
|
|
keyRevision, continuityNum) <- points (fromIntegral p)
|
|
|
|
|
|
|
|
let escapeState = if escapeRequested
|
|
|
|
then Just $ Ship $ fromIntegral escapeTo
|
|
|
|
else Nothing
|
|
|
|
|
|
|
|
let epOwn = (addressToAtom owner,
|
|
|
|
addressToAtom managementProxy,
|
|
|
|
addressToAtom votingProxy,
|
|
|
|
addressToAtom transferProxy)
|
|
|
|
|
|
|
|
let epNet = if (not active)
|
|
|
|
then Nothing
|
|
|
|
else Just (
|
|
|
|
(fromIntegral keyRevision),
|
|
|
|
(passFromEth encryptionKey authenticationKey cryptoSuite),
|
|
|
|
(fromIntegral continuityNum),
|
|
|
|
(hasSponsor, Ship (fromIntegral sponsor)),
|
|
|
|
escapeState
|
|
|
|
)
|
|
|
|
|
|
|
|
let epKid = case (clanFromShip $ Ship $ fromIntegral p) of
|
|
|
|
Ob.Galaxy -> Just (addressToAtom spawnProxy, None)
|
|
|
|
Ob.Star -> Just (addressToAtom spawnProxy, None)
|
|
|
|
_ -> Nothing
|
|
|
|
|
|
|
|
pure EthPoint{..}
|
|
|
|
|
|
|
|
-- Retrieves information about all the galaxies from Ethereum.
|
|
|
|
retrieveGalaxyTable :: Quantity -> Address -> Web3 (Map Ship (Rift, Life, Pass))
|
|
|
|
retrieveGalaxyTable bloq azimuth =
|
|
|
|
withAzimuth bloq azimuth $ M.fromList <$> mapM getRow [0..5]
|
2019-09-20 01:40:23 +03:00
|
|
|
where
|
|
|
|
getRow idx = do
|
2019-09-25 00:01:39 +03:00
|
|
|
(encryptionKey, authenticationKey, _, _, _, _, _, cryptoSuite,
|
|
|
|
keyRev, continuity) <- points idx
|
|
|
|
pure (fromIntegral idx,
|
|
|
|
(fromIntegral continuity,
|
|
|
|
fromIntegral keyRev,
|
|
|
|
(passFromEth encryptionKey authenticationKey cryptoSuite)))
|
2019-09-20 01:40:23 +03:00
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
-- Reads the three Ames domains from Ethereum.
|
2019-09-20 20:41:58 +03:00
|
|
|
readAmesDomains :: Quantity -> Address -> Web3 ([Turf])
|
|
|
|
readAmesDomains bloq azimuth =
|
2019-09-21 02:10:03 +03:00
|
|
|
withAzimuth bloq azimuth $ mapM getTurf [0..2]
|
2019-09-20 01:40:23 +03:00
|
|
|
where
|
|
|
|
getTurf idx = do
|
|
|
|
str <- dnsDomains idx
|
|
|
|
pure $ Turf $ fmap Cord $ reverse $ splitOn "." str
|
|
|
|
|
|
|
|
{-
|
2019-09-21 02:10:03 +03:00
|
|
|
[%dawn seed sponsor galaxies domains block eth-url]
|
2019-09-20 01:40:23 +03:00
|
|
|
-}
|
|
|
|
|
|
|
|
-- Produces either an error or a validated boot event structure.
|
|
|
|
dawnVent :: Seed -> RIO e (Either Text Dawn)
|
|
|
|
dawnVent (Seed (Ship ship) life ring oaf) = do
|
2019-09-21 02:10:03 +03:00
|
|
|
--
|
|
|
|
-- Everyone needs the Ethereum node instead of just the galaxies.
|
2019-09-20 01:40:23 +03:00
|
|
|
|
|
|
|
hs <- runWeb3' provider $ do
|
|
|
|
-- Block number (dBloq)
|
|
|
|
dBloq <- blockNumber
|
2019-09-21 02:10:03 +03:00
|
|
|
print ("boot: eth block: " ++ (show dBloq))
|
2019-09-20 20:41:58 +03:00
|
|
|
|
|
|
|
azimuth <- withAccount () $ Ens.resolve "azimuth.eth"
|
|
|
|
print ("Azimuth: " ++ (show azimuth))
|
2019-09-20 01:40:23 +03:00
|
|
|
|
2019-09-21 02:10:03 +03:00
|
|
|
-- TODO: We're retrieving point information, but we don't have
|
2019-09-25 00:01:39 +03:00
|
|
|
--
|
|
|
|
-- TODO: Comets don't go through retrievePoint.
|
|
|
|
p <- retrievePoint dBloq azimuth (fromIntegral ship)
|
2019-09-21 02:10:03 +03:00
|
|
|
print $ show p
|
2019-09-20 01:40:23 +03:00
|
|
|
|
2019-09-20 02:51:34 +03:00
|
|
|
-- Retrieve the galaxy table [MUST FIX s/5/255/]
|
2019-09-20 20:41:58 +03:00
|
|
|
-- galaxyTable <- retrieveGalaxyTable dBloq azimuth
|
|
|
|
-- print $ show galaxyTable
|
2019-09-20 01:40:23 +03:00
|
|
|
|
|
|
|
-- Read Ames domains [DONE]
|
2019-09-20 20:41:58 +03:00
|
|
|
-- dTurf <- readAmesDomains dBloq azimuth
|
2019-09-20 01:40:23 +03:00
|
|
|
-- print $ show dTurf
|
|
|
|
|
|
|
|
pure (dBloq)
|
|
|
|
|
|
|
|
print $ show hs
|
|
|
|
|
|
|
|
pure (Left "bad")
|
|
|
|
|