diff --git a/chainweb.cabal b/chainweb.cabal index 6539136bd4..ccd1770444 100644 --- a/chainweb.cabal +++ b/chainweb.cabal @@ -148,6 +148,8 @@ library , Chainweb.BlockHeader.Genesis.QuirkedGasPact5InstantTimedCPM1to9Payload , Chainweb.BlockHeader.Genesis.Testnet040Payload , Chainweb.BlockHeader.Genesis.Testnet041to19Payload + , Chainweb.BlockHeader.Genesis.Testnet060Payload + , Chainweb.BlockHeader.Genesis.Testnet061to19Payload , Chainweb.BlockHeader.Genesis.Mainnet0Payload , Chainweb.BlockHeader.Genesis.Mainnet1Payload , Chainweb.BlockHeader.Genesis.Mainnet2Payload @@ -273,6 +275,7 @@ library , Chainweb.Version.RecapDevelopment , Chainweb.Version.Registry , Chainweb.Version.Testnet04 + , Chainweb.Version.Testnet06 , Chainweb.Version.Utils , Chainweb.WebBlockHeaderDB , Chainweb.WebPactExecutionService diff --git a/cwtools/ea/Ea.hs b/cwtools/ea/Ea.hs index a00b2fe738..fb182dcbf9 100644 --- a/cwtools/ea/Ea.hs +++ b/cwtools/ea/Ea.hs @@ -84,6 +84,7 @@ main = do , pact53Transitionnet , quirkedPact5Instantnet , testnet04 + , testnet06 , mainnet , genTxModules , genCoinV3Payloads @@ -108,6 +109,7 @@ main = do pact53Transitionnet = mkPayloads [pact53TransitionCPM0, pact53TransitionCPMN] quirkedPact5Instantnet = mkPayloads [quirkedPact5InstantCPM0, quirkedPact5InstantCPMN] testnet04 = mkPayloads [testnet040, testnet04N] + testnet06 = mkPayloads [testnet060, testnet06N] mainnet = mkPayloads [ mainnet0 , mainnet1 diff --git a/cwtools/ea/Ea/Genesis.hs b/cwtools/ea/Ea/Genesis.hs index 54aa32b47d..920020137b 100644 --- a/cwtools/ea/Ea/Genesis.hs +++ b/cwtools/ea/Ea/Genesis.hs @@ -36,6 +36,10 @@ module Ea.Genesis , testnet040 , testnet04N +-- * Testnet06 Genesis txs +, testnet060 +, testnet06N + -- * Mainnet Genesis txs , mainnet0 , mainnet1 @@ -77,6 +81,7 @@ import Chainweb.Version.Development import Chainweb.Version.RecapDevelopment import Chainweb.Version.Mainnet import Chainweb.Version.Testnet04 +import Chainweb.Version.Testnet06 -- ---------------------------------------------------------------------- -- -- Genesis Tx Data @@ -372,6 +377,28 @@ testnetAllocations = "pact/genesis/testnet04/allocations.yaml" testnetKeysets :: FilePath testnetKeysets = "pact/genesis/testnet04/keysets.yaml" + +-- ---------------------------------------------------------------------- -- +-- Testnet 06 + +testnet060 :: Genesis +testnet060 = Genesis + { _version = Testnet06 + , _tag = "Testnet06" + , _txChainIds = onlyChainId 0 + , _coinbase = Just "pact/genesis/testnet06/grants0.yaml" + , _keysets = Just "pact/genesis/testnet06/keysets.yaml" + , _allocations = Just "pact/genesis/testnet06/allocations.yaml" + , _namespaces = Just testNs + , _coinContract = [fungibleAssetV1, coinContractV1, gasPayer] + } + +testnet06N :: Genesis +testnet06N = testnet060 + & txChainIds .~ mkChainIdRange 1 19 + & coinbase ?~ "pact/genesis/testnet06/grantsN.yaml" + + -- ---------------------------------------------------------------------- -- -- Mainnet diff --git a/pact/genesis/testnet06/allocations.yaml b/pact/genesis/testnet06/allocations.yaml new file mode 100644 index 0000000000..4780125039 --- /dev/null +++ b/pact/genesis/testnet06/allocations.yaml @@ -0,0 +1,6 @@ +code: |- + (coin.create-allocation-account "allocation00" (time "1900-10-15T18:00:00Z") "allocation00" 1000000.0) + (coin.create-allocation-account "allocation01" (time "2026-01-31T18:00:00Z") "allocation01" 1000000.0) + (coin.create-allocation-account "allocation02" (time "2026-06-31T18:00:00Z") "allocation02" 1000000.0) +nonce: testnet-allocations-0 +keyPairs: [] diff --git a/pact/genesis/testnet06/grants0.yaml b/pact/genesis/testnet06/grants0.yaml new file mode 100644 index 0000000000..c7d5976a68 --- /dev/null +++ b/pact/genesis/testnet06/grants0.yaml @@ -0,0 +1,21 @@ +# Internal testnet accounts +code: |- + (coin.coinbase "sender0" (read-keyset "sender0") 1.0) + (coin.coinbase "pascal" (read-keyset "pascal") 100000.0) + (coin.coinbase "edmund" (read-keyset "edmund") 100000.0) + (coin.coinbase "jose" (read-keyset "jose") 100000.0) + (coin.coinbase "jumbo" (read-keyset "jumbo") 100000.0) + (coin.coinbase "philipp" (read-keyset "edmund") 100000.0) + (coin.coinbase "louis" (read-keyset "louis") 100000.0) + + +data: + sender0: ["c03e76c81a56f831a1ea91b93545d6b49daffc53c7416e065fcffb025faee47e"] + pascal: ["85e745b37605de2e4be123c8e9e65404227912c429e45f87b348b5c19ffb1f38"] + edmund: ["bb42ed45d8f2676b0b75cc11abe834f8287841322271ba574011e4fdf23631b7"] + jose: ["f5a17c5fd9f9d2623ba5e6283de1445f694c0d5a6f7df7b6a44a4f887b3e9eae"] + jumbo: ["eeb0fca9d4d8a72178e4d150152858484a63c27d3991caf48cc9003479156f00"] + philipp: ["3c7cd44578bb90b793e62a10f05a99ce9a2e0f067fbb4ff094d19e13a279ef9b"] + louis: ["c389e2fab9fabdc2958d3d9916ea7899f35b906d192b69f211a79bb4c2379bcf"] +nonce: testnet06-grants-0 +keyPairs: [] diff --git a/pact/genesis/testnet06/grantsN.yaml b/pact/genesis/testnet06/grantsN.yaml new file mode 100644 index 0000000000..8cab5ff784 --- /dev/null +++ b/pact/genesis/testnet06/grantsN.yaml @@ -0,0 +1,8 @@ +# Internal testnet accounts +code: |- + (coin.coinbase "sender0" (read-keyset "sender0") 1.0) + +data: + sender0: ["c03e76c81a56f831a1ea91b93545d6b49daffc53c7416e065fcffb025faee47e"] +nonce: testnet06-grants-N +keyPairs: [] diff --git a/pact/genesis/testnet06/keysets.yaml b/pact/genesis/testnet06/keysets.yaml new file mode 100644 index 0000000000..9007d691e5 --- /dev/null +++ b/pact/genesis/testnet06/keysets.yaml @@ -0,0 +1,12 @@ +code: |- + (define-keyset "allocation00" (read-keyset "allocation00")) + (define-keyset "allocation01" (read-keyset "allocation01")) + (define-keyset "allocation02" (read-keyset "allocation02")) + +data: + allocation00: ["f8a1c425cb65899ae0bf0dd4c6a982554f2ce8914c5084446f4999c3c0019990"] + allocation01: ["e55612401fb67190d3c0a905749264806a3b66c9b9d048b84b96e120e4a387dc"] + allocation02: ["d2502cdab08ef8649090e924d0fd69ce89c2c314fb52bf54405769aa5b425eeb"] + +nonce: testnet06-keysets-N +keyPairs: [] diff --git a/src/Chainweb/BlockHeader/Genesis/Testnet060Payload.hs b/src/Chainweb/BlockHeader/Genesis/Testnet060Payload.hs new file mode 100644 index 0000000000..bd9d31a087 --- /dev/null +++ b/src/Chainweb/BlockHeader/Genesis/Testnet060Payload.hs @@ -0,0 +1,37 @@ +{-# LANGUAGE OverloadedStrings #-} + +-- This module is auto-generated. DO NOT EDIT IT MANUALLY. + +module Chainweb.BlockHeader.Genesis.Testnet060Payload ( payloadBlock ) where + +import qualified Data.Text as T +import qualified Data.Vector as V +import GHC.Stack + +import Chainweb.Payload +import Chainweb.Utils (unsafeFromText, toText) + +payloadBlock :: HasCallStack => PayloadWithOutputs +payloadBlock + | actualHash == expectedHash = payload + | otherwise = error + $ "inconsistent genesis payload detected. THIS IS A BUG in chainweb-node" + <> ". Expected: " <> T.unpack (toText expectedHash) + <> ", actual: " <> T.unpack (toText actualHash) + where + actualHash, expectedHash :: BlockPayloadHash + actualHash = _payloadWithOutputsPayloadHash payload + expectedHash = unsafeFromText "EyMMEitzZjPXEtL8cVgEgiN4w48CBBx0wMWZGQDretw" + + payload = newPayloadWithOutputs minerData coinbase txs + minerData = unsafeFromText "eyJhY2NvdW50IjoiTm9NaW5lciIsInByZWRpY2F0ZSI6IjwiLCJwdWJsaWMta2V5cyI6W119" + coinbase = unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6Ik5PX0NPSU5CQVNFIn0sInJlcUtleSI6IkRsZFJ3Q2JsUTdMb3F5NndZSm5hb2RIbDMwZDNqM2VILXF0RnpmRXY0NmciLCJsb2dzIjpudWxsLCJtZXRhRGF0YSI6bnVsbCwiY29udGludWF0aW9uIjpudWxsLCJ0eElkIjpudWxsfQ" + txs = V.fromList + [ (unsafeFromText "eyJoYXNoIjoiNDhUMExqQW5TRnBGV3h2dmFQVi1fNkUtQ2pEQVBoV1lVRldidnlmMmxGcyIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6bnVsbCxcImNvZGVcIjpcIihpbnRlcmZhY2UgZnVuZ2libGUtdjFcXG5cXG4gIFxcXCIgU3RhbmRhcmQgZm9yIGZ1bmdpYmxlIGNvaW5zIGFuZCB0b2tlbnMgYXMgc3BlY2lmaWVkIGluIEtJUC0wMDAyLiBcXFwiXFxuXFxuICAgOyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXFxuICAgOyBTY2hlbWFcXG5cXG4gICAoZGVmc2NoZW1hIGFjY291bnQtZGV0YWlsc1xcbiAgICBAZG9jIFxcXCJTY2hlbWEgZm9yIHJlc3VsdHMgb2YgJ2FjY291bnQnIG9wZXJhdGlvbi5cXFwiXFxuICAgIEBtb2RlbCBbIChpbnZhcmlhbnQgKCE9IFxcXCJcXFwiIHNlbmRlcikpIF1cXG5cXG4gICAgYWNjb3VudDpzdHJpbmdcXG4gICAgYmFsYW5jZTpkZWNpbWFsXFxuICAgIGd1YXJkOmd1YXJkKVxcblxcblxcbiAgIDsgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxcbiAgIDsgQ2Fwc1xcblxcbiAgIChkZWZjYXAgVFJBTlNGRVI6Ym9vbFxcbiAgICAgKCBzZW5kZXI6c3RyaW5nXFxuICAgICAgIHJlY2VpdmVyOnN0cmluZ1xcbiAgICAgICBhbW91bnQ6ZGVjaW1hbFxcbiAgICAgKVxcbiAgICAgQGRvYyBcXFwiIE1hbmFnZWQgY2FwYWJpbGl0eSBzZWFsaW5nIEFNT1VOVCBmb3IgdHJhbnNmZXIgZnJvbSBTRU5ERVIgdG8gXFxcXFxcbiAgICAgICAgICBcXFxcIFJFQ0VJVkVSLiBQZXJtaXRzIGFueSBudW1iZXIgb2YgdHJhbnNmZXJzIHVwIHRvIEFNT1VOVC5cXFwiXFxuICAgICBAbWFuYWdlZCBhbW91bnQgVFJBTlNGRVItbWdyXFxuICAgICApXFxuXFxuICAgKGRlZnVuIFRSQU5TRkVSLW1ncjpkZWNpbWFsXFxuICAgICAoIG1hbmFnZWQ6ZGVjaW1hbFxcbiAgICAgICByZXF1ZXN0ZWQ6ZGVjaW1hbFxcbiAgICAgKVxcbiAgICAgQGRvYyBcXFwiIE1hbmFnZXMgVFJBTlNGRVIgQU1PVU5UIGxpbmVhcmx5LCBcXFxcXFxuICAgICAgICAgIFxcXFwgc3VjaCB0aGF0IGEgcmVxdWVzdCBmb3IgMS4wIGFtb3VudCBvbiBhIDMuMCBcXFxcXFxuICAgICAgICAgIFxcXFwgbWFuYWdlZCBxdWFudGl0eSBlbWl0cyB1cGRhdGVkIGFtb3VudCAyLjAuXFxcIlxcbiAgICAgKVxcblxcbiAgIDsgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxcbiAgIDsgRnVuY3Rpb25hbGl0eVxcblxcbiAgIChkZWZ1biB0cmFuc2Zlci1jcmVhdGU6c3RyaW5nXFxuICAgICAoIHNlbmRlcjpzdHJpbmdcXG4gICAgICAgcmVjZWl2ZXI6c3RyaW5nXFxuICAgICAgIHJlY2VpdmVyLWd1YXJkOmd1YXJkXFxuICAgICAgIGFtb3VudDpkZWNpbWFsXFxuICAgICApXFxuICAgICBAZG9jIFxcXCIgVHJhbnNmZXIgQU1PVU5UIGJldHdlZW4gYWNjb3VudHMgU0VOREVSIGFuZCBSRUNFSVZFUi4gXFxcXFxcbiAgICAgICAgICBcXFxcIEZhaWxzIGlmIFNFTkRFUiBkb2VzIG5vdCBleGlzdC4gSWYgUkVDRUlWRVIgZXhpc3RzLCBndWFyZCBcXFxcXFxuICAgICAgICAgIFxcXFwgbXVzdCBtYXRjaCBleGlzdGluZyB2YWx1ZS4gSWYgUkVDRUlWRVIgZG9lcyBub3QgZXhpc3QsIFxcXFxcXG4gICAgICAgICAgXFxcXCBSRUNFSVZFUiBhY2NvdW50IGlzIGNyZWF0ZWQgdXNpbmcgUkVDRUlWRVItR1VBUkQuIFxcXFxcXG4gICAgICAgICAgXFxcXCBTdWJqZWN0IHRvIG1hbmFnZW1lbnQgYnkgVFJBTlNGRVIgY2FwYWJpbGl0eS5cXFwiXFxuICAgICBAbW9kZWwgWyAocHJvcGVydHkgKD4gYW1vdW50IDAuMCkpXFxuICAgICAgICAgICAgICAocHJvcGVydHkgKCE9IHNlbmRlciBcXFwiXFxcIikpXFxuICAgICAgICAgICAgICAocHJvcGVydHkgKCE9IHJlY2VpdmVyIFxcXCJcXFwiKSlcXG4gICAgICAgICAgICAgIChwcm9wZXJ0eSAoIT0gc2VuZGVyIHJlY2VpdmVyKSlcXG4gICAgICAgICAgICBdXFxuICAgICApXFxuXFxuICAgKGRlZnBhY3QgdHJhbnNmZXItY3Jvc3NjaGFpbjpzdHJpbmdcXG4gICAgICggc2VuZGVyOnN0cmluZ1xcbiAgICAgICByZWNlaXZlcjpzdHJpbmdcXG4gICAgICAgcmVjZWl2ZXItZ3VhcmQ6Z3VhcmRcXG4gICAgICAgdGFyZ2V0LWNoYWluOnN0cmluZ1xcbiAgICAgICBhbW91bnQ6ZGVjaW1hbFxcbiAgICAgKVxcbiAgICAgQGRvYyBcXFwiIDItc3RlcCBwYWN0IHRvIHRyYW5zZmVyIEFNT1VOVCBmcm9tIFNFTkRFUiBvbiBjdXJyZW50IGNoYWluIFxcXFxcXG4gICAgICAgICAgXFxcXCB0byBSRUNFSVZFUiBvbiBUQVJHRVQtQ0hBSU4gdmlhIFNQViBwcm9vZi4gXFxcXFxcbiAgICAgICAgICBcXFxcIFRBUkdFVC1DSEFJTiBtdXN0IGJlIGRpZmZlcmVudCB0aGFuIGN1cnJlbnQgY2hhaW4gaWQuIFxcXFxcXG4gICAgICAgICAgXFxcXCBGaXJzdCBzdGVwIGRlYml0cyBBTU9VTlQgY29pbnMgaW4gU0VOREVSIGFjY291bnQgYW5kIHlpZWxkcyBcXFxcXFxuICAgICAgICAgIFxcXFwgUkVDRUlWRVIsIFJFQ0VJVkVSX0dVQVJEIGFuZCBBTU9VTlQgdG8gVEFSR0VULUNIQUlOLiBcXFxcXFxuICAgICAgICAgIFxcXFwgU2Vjb25kIHN0ZXAgY29udGludWF0aW9uIGlzIHNlbnQgaW50byBUQVJHRVQtQ0hBSU4gd2l0aCBwcm9vZiBcXFxcXFxuICAgICAgICAgIFxcXFwgb2J0YWluZWQgZnJvbSB0aGUgc3B2ICdvdXRwdXQnIGVuZHBvaW50IG9mIENoYWlud2ViLiBcXFxcXFxuICAgICAgICAgIFxcXFwgUHJvb2YgaXMgdmFsaWRhdGVkIGFuZCBSRUNFSVZFUiBpcyBjcmVkaXRlZCB3aXRoIEFNT1VOVCBcXFxcXFxuICAgICAgICAgIFxcXFwgY3JlYXRpbmcgYWNjb3VudCB3aXRoIFJFQ0VJVkVSX0dVQVJEIGFzIG5lY2Vzc2FyeS5cXFwiXFxuICAgICBAbW9kZWwgWyAocHJvcGVydHkgKD4gYW1vdW50IDAuMCkpXFxuICAgICAgICAgICAgICAocHJvcGVydHkgKCE9IHNlbmRlciBcXFwiXFxcIikpXFxuICAgICAgICAgICAgICAocHJvcGVydHkgKCE9IHJlY2VpdmVyIFxcXCJcXFwiKSlcXG4gICAgICAgICAgICAgIChwcm9wZXJ0eSAoIT0gc2VuZGVyIHJlY2VpdmVyKSlcXG4gICAgICAgICAgICAgIChwcm9wZXJ0eSAoIT0gdGFyZ2V0LWNoYWluIFxcXCJcXFwiKSlcXG4gICAgICAgICAgICBdXFxuICAgICApXFxuXFxuICAgKGRlZnVuIGdldC1iYWxhbmNlOmRlY2ltYWxcXG4gICAgICggYWNjb3VudDpzdHJpbmcgKVxcbiAgICAgXFxcIiBHZXQgYmFsYW5jZSBmb3IgQUNDT1VOVC4gRmFpbHMgaWYgYWNjb3VudCBkb2VzIG5vdCBleGlzdC5cXFwiXFxuICAgICApXFxuXFxuICAgKGRlZnVuIGRldGFpbHM6b2JqZWN0e2FjY291bnQtZGV0YWlsc31cXG4gICAgICggYWNjb3VudDogc3RyaW5nIClcXG4gICAgIFxcXCIgR2V0IGFuIG9iamVjdCB3aXRoIGRldGFpbHMgb2YgQUNDT1VOVC4gXFxcXFxcbiAgICAgXFxcXCBGYWlscyBpZiBhY2NvdW50IGRvZXMgbm90IGV4aXN0LlxcXCJcXG4gICAgIClcXG5cXG4gICAoZGVmdW4gcHJlY2lzaW9uOmludGVnZXJcXG4gICAgICgpXFxuICAgICBcXFwiUmV0dXJuIHRoZSBtYXhpbXVtIGFsbG93ZWQgZGVjaW1hbCBwcmVjaXNpb24uXFxcIlxcbiAgICAgKVxcblxcbiAgIChkZWZ1biBlbmZvcmNlLXVuaXQ6Ym9vbFxcbiAgICAgKCBhbW91bnQ6ZGVjaW1hbCApXFxuICAgICBcXFwiIEVuZm9yY2UgbWluaW11bSBwcmVjaXNpb24gYWxsb3dlZCBmb3IgdHJhbnNhY3Rpb25zLlxcXCJcXG4gICAgIClcXG5cXG4gICAoZGVmdW4gY3JlYXRlLWFjY291bnQ6c3RyaW5nXFxuICAgICAoIGFjY291bnQ6c3RyaW5nXFxuICAgICAgIGd1YXJkOmd1YXJkXFxuICAgICApXFxuICAgICBcXFwiIENyZWF0ZSBBQ0NPVU5UIHdpdGggMC4wIGJhbGFuY2UsIHdpdGggR1VBUkQgY29udHJvbGxpbmcgYWNjZXNzLlxcXCJcXG4gICAgIClcXG5cXG4gICAoZGVmdW4gcm90YXRlOnN0cmluZ1xcbiAgICAgKCBhY2NvdW50OnN0cmluZ1xcbiAgICAgICBuZXctZ3VhcmQ6Z3VhcmRcXG4gICAgIClcXG4gICAgIFxcXCIgUm90YXRlIGd1YXJkIGZvciBBQ0NPVU5ULiBUcmFuc2FjdGlvbiBpcyB2YWxpZGF0ZWQgYWdhaW5zdCBcXFxcXFxuICAgICBcXFxcIGV4aXN0aW5nIGd1YXJkIGJlZm9yZSBpbnN0YWxsaW5nIG5ldyBndWFyZC4gXFxcIlxcbiAgICAgKVxcblxcbilcXG5cIn19LFwic2lnbmVyc1wiOltdLFwibWV0YVwiOntcImNyZWF0aW9uVGltZVwiOjAsXCJ0dGxcIjoxNzI4MDAsXCJnYXNMaW1pdFwiOjAsXCJjaGFpbklkXCI6XCJcIixcImdhc1ByaWNlXCI6MCxcInNlbmRlclwiOlwiXCJ9LFwibm9uY2VcIjpcImdlbmVzaXMtMDFcIn0ifQ", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IkxvYWRlZCBpbnRlcmZhY2UgZnVuZ2libGUtdjEifSwicmVxS2V5IjoiNDhUMExqQW5TRnBGV3h2dmFQVi1fNkUtQ2pEQVBoV1lVRldidnlmMmxGcyIsImxvZ3MiOiJRRmEyOHRuOXkydFdMVzdUc3lHdzNOTkhpYzhxYjJ6UGtudXRpWWhnQXc4IiwibWV0YURhdGEiOm51bGwsImNvbnRpbnVhdGlvbiI6bnVsbCwidHhJZCI6MH0") + , (unsafeFromText "{"hash":"XGPEQDk5PIvQkpq0GGkgNTmo-mjki63ZPgER_kovxq4","sigs":[],"cmd":"{\"networkId\":null,\"payload\":{\"exec\":{\"data\":null,\"code\":\"(module coin GOVERNANCE\\n\\n  @doc \\\"'coin' represents the Kadena Coin Contract. This contract provides both the \\\\\\n  \\\\buy/redeem gas support in the form of 'fund-tx', as well as transfer,       \\\\\\n  \\\\credit, debit, coinbase, account creation and query, as well as SPV burn    \\\\\\n  \\\\create. To access the coin contract, you may use its fully-qualified name,  \\\\\\n  \\\\or issue the '(use coin)' command in the body of a module declaration.\\\"\\n\\n  @model\\n    [ (defproperty conserves-mass\\n        (= (column-delta coin-table 'balance) 0.0))\\n\\n      (defproperty valid-account (account:string)\\n        (and\\n          (>= (length account) 3)\\n          (<= (length account) 256)))\\n    ]\\n\\n  (implements fungible-v1)\\n\\n  ; --------------------------------------------------------------------------\\n  ; Schemas and Tables\\n\\n  (defschema coin-schema\\n    @doc \\\"The coin contract token schema\\\"\\n    @model [ (invariant (>= balance 0.0)) ]\\n\\n    balance:decimal\\n    guard:guard)\\n\\n  (deftable coin-table:{coin-schema})\\n\\n  ; --------------------------------------------------------------------------\\n  ; Capabilities\\n\\n  (defcap GOVERNANCE ()\\n    (enforce false \\\"Enforce non-upgradeability\\\"))\\n\\n  (defcap GAS ()\\n    \\\"Magic capability to protect gas buy and redeem\\\"\\n    true)\\n\\n  (defcap COINBASE ()\\n    \\\"Magic capability to protect miner reward\\\"\\n    true)\\n\\n  (defcap GENESIS ()\\n    \\\"Magic capability constraining genesis transactions\\\"\\n    true)\\n\\n  (defcap DEBIT (sender:string)\\n    \\\"Capability for managing debiting operations\\\"\\n    (enforce-guard (at 'guard (read coin-table sender)))\\n    (enforce (!= sender \\\"\\\") \\\"valid sender\\\"))\\n\\n  (defcap CREDIT (receiver:string)\\n    \\\"Capability for managing crediting operations\\\"\\n    (enforce (!= receiver \\\"\\\") \\\"valid receiver\\\"))\\n\\n  (defcap TRANSFER:bool\\n    ( sender:string\\n      receiver:string\\n      amount:decimal\\n    )\\n    @managed amount TRANSFER-mgr\\n    (enforce (!= sender receiver) \\\"same sender and receiver\\\")\\n    (enforce-unit amount)\\n    (enforce (> amount 0.0) \\\"Positive amount\\\")\\n    (compose-capability (DEBIT sender))\\n    (compose-capability (CREDIT receiver))\\n  )\\n\\n  (defun TRANSFER-mgr:decimal\\n    ( managed:decimal\\n      requested:decimal\\n    )\\n\\n    (let ((newbal (- managed requested)))\\n      (enforce (>= newbal 0.0)\\n        (format \\\"TRANSFER exceeded for balance {}\\\" [managed]))\\n      newbal)\\n  )\\n\\n  ; --------------------------------------------------------------------------\\n  ; Constants\\n\\n  (defconst COIN_CHARSET CHARSET_LATIN1\\n    \\\"The default coin contract character set\\\")\\n\\n  (defconst MINIMUM_PRECISION 12\\n    \\\"Minimum allowed precision for coin transactions\\\")\\n\\n  (defconst MINIMUM_ACCOUNT_LENGTH 3\\n    \\\"Minimum account length admissible for coin accounts\\\")\\n\\n  (defconst MAXIMUM_ACCOUNT_LENGTH 256\\n    \\\"Maximum account name length admissible for coin accounts\\\")\\n\\n  ; --------------------------------------------------------------------------\\n  ; Utilities\\n\\n  (defun enforce-unit:bool (amount:decimal)\\n    @doc \\\"Enforce minimum precision allowed for coin transactions\\\"\\n\\n    (enforce\\n      (= (floor amount MINIMUM_PRECISION)\\n         amount)\\n      (format \\\"Amount violates minimum precision: {}\\\" [amount]))\\n    )\\n\\n  (defun validate-account (account:string)\\n    @doc \\\"Enforce that an account name conforms to the coin contract \\\\\\n         \\\\minimum and maximum length requirements, as well as the    \\\\\\n         \\\\latin-1 character set.\\\"\\n\\n    (enforce\\n      (is-charset COIN_CHARSET account)\\n      (format\\n        \\\"Account does not conform to the coin contract charset: {}\\\"\\n        [account]))\\n\\n    (let ((account-length (length account)))\\n\\n      (enforce\\n        (>= account-length MINIMUM_ACCOUNT_LENGTH)\\n        (format\\n          \\\"Account name does not conform to the min length requirement: {}\\\"\\n          [account]))\\n\\n      (enforce\\n        (<= account-length MAXIMUM_ACCOUNT_LENGTH)\\n        (format\\n          \\\"Account name does not conform to the max length requirement: {}\\\"\\n          [account]))\\n      )\\n  )\\n\\n  ; --------------------------------------------------------------------------\\n  ; Coin Contract\\n\\n  (defun gas-only ()\\n    \\\"Predicate for gas-only user guards.\\\"\\n    (require-capability (GAS)))\\n\\n  (defun gas-guard (guard:guard)\\n    \\\"Predicate for gas + single key user guards\\\"\\n    (enforce-one\\n      \\\"Enforce either the presence of a GAS cap or keyset\\\"\\n      [ (gas-only)\\n        (enforce-guard guard)\\n      ]))\\n\\n  (defun buy-gas:string (sender:string total:decimal)\\n    @doc \\\"This function describes the main 'gas buy' operation. At this point \\\\\\n    \\\\MINER has been chosen from the pool, and will be validated. The SENDER   \\\\\\n    \\\\of this transaction has specified a gas limit LIMIT (maximum gas) for    \\\\\\n    \\\\the transaction, and the price is the spot price of gas at that time.    \\\\\\n    \\\\The gas buy will be executed prior to executing SENDER's code.\\\"\\n\\n    @model [ (property (> total 0.0))\\n             (property (valid-account sender))\\n           ]\\n\\n    (validate-account sender)\\n\\n    (enforce-unit total)\\n    (enforce (> total 0.0) \\\"gas supply must be a positive quantity\\\")\\n\\n    (require-capability (GAS))\\n    (with-capability (DEBIT sender)\\n      (debit sender total))\\n    )\\n\\n  (defun redeem-gas:string (miner:string miner-guard:guard sender:string total:decimal)\\n    @doc \\\"This function describes the main 'redeem gas' operation. At this    \\\\\\n    \\\\point, the SENDER's transaction has been executed, and the gas that      \\\\\\n    \\\\was charged has been calculated. MINER will be credited the gas cost,    \\\\\\n    \\\\and SENDER will receive the remainder up to the limit\\\"\\n\\n    @model [ (property (> total 0.0))\\n             (property (valid-account sender))\\n             (property (valid-account miner))\\n           ]\\n\\n    (validate-account sender)\\n    (validate-account miner)\\n    (enforce-unit total)\\n\\n    (require-capability (GAS))\\n    (let*\\n      ((fee (read-decimal \\\"fee\\\"))\\n       (refund (- total fee)))\\n\\n      (enforce-unit fee)\\n      (enforce (>= fee 0.0)\\n        \\\"fee must be a non-negative quantity\\\")\\n\\n      (enforce (>= refund 0.0)\\n        \\\"refund must be a non-negative quantity\\\")\\n\\n        ; directly update instead of credit\\n      (with-capability (CREDIT sender)\\n        (if (> refund 0.0)\\n          (with-read coin-table sender\\n            { \\\"balance\\\" := balance }\\n            (update coin-table sender\\n              { \\\"balance\\\": (+ balance refund) }))\\n\\n          \\\"noop\\\"))\\n\\n      (with-capability (CREDIT miner)\\n        (if (> fee 0.0)\\n          (credit miner miner-guard fee)\\n          \\\"noop\\\"))\\n      )\\n\\n    )\\n\\n  (defun create-account:string (account:string guard:guard)\\n    @model [ (property (valid-account account)) ]\\n\\n    (validate-account account)\\n\\n    (insert coin-table account\\n      { \\\"balance\\\" : 0.0\\n      , \\\"guard\\\"   : guard\\n      })\\n    )\\n\\n  (defun get-balance:decimal (account:string)\\n    (with-read coin-table account\\n      { \\\"balance\\\" := balance }\\n      balance\\n      )\\n    )\\n\\n  (defun details:object{fungible-v1.account-details}\\n    ( account:string )\\n    (with-read coin-table account\\n      { \\\"balance\\\" := bal\\n      , \\\"guard\\\" := g }\\n      { \\\"account\\\" : account\\n      , \\\"balance\\\" : bal\\n      , \\\"guard\\\": g })\\n    )\\n\\n  (defun rotate:string (account:string new-guard:guard)\\n\\n    (with-read coin-table account\\n      { \\\"guard\\\" := old-guard }\\n\\n      (enforce-guard old-guard)\\n      (enforce-guard new-guard)\\n\\n      (update coin-table account\\n        { \\\"guard\\\" : new-guard }\\n        )))\\n\\n\\n  (defun precision:integer\\n    ()\\n    MINIMUM_PRECISION)\\n\\n  (defun transfer:string (sender:string receiver:string amount:decimal)\\n    @model [ (property conserves-mass)\\n             (property (> amount 0.0))\\n             (property (valid-account sender))\\n             (property (valid-account receiver))\\n             (property (!= sender receiver)) ]\\n\\n    (enforce (!= sender receiver)\\n      \\\"sender cannot be the receiver of a transfer\\\")\\n\\n    (validate-account sender)\\n    (validate-account receiver)\\n\\n    (enforce (> amount 0.0)\\n      \\\"transfer amount must be positive\\\")\\n\\n    (enforce-unit amount)\\n\\n    (with-capability (TRANSFER sender receiver amount)\\n      (debit sender amount)\\n      (with-read coin-table receiver\\n        { \\\"guard\\\" := g }\\n\\n        (credit receiver g amount))\\n      )\\n    )\\n\\n  (defun transfer-create:string\\n    ( sender:string\\n      receiver:string\\n      receiver-guard:guard\\n      amount:decimal )\\n\\n    @model [ (property conserves-mass) ]\\n\\n    (enforce (!= sender receiver)\\n      \\\"sender cannot be the receiver of a transfer\\\")\\n\\n    (validate-account sender)\\n    (validate-account receiver)\\n\\n    (enforce (> amount 0.0)\\n      \\\"transfer amount must be positive\\\")\\n\\n    (enforce-unit amount)\\n\\n    (with-capability (TRANSFER sender receiver amount)\\n      (debit sender amount)\\n      (credit receiver receiver-guard amount))\\n    )\\n\\n  (defun coinbase:string (account:string account-guard:guard amount:decimal)\\n    @doc \\\"Internal function for the initial creation of coins.  This function \\\\\\n    \\\\cannot be used outside of the coin contract.\\\"\\n\\n    @model [ (property (valid-account account)) ]\\n\\n    (validate-account account)\\n    (enforce-unit amount)\\n\\n    (require-capability (COINBASE))\\n    (with-capability (CREDIT account)\\n      (credit account account-guard amount))\\n    )\\n\\n  (defpact fund-tx (sender:string miner:string miner-guard:guard total:decimal)\\n    @doc \\\"'fund-tx' is a special pact to fund a transaction in two steps,     \\\\\\n    \\\\with the actual transaction transpiring in the middle:                   \\\\\\n    \\\\                                                                         \\\\\\n    \\\\  1) A buying phase, debiting the sender for total gas and fee, yielding \\\\\\n    \\\\     TX_MAX_CHARGE.                                                      \\\\\\n    \\\\  2) A settlement phase, resuming TX_MAX_CHARGE, and allocating to the   \\\\\\n    \\\\     coinbase account for used gas and fee, and sender account for bal-  \\\\\\n    \\\\     ance (unused gas, if any).\\\"\\n\\n    @model [ (property (> total 0.0))\\n             (property (valid-account sender))\\n             (property (valid-account miner))\\n             ;(property conserves-mass) not supported yet\\n           ]\\n\\n    (step (buy-gas sender total))\\n    (step (redeem-gas miner miner-guard sender total))\\n    )\\n\\n  (defun debit:string (account:string amount:decimal)\\n    @doc \\\"Debit AMOUNT from ACCOUNT balance\\\"\\n\\n    @model [ (property (> amount 0.0))\\n             (property (valid-account account))\\n           ]\\n\\n    (validate-account account)\\n\\n    (enforce (> amount 0.0)\\n      \\\"debit amount must be positive\\\")\\n\\n    (enforce-unit amount)\\n\\n    (require-capability (DEBIT account))\\n    (with-read coin-table account\\n      { \\\"balance\\\" := balance }\\n\\n      (enforce (<= amount balance) \\\"Insufficient funds\\\")\\n\\n      (update coin-table account\\n        { \\\"balance\\\" : (- balance amount) }\\n        ))\\n    )\\n\\n\\n  (defun credit:string (account:string guard:guard amount:decimal)\\n    @doc \\\"Credit AMOUNT to ACCOUNT balance\\\"\\n\\n    @model [ (property (> amount 0.0))\\n             (property (valid-account account))\\n           ]\\n\\n    (validate-account account)\\n\\n    (enforce (> amount 0.0) \\\"credit amount must be positive\\\")\\n    (enforce-unit amount)\\n\\n    (require-capability (CREDIT account))\\n    (with-default-read coin-table account\\n      { \\\"balance\\\" : 0.0, \\\"guard\\\" : guard }\\n      { \\\"balance\\\" := balance, \\\"guard\\\" := retg }\\n      ; we don't want to overwrite an existing guard with the user-supplied one\\n      (enforce (= retg guard)\\n        \\\"account guards do not match\\\")\\n\\n      (write coin-table account\\n        { \\\"balance\\\" : (+ balance amount)\\n        , \\\"guard\\\"   : retg\\n        })\\n      ))\\n\\n\\n  (defschema crosschain-schema\\n    @doc \\\"Schema for yielded value in cross-chain transfers\\\"\\n    receiver:string\\n    receiver-guard:guard\\n    amount:decimal)\\n\\n  (defpact transfer-crosschain:string\\n    ( sender:string\\n      receiver:string\\n      receiver-guard:guard\\n      target-chain:string\\n      amount:decimal )\\n\\n    @model [ (property (> amount 0.0))\\n             (property (!= receiver \\\"\\\"))\\n             (property (valid-account sender))\\n             (property (valid-account receiver))\\n           ]\\n\\n    (step\\n      (with-capability (DEBIT sender)\\n\\n        (validate-account sender)\\n        (validate-account receiver)\\n\\n        (enforce (!= \\\"\\\" target-chain) \\\"empty target-chain\\\")\\n        (enforce (!= (at 'chain-id (chain-data)) target-chain)\\n          \\\"cannot run cross-chain transfers to the same chain\\\")\\n\\n        (enforce (> amount 0.0)\\n          \\\"transfer quantity must be positive\\\")\\n\\n        (enforce-unit amount)\\n\\n        ;; step 1 - debit delete-account on current chain\\n        (debit sender amount)\\n\\n        (let\\n          ((crosschain-details:object{crosschain-schema}\\n            { \\\"receiver\\\" : receiver\\n            , \\\"receiver-guard\\\" : receiver-guard\\n            , \\\"amount\\\" : amount\\n            }))\\n          (yield crosschain-details target-chain)\\n          )))\\n\\n    (step\\n      (resume\\n        { \\\"receiver\\\" := receiver\\n        , \\\"receiver-guard\\\" := receiver-guard\\n        , \\\"amount\\\" := amount\\n        }\\n\\n        ;; step 2 - credit create account on target chain\\n        (with-capability (CREDIT receiver)\\n          (credit receiver receiver-guard amount))\\n        ))\\n    )\\n\\n\\n  ; --------------------------------------------------------------------------\\n  ; Coin allocations\\n\\n  (defschema allocation-schema\\n    @doc \\\"Genesis allocation registry\\\"\\n    ;@model [ (invariant (>= balance 0.0)) ]\\n\\n    balance:decimal\\n    date:time\\n    guard:guard\\n    redeemed:bool)\\n\\n  (deftable allocation-table:{allocation-schema})\\n\\n  (defun create-allocation-account\\n    ( account:string\\n      date:time\\n      keyset-ref:string\\n      amount:decimal\\n    )\\n\\n    @doc \\\"Add an entry to the coin allocation table. This function \\\\\\n         \\\\also creates a corresponding empty coin contract account \\\\\\n         \\\\of the same name and guard. Requires GENESIS capability. \\\"\\n\\n    @model [ (property (valid-account account)) ]\\n\\n    (require-capability (GENESIS))\\n\\n    (validate-account account)\\n    (enforce (>= amount 0.0)\\n      \\\"allocation amount must be non-negative\\\")\\n\\n    (enforce-unit amount)\\n\\n    (let\\n      ((guard:guard (keyset-ref-guard keyset-ref)))\\n\\n      (create-account account guard)\\n\\n      (insert allocation-table account\\n        { \\\"balance\\\" : amount\\n        , \\\"date\\\" : date\\n        , \\\"guard\\\" : guard\\n        , \\\"redeemed\\\" : false\\n        })))\\n\\n  (defun release-allocation\\n    ( account:string )\\n\\n    @doc \\\"Release funds associated with allocation ACCOUNT into main ledger.   \\\\\\n         \\\\ACCOUNT must already exist in main ledger. Allocation is deactivated \\\\\\n         \\\\after release.\\\"\\n    @model [ (property (valid-account account)) ]\\n\\n    (validate-account account)\\n\\n    (with-read allocation-table account\\n      { \\\"balance\\\" := balance\\n      , \\\"date\\\" := release-time\\n      , \\\"redeemed\\\" := redeemed\\n      , \\\"guard\\\" := guard\\n      }\\n\\n      (let ((curr-time:time (at 'block-time (chain-data))))\\n\\n        (enforce (not redeemed)\\n          \\\"allocation funds have already been redeemed\\\")\\n\\n        (enforce\\n          (>= curr-time release-time)\\n          (format \\\"funds locked until {}. current time: {}\\\" [release-time curr-time]))\\n\\n        (enforce-guard guard)\\n\\n        (with-capability (CREDIT account)\\n          (credit account guard balance)\\n\\n          (update allocation-table account\\n            { \\\"redeemed\\\" : true\\n            , \\\"balance\\\" : 0.0\\n            })\\n\\n          \\\"Allocation successfully released to main ledger\\\")\\n    )))\\n\\n)\\n\\n(create-table coin-table)\\n(create-table allocation-table)\\n\"}},\"signers\":[],\"meta\":{\"creationTime\":0,\"ttl\":172800,\"gasLimit\":0,\"chainId\":\"\",\"gasPrice\":0,\"sender\":\"\"},\"nonce\":\"genesis-01\"}"}", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IlRhYmxlQ3JlYXRlZCJ9LCJyZXFLZXkiOiJYR1BFUURrNVBJdlFrcHEwR0drZ05UbW8tbWpraTYzWlBnRVJfa292eHE0IiwibG9ncyI6IkpvNXhfMEl0dnZLVTY5dHJFbjh1a3FCeWpSTy1XaEZJZFlmMDV5dDlfN0UiLCJtZXRhRGF0YSI6bnVsbCwiY29udGludWF0aW9uIjpudWxsLCJ0eElkIjoxfQ") + , (unsafeFromText "eyJoYXNoIjoiU0IzVzVFTGl6azl4elNWWk9MX3dsem5VNjh5aUhPQzlwWUhreHBVXzBnbyIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6bnVsbCxcImNvZGVcIjpcIihpbnRlcmZhY2UgZ2FzLXBheWVyLXYxXFxuXFxuICAoZGVmY2FwIEdBU19QQVlFUjpib29sXFxuICAgICggdXNlcjpzdHJpbmdcXG4gICAgICBsaW1pdDppbnRlZ2VyXFxuICAgICAgcHJpY2U6ZGVjaW1hbFxcbiAgICApXFxuICAgIEBkb2NcXG4gICAgXFxcIiBQcm92aWRlIGEgY2FwYWJpbGl0eSBpbmRpY2F0aW5nIHRoYXQgZGVjbGFyaW5nIG1vZHVsZSBzdXBwb3J0cyBcXFxcXFxuICAgIFxcXFwgZ2FzIHBheW1lbnQgZm9yIFVTRVIgZm9yIGdhcyBMSU1JVCBhbmQgUFJJQ0UuIEZ1bmN0aW9uYWxpdHkgXFxcXFxcbiAgICBcXFxcIHNob3VsZCByZXF1aXJlIGNhcGFiaWxpdHkgKGNvaW4uRlVORF9UWCksIGFuZCBzaG91bGQgdmFsaWRhdGUgXFxcXFxcbiAgICBcXFxcIHRoZSBzcGVuZCBvZiAobGltaXQgKiBwcmljZSksIHBvc3NpYmx5IHVwZGF0aW5nIHNvbWUgZGF0YWJhc2UgXFxcXFxcbiAgICBcXFxcIGVudHJ5LiBcXFxcXFxuICAgIFxcXFwgU2hvdWxkIGNvbXBvc2UgY2FwYWJpbGl0eSByZXF1aXJlZCBmb3IgJ2NyZWF0ZS1nYXMtcGF5ZXItZ3VhcmQnLlxcXCJcXG4gICAgQG1vZGVsXFxuICAgIFsgKHByb3BlcnR5ICh1c2VyICE9IFxcXCJcXFwiKSlcXG4gICAgICAocHJvcGVydHkgKGxpbWl0ID4gMCkpXFxuICAgICAgKHByb3BlcnR5IChwcmljZSA-IDAuMCkpXFxuICAgIF1cXG4gIClcXG5cXG4gIChkZWZ1biBjcmVhdGUtZ2FzLXBheWVyLWd1YXJkOmd1YXJkICgpXFxuICAgIEBkb2NcXG4gICAgXFxcIiBQcm92aWRlIGEgZ3VhcmQgc3VpdGFibGUgZm9yIGNvbnRyb2xsaW5nIGEgY29pbiBhY2NvdW50IHRoYXQgY2FuIFxcXFxcXG4gICAgXFxcXCBwYXkgZ2FzIHZpYSBHQVNfUEFZRVIgbWVjaGFuaWNzLiBHZW5lcmFsbHkgdGhpcyBpcyBhY2NvbXBsaXNoZWQgXFxcXFxcbiAgICBcXFxcIGJ5IGhhdmluZyBHQVNfUEFZRVIgY29tcG9zZSBhbiB1bnBhcmFtZXRlcml6ZWQsIHVubWFuYWdlZCBjYXBhYmlsaXR5IFxcXFxcXG4gICAgXFxcXCB0aGF0IGlzIHJlcXVpcmVkIGluIHRoaXMgZ3VhcmQuIFRodXMsIGlmIGNvaW4gY29udHJhY3QgaXMgYWJsZSB0byBcXFxcXFxuICAgIFxcXFwgc3VjY2Vzc2Z1bGx5IGFjcXVpcmUgR0FTX1BBWUVSLCB0aGUgY29tcG9zZWQgJ2Fub255bW91cycgY2FwIHJlcXVpcmVkIFxcXFxcXG4gICAgXFxcXCBoZXJlIHdpbGwgYmUgaW4gc2NvcGUsIGFuZCBnYXMgYnV5IHdpbGwgc3VjY2VlZC5cXFwiXFxuICApXFxuXFxuKVxcblwifX0sXCJzaWduZXJzXCI6W10sXCJtZXRhXCI6e1wiY3JlYXRpb25UaW1lXCI6MCxcInR0bFwiOjE3MjgwMCxcImdhc0xpbWl0XCI6MCxcImNoYWluSWRcIjpcIlwiLFwiZ2FzUHJpY2VcIjowLFwic2VuZGVyXCI6XCJcIn0sXCJub25jZVwiOlwiZ2VuZXNpcy0wMVwifSJ9", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IkxvYWRlZCBpbnRlcmZhY2UgZ2FzLXBheWVyLXYxIn0sInJlcUtleSI6IlNCM1c1RUxpems5eHpTVlpPTF93bHpuVTY4eWlIT0M5cFlIa3hwVV8wZ28iLCJsb2dzIjoiZlZuSFlta19QNmJSY3VjeVg1RDdLamNLYkVsVDlEcU9vZW9yUFEtUXdsMCIsIm1ldGFEYXRhIjpudWxsLCJjb250aW51YXRpb24iOm51bGwsInR4SWQiOjJ9") + , (unsafeFromText "eyJoYXNoIjoia2ZMd2Y2a0FzdEVnc0NLYnZPOHR2YTNnWktBWXgzT0dHYTZYRURMaU9hMCIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6e1wibnMtYWRtaW4ta2V5c2V0XCI6W1wiMzY4ODIwZjgwYzMyNGJiYzdjMmIwNjEwNjg4YTdkYTQzZTM5ZjkxZDExODczMjY3MWNkOWM3NTAwZmY0M2NjYVwiXSxcIm5zLW9wZXJhdGUta2V5c2V0XCI6W1wiMzY4ODIwZjgwYzMyNGJiYzdjMmIwNjEwNjg4YTdkYTQzZTM5ZjkxZDExODczMjY3MWNkOWM3NTAwZmY0M2NjYVwiXSxcIm5zLWdlbmVzaXMta2V5c2V0XCI6e1wicHJlZFwiOlwiPVwiLFwia2V5c1wiOltdfX0sXCJjb2RlXCI6XCJcXG4oZGVmaW5lLWtleXNldCAnbnMtYWRtaW4ta2V5c2V0IChyZWFkLWtleXNldCAnbnMtYWRtaW4ta2V5c2V0KSlcXG4oZGVmaW5lLWtleXNldCAnbnMtb3BlcmF0ZS1rZXlzZXQgKHJlYWQta2V5c2V0ICducy1nZW5lc2lzLWtleXNldCkpXFxuXFxuKG1vZHVsZSBucyBHT1ZFUk5BTkNFXFxuICBcXFwiQWRtaW5pc3RlcnMgZGVmaW5pdGlvbiBvZiBuZXcgbmFtZXNwYWNlcyBpbiBDaGFpbndlYi5cXFwiXFxuXFxuICAoZGVmc2NoZW1hIHJlZy1lbnRyeVxcbiAgICBhZG1pbi1ndWFyZDpndWFyZFxcbiAgICBhY3RpdmU6Ym9vbClcXG5cXG4gIChkZWZ0YWJsZSByZWdpc3RyeTp7cmVnLWVudHJ5fSlcXG5cXG4gIChkZWZjYXAgR09WRVJOQU5DRSAoKVxcbiAgICAoZW5mb3JjZS1rZXlzZXQgJ25zLWFkbWluLWtleXNldCkpXFxuXFxuICAoZGVmY2FwIE9QRVJBVEUgKClcXG4gICAgKGVuZm9yY2Uta2V5c2V0ICducy1vcGVyYXRlLWtleXNldCkpXFxuXFxuICAoZGVmY29uc3QgR1VBUkRfU1VDQ0VTUyAoY3JlYXRlLXVzZXItZ3VhcmQgKHN1Y2Nlc3MpKSlcXG4gIChkZWZjb25zdCBHVUFSRF9GQUlMVVJFIChjcmVhdGUtdXNlci1ndWFyZCAoZmFpbHVyZSkpKVxcblxcbiAgKGRlZnVuIHN1Y2Nlc3MgKClcXG4gICAgdHJ1ZSlcXG4gIChkZWZ1biBmYWlsdXJlICgpXFxuICAgIChlbmZvcmNlIGZhbHNlIFxcXCJEaXNhYmxlZFxcXCIpKVxcblxcbiAgKGRlZnVuIHZhbGlkYXRlLW5hbWUgKG5hbWUpXFxuICAgIChlbmZvcmNlICghPSBcXFwiXFxcIiBuYW1lKSBcXFwiRW1wdHkgbmFtZSBub3QgYWxsb3dlZFxcXCIpXFxuICAgIChlbmZvcmNlICg8IChsZW5ndGggbmFtZSkgNjQpIFxcXCJOYW1lIG11c3QgYmUgbGVzcyB0aGFuIDY0IGNoYXJhY3RlcnMgbG9uZ1xcXCIpXFxuICAgIChlbmZvcmNlIChpcy1jaGFyc2V0IENIQVJTRVRfTEFUSU4xIG5hbWUpXFxuICAgICAgICAgICAgIFxcXCJOYW1lIG11c3QgYmUgaW4gbGF0aW4xIGNoYXJzZXRcXFwiKSlcXG5cXG4gIChkZWZ1biB2YWxpZGF0ZTpib29sXFxuICAgICAgKCBucy1uYW1lOnN0cmluZ1xcbiAgICAgICAgbnMtYWRtaW46Z3VhcmRcXG4gICAgICAgIClcXG4gICAgXFxcIiBNYW5hZ2VzIG5hbWVzcGFjZSBpbnN0YWxsIGZvciBDaGFpbndlYi4gUmVxdWlyZXMgYWN0aXZlIHJvdyBpbiByZWdpc3RyeSBcXFxcXFxuICAgIFxcXFwgZm9yIE5TLU5BTUUgd2l0aCBndWFyZCBtYXRjaGluZyBOUy1BRE1JTi5cXFwiXFxuXFxuICAgICh2YWxpZGF0ZS1uYW1lIG5zLW5hbWUpXFxuXFxuICAgICh3aXRoLWRlZmF1bHQtcmVhZCByZWdpc3RyeSBucy1uYW1lXFxuICAgICAgeyAnYWRtaW4tZ3VhcmQgOiBucy1hZG1pblxcbiAgICAgICwgJ2FjdGl2ZSA6IGZhbHNlIH1cXG4gICAgICB7ICdhZG1pbi1ndWFyZCA6PSBhZ1xcbiAgICAgICwgJ2FjdGl2ZSA6PSBpcy1hY3RpdmUgfVxcblxcbiAgICAgICAgKGVuZm9yY2UgaXMtYWN0aXZlIFxcXCJJbmFjdGl2ZSBvciB1bnJlZ2lzdGVyZWQgbmFtZXNwYWNlXFxcIilcXG4gICAgICAgIChlbmZvcmNlICg9IG5zLWFkbWluIGFnKSBcXFwiQWRtaW4gZ3VhcmQgbXVzdCBtYXRjaCBndWFyZCBpbiByZWdpc3RyeVxcXCIpXFxuXFxuICAgICAgICB0cnVlKSlcXG5cXG4gIChkZWZ1biB3cml0ZS1yZWdpc3RyeTpzdHJpbmdcXG4gICAgICAoIG5zLW5hbWU6c3RyaW5nXFxuICAgICAgICBndWFyZDpndWFyZFxcbiAgICAgICAgYWN0aXZlOmJvb2xcXG4gICAgICAgIClcXG4gICAgXFxcIiBXcml0ZSBlbnRyeSB3aXRoIEdVQVJEIGFuZCBBQ1RJVkUgaW50byByZWdpc3RyeSBmb3IgTkFNRS4gXFxcXFxcbiAgICBcXFxcIEd1YXJkZWQgYnkgb3BlcmF0ZSBrZXlzZXQuIFxcXCJcXG5cXG4gICAgKHdpdGgtY2FwYWJpbGl0eSAoT1BFUkFURSlcXG5cXG4gICAgICAodmFsaWRhdGUtbmFtZSBucy1uYW1lKVxcblxcbiAgICAgICh3cml0ZSByZWdpc3RyeSBucy1uYW1lXFxuICAgICAgICB7ICdhZG1pbi1ndWFyZDogZ3VhcmRcXG4gICAgICAgICwgJ2FjdGl2ZTogYWN0aXZlIH0pXFxuXFxuICAgICAgXFxcIlJlZ2lzdGVyIGVudHJ5IHdyaXR0ZW5cXFwiKSlcXG5cXG4gIChkZWZ1biBxdWVyeTpvYmplY3R7cmVnLWVudHJ5fVxcbiAgICAgICggbnMtbmFtZTpzdHJpbmcgKVxcbiAgICAocmVhZCByZWdpc3RyeSBucy1uYW1lKSlcXG5cXG4gIClcXG5cXG4oY3JlYXRlLXRhYmxlIHJlZ2lzdHJ5KVxcblxcbih3cml0ZS1yZWdpc3RyeSBcXFwia2FkZW5hXFxcIlxcbiAgKGtleXNldC1yZWYtZ3VhcmQgJ25zLW9wZXJhdGUta2V5c2V0KSB0cnVlKVxcbih3cml0ZS1yZWdpc3RyeSBcXFwidXNlclxcXCIgR1VBUkRfRkFJTFVSRSB0cnVlKVxcbih3cml0ZS1yZWdpc3RyeSBcXFwiZnJlZVxcXCIgR1VBUkRfRkFJTFVSRSB0cnVlKVxcblxcbihkZWZpbmUtbmFtZXNwYWNlIFxcXCJrYWRlbmFcXFwiXFxuICAoa2V5c2V0LXJlZi1ndWFyZCAnbnMtb3BlcmF0ZS1rZXlzZXQpXFxuICAoa2V5c2V0LXJlZi1ndWFyZCAnbnMtb3BlcmF0ZS1rZXlzZXQpKVxcblxcbihkZWZpbmUtbmFtZXNwYWNlIFxcXCJ1c2VyXFxcIiBHVUFSRF9TVUNDRVNTIEdVQVJEX0ZBSUxVUkUpXFxuKGRlZmluZS1uYW1lc3BhY2UgXFxcImZyZWVcXFwiIEdVQVJEX1NVQ0NFU1MgR1VBUkRfRkFJTFVSRSlcXG47O3JvdGF0ZSB0byByZWFsIG9wZXJhdGUga2V5c2V0XFxuKGRlZmluZS1rZXlzZXQgJ25zLW9wZXJhdGUta2V5c2V0IChyZWFkLWtleXNldCAnbnMtb3BlcmF0ZS1rZXlzZXQpKVxcblwifX0sXCJzaWduZXJzXCI6W10sXCJtZXRhXCI6e1wiY3JlYXRpb25UaW1lXCI6MCxcInR0bFwiOjE3MjgwMCxcImdhc0xpbWl0XCI6MCxcImNoYWluSWRcIjpcIlwiLFwiZ2FzUHJpY2VcIjowLFwic2VuZGVyXCI6XCJcIn0sXCJub25jZVwiOlwibG9hZC1ucy1kZXZuZXQtc2VuZGVyMDBcIn0ifQ", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IktleXNldCBkZWZpbmVkIn0sInJlcUtleSI6ImtmTHdmNmtBc3RFZ3NDS2J2Tzh0dmEzZ1pLQVl4M09HR2E2WEVETGlPYTAiLCJsb2dzIjoiZ1BaazFQZzY5UDZUYXlTSEpxZE9QeEthdHc5WTV3eHZXdjNBRFRqMHVRdyIsIm1ldGFEYXRhIjpudWxsLCJjb250aW51YXRpb24iOm51bGwsInR4SWQiOjN9") + , (unsafeFromText "eyJoYXNoIjoiYWlyX21QQWdzOW1hOWlaMS1FdHhYcjV5X1E0bm9xZ0k1T245MjZsWVRqOCIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6e1wiYWxsb2NhdGlvbjAyXCI6W1wiZDI1MDJjZGFiMDhlZjg2NDkwOTBlOTI0ZDBmZDY5Y2U4OWMyYzMxNGZiNTJiZjU0NDA1NzY5YWE1YjQyNWVlYlwiXSxcImFsbG9jYXRpb24wMFwiOltcImY4YTFjNDI1Y2I2NTg5OWFlMGJmMGRkNGM2YTk4MjU1NGYyY2U4OTE0YzUwODQ0NDZmNDk5OWMzYzAwMTk5OTBcIl0sXCJhbGxvY2F0aW9uMDFcIjpbXCJlNTU2MTI0MDFmYjY3MTkwZDNjMGE5MDU3NDkyNjQ4MDZhM2I2NmM5YjlkMDQ4Yjg0Yjk2ZTEyMGU0YTM4N2RjXCJdfSxcImNvZGVcIjpcIihkZWZpbmUta2V5c2V0IFxcXCJhbGxvY2F0aW9uMDBcXFwiIChyZWFkLWtleXNldCBcXFwiYWxsb2NhdGlvbjAwXFxcIikpXFxuKGRlZmluZS1rZXlzZXQgXFxcImFsbG9jYXRpb24wMVxcXCIgKHJlYWQta2V5c2V0IFxcXCJhbGxvY2F0aW9uMDFcXFwiKSlcXG4oZGVmaW5lLWtleXNldCBcXFwiYWxsb2NhdGlvbjAyXFxcIiAocmVhZC1rZXlzZXQgXFxcImFsbG9jYXRpb24wMlxcXCIpKVwifX0sXCJzaWduZXJzXCI6W10sXCJtZXRhXCI6e1wiY3JlYXRpb25UaW1lXCI6MCxcInR0bFwiOjE3MjgwMCxcImdhc0xpbWl0XCI6MCxcImNoYWluSWRcIjpcIlwiLFwiZ2FzUHJpY2VcIjowLFwic2VuZGVyXCI6XCJcIn0sXCJub25jZVwiOlwidGVzdG5ldDA2LWtleXNldHMtTlwifSJ9", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IktleXNldCBkZWZpbmVkIn0sInJlcUtleSI6ImFpcl9tUEFnczltYTlpWjEtRXR4WHI1eV9RNG5vcWdJNU9uOTI2bFlUajgiLCJsb2dzIjoiMmtJc0VCSUlQU0c5MmZuRW96aGVtR1k1dmQ5Vlowb3ZCSXhNYmhJUUlUdyIsIm1ldGFEYXRhIjpudWxsLCJjb250aW51YXRpb24iOm51bGwsInR4SWQiOjR9") + , (unsafeFromText "eyJoYXNoIjoiRjlTLUpKX0Zrdm04dlQtZXdlSkRZZlZTMnVSR0lRcUVHdElUTHRyeDJZbyIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6bnVsbCxcImNvZGVcIjpcIihjb2luLmNyZWF0ZS1hbGxvY2F0aW9uLWFjY291bnQgXFxcImFsbG9jYXRpb24wMFxcXCIgKHRpbWUgXFxcIjE5MDAtMTAtMTVUMTg6MDA6MDBaXFxcIikgXFxcImFsbG9jYXRpb24wMFxcXCIgMTAwMDAwMC4wKVxcbihjb2luLmNyZWF0ZS1hbGxvY2F0aW9uLWFjY291bnQgXFxcImFsbG9jYXRpb24wMVxcXCIgKHRpbWUgXFxcIjIwMjYtMDEtMzFUMTg6MDA6MDBaXFxcIikgXFxcImFsbG9jYXRpb24wMVxcXCIgMTAwMDAwMC4wKVxcbihjb2luLmNyZWF0ZS1hbGxvY2F0aW9uLWFjY291bnQgXFxcImFsbG9jYXRpb24wMlxcXCIgKHRpbWUgXFxcIjIwMjYtMDYtMzFUMTg6MDA6MDBaXFxcIikgXFxcImFsbG9jYXRpb24wMlxcXCIgMTAwMDAwMC4wKVwifX0sXCJzaWduZXJzXCI6W10sXCJtZXRhXCI6e1wiY3JlYXRpb25UaW1lXCI6MCxcInR0bFwiOjE3MjgwMCxcImdhc0xpbWl0XCI6MCxcImNoYWluSWRcIjpcIlwiLFwiZ2FzUHJpY2VcIjowLFwic2VuZGVyXCI6XCJcIn0sXCJub25jZVwiOlwidGVzdG5ldC1hbGxvY2F0aW9ucy0wXCJ9In0", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IldyaXRlIHN1Y2NlZWRlZCJ9LCJyZXFLZXkiOiJGOVMtSkpfRmt2bTh2VC1ld2VKRFlmVlMydVJHSVFxRUd0SVRMdHJ4MllvIiwibG9ncyI6Ik8tWExZNzBQWDhHRzZiU3hTdk8xZUZNVDg1VUkxUnBFZ1BKVVJTMFJzTVkiLCJtZXRhRGF0YSI6bnVsbCwiY29udGludWF0aW9uIjpudWxsLCJ0eElkIjo1fQ") + , (unsafeFromText "eyJoYXNoIjoiRFN0b2hEZGtVWVA5MnQwVVh2V2lnUlcyMWEtbEtqejFFUzdkYlozMFA3YyIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6e1wibG91aXNcIjpbXCJjMzg5ZTJmYWI5ZmFiZGMyOTU4ZDNkOTkxNmVhNzg5OWYzNWI5MDZkMTkyYjY5ZjIxMWE3OWJiNGMyMzc5YmNmXCJdLFwicGFzY2FsXCI6W1wiODVlNzQ1YjM3NjA1ZGUyZTRiZTEyM2M4ZTllNjU0MDQyMjc5MTJjNDI5ZTQ1Zjg3YjM0OGI1YzE5ZmZiMWYzOFwiXSxcImp1bWJvXCI6W1wiZWViMGZjYTlkNGQ4YTcyMTc4ZTRkMTUwMTUyODU4NDg0YTYzYzI3ZDM5OTFjYWY0OGNjOTAwMzQ3OTE1NmYwMFwiXSxcImpvc2VcIjpbXCJmNWExN2M1ZmQ5ZjlkMjYyM2JhNWU2MjgzZGUxNDQ1ZjY5NGMwZDVhNmY3ZGY3YjZhNDRhNGY4ODdiM2U5ZWFlXCJdLFwicGhpbGlwcFwiOltcIjNjN2NkNDQ1NzhiYjkwYjc5M2U2MmExMGYwNWE5OWNlOWEyZTBmMDY3ZmJiNGZmMDk0ZDE5ZTEzYTI3OWVmOWJcIl0sXCJzZW5kZXIwXCI6W1wiYzAzZTc2YzgxYTU2ZjgzMWExZWE5MWI5MzU0NWQ2YjQ5ZGFmZmM1M2M3NDE2ZTA2NWZjZmZiMDI1ZmFlZTQ3ZVwiXSxcImVkbXVuZFwiOltcImJiNDJlZDQ1ZDhmMjY3NmIwYjc1Y2MxMWFiZTgzNGY4Mjg3ODQxMzIyMjcxYmE1NzQwMTFlNGZkZjIzNjMxYjdcIl19LFwiY29kZVwiOlwiKGNvaW4uY29pbmJhc2UgXFxcInNlbmRlcjBcXFwiIChyZWFkLWtleXNldCBcXFwic2VuZGVyMFxcXCIpIDEuMClcXG4oY29pbi5jb2luYmFzZSBcXFwicGFzY2FsXFxcIiAgKHJlYWQta2V5c2V0IFxcXCJwYXNjYWxcXFwiKSAxMDAwMDAuMClcXG4oY29pbi5jb2luYmFzZSBcXFwiZWRtdW5kXFxcIiAgKHJlYWQta2V5c2V0IFxcXCJlZG11bmRcXFwiKSAxMDAwMDAuMClcXG4oY29pbi5jb2luYmFzZSBcXFwiam9zZVxcXCIgICAgKHJlYWQta2V5c2V0IFxcXCJqb3NlXFxcIikgICAxMDAwMDAuMClcXG4oY29pbi5jb2luYmFzZSBcXFwianVtYm9cXFwiICAgKHJlYWQta2V5c2V0IFxcXCJqdW1ib1xcXCIpICAxMDAwMDAuMClcXG4oY29pbi5jb2luYmFzZSBcXFwicGhpbGlwcFxcXCIgKHJlYWQta2V5c2V0IFxcXCJlZG11bmRcXFwiKSAxMDAwMDAuMClcXG4oY29pbi5jb2luYmFzZSBcXFwibG91aXNcXFwiICAgKHJlYWQta2V5c2V0IFxcXCJsb3Vpc1xcXCIpICAxMDAwMDAuMClcIn19LFwic2lnbmVyc1wiOltdLFwibWV0YVwiOntcImNyZWF0aW9uVGltZVwiOjAsXCJ0dGxcIjoxNzI4MDAsXCJnYXNMaW1pdFwiOjAsXCJjaGFpbklkXCI6XCJcIixcImdhc1ByaWNlXCI6MCxcInNlbmRlclwiOlwiXCJ9LFwibm9uY2VcIjpcInRlc3RuZXQwNi1ncmFudHMtMFwifSJ9", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IldyaXRlIHN1Y2NlZWRlZCJ9LCJyZXFLZXkiOiJEU3RvaERka1VZUDkydDBVWHZXaWdSVzIxYS1sS2p6MUVTN2RiWjMwUDdjIiwibG9ncyI6ImVzRU5WZjdlVHp4WmVKNTl0OWRCenVZR2VRZ3paM1FKR0FKRUgwWXBNaFkiLCJtZXRhRGF0YSI6bnVsbCwiY29udGludWF0aW9uIjpudWxsLCJ0eElkIjo2fQ") + ] diff --git a/src/Chainweb/BlockHeader/Genesis/Testnet061to19Payload.hs b/src/Chainweb/BlockHeader/Genesis/Testnet061to19Payload.hs new file mode 100644 index 0000000000..e5f0dbc51d --- /dev/null +++ b/src/Chainweb/BlockHeader/Genesis/Testnet061to19Payload.hs @@ -0,0 +1,37 @@ +{-# LANGUAGE OverloadedStrings #-} + +-- This module is auto-generated. DO NOT EDIT IT MANUALLY. + +module Chainweb.BlockHeader.Genesis.Testnet061to19Payload ( payloadBlock ) where + +import qualified Data.Text as T +import qualified Data.Vector as V +import GHC.Stack + +import Chainweb.Payload +import Chainweb.Utils (unsafeFromText, toText) + +payloadBlock :: HasCallStack => PayloadWithOutputs +payloadBlock + | actualHash == expectedHash = payload + | otherwise = error + $ "inconsistent genesis payload detected. THIS IS A BUG in chainweb-node" + <> ". Expected: " <> T.unpack (toText expectedHash) + <> ", actual: " <> T.unpack (toText actualHash) + where + actualHash, expectedHash :: BlockPayloadHash + actualHash = _payloadWithOutputsPayloadHash payload + expectedHash = unsafeFromText "eq3MSmd5bkjQXSqibeEjWvL2_AZqwwipPEWOTMl0FkY" + + payload = newPayloadWithOutputs minerData coinbase txs + minerData = unsafeFromText "eyJhY2NvdW50IjoiTm9NaW5lciIsInByZWRpY2F0ZSI6IjwiLCJwdWJsaWMta2V5cyI6W119" + coinbase = unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6Ik5PX0NPSU5CQVNFIn0sInJlcUtleSI6IkRsZFJ3Q2JsUTdMb3F5NndZSm5hb2RIbDMwZDNqM2VILXF0RnpmRXY0NmciLCJsb2dzIjpudWxsLCJtZXRhRGF0YSI6bnVsbCwiY29udGludWF0aW9uIjpudWxsLCJ0eElkIjpudWxsfQ" + txs = V.fromList + [ (unsafeFromText "eyJoYXNoIjoiNDhUMExqQW5TRnBGV3h2dmFQVi1fNkUtQ2pEQVBoV1lVRldidnlmMmxGcyIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6bnVsbCxcImNvZGVcIjpcIihpbnRlcmZhY2UgZnVuZ2libGUtdjFcXG5cXG4gIFxcXCIgU3RhbmRhcmQgZm9yIGZ1bmdpYmxlIGNvaW5zIGFuZCB0b2tlbnMgYXMgc3BlY2lmaWVkIGluIEtJUC0wMDAyLiBcXFwiXFxuXFxuICAgOyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXFxuICAgOyBTY2hlbWFcXG5cXG4gICAoZGVmc2NoZW1hIGFjY291bnQtZGV0YWlsc1xcbiAgICBAZG9jIFxcXCJTY2hlbWEgZm9yIHJlc3VsdHMgb2YgJ2FjY291bnQnIG9wZXJhdGlvbi5cXFwiXFxuICAgIEBtb2RlbCBbIChpbnZhcmlhbnQgKCE9IFxcXCJcXFwiIHNlbmRlcikpIF1cXG5cXG4gICAgYWNjb3VudDpzdHJpbmdcXG4gICAgYmFsYW5jZTpkZWNpbWFsXFxuICAgIGd1YXJkOmd1YXJkKVxcblxcblxcbiAgIDsgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxcbiAgIDsgQ2Fwc1xcblxcbiAgIChkZWZjYXAgVFJBTlNGRVI6Ym9vbFxcbiAgICAgKCBzZW5kZXI6c3RyaW5nXFxuICAgICAgIHJlY2VpdmVyOnN0cmluZ1xcbiAgICAgICBhbW91bnQ6ZGVjaW1hbFxcbiAgICAgKVxcbiAgICAgQGRvYyBcXFwiIE1hbmFnZWQgY2FwYWJpbGl0eSBzZWFsaW5nIEFNT1VOVCBmb3IgdHJhbnNmZXIgZnJvbSBTRU5ERVIgdG8gXFxcXFxcbiAgICAgICAgICBcXFxcIFJFQ0VJVkVSLiBQZXJtaXRzIGFueSBudW1iZXIgb2YgdHJhbnNmZXJzIHVwIHRvIEFNT1VOVC5cXFwiXFxuICAgICBAbWFuYWdlZCBhbW91bnQgVFJBTlNGRVItbWdyXFxuICAgICApXFxuXFxuICAgKGRlZnVuIFRSQU5TRkVSLW1ncjpkZWNpbWFsXFxuICAgICAoIG1hbmFnZWQ6ZGVjaW1hbFxcbiAgICAgICByZXF1ZXN0ZWQ6ZGVjaW1hbFxcbiAgICAgKVxcbiAgICAgQGRvYyBcXFwiIE1hbmFnZXMgVFJBTlNGRVIgQU1PVU5UIGxpbmVhcmx5LCBcXFxcXFxuICAgICAgICAgIFxcXFwgc3VjaCB0aGF0IGEgcmVxdWVzdCBmb3IgMS4wIGFtb3VudCBvbiBhIDMuMCBcXFxcXFxuICAgICAgICAgIFxcXFwgbWFuYWdlZCBxdWFudGl0eSBlbWl0cyB1cGRhdGVkIGFtb3VudCAyLjAuXFxcIlxcbiAgICAgKVxcblxcbiAgIDsgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxcbiAgIDsgRnVuY3Rpb25hbGl0eVxcblxcbiAgIChkZWZ1biB0cmFuc2Zlci1jcmVhdGU6c3RyaW5nXFxuICAgICAoIHNlbmRlcjpzdHJpbmdcXG4gICAgICAgcmVjZWl2ZXI6c3RyaW5nXFxuICAgICAgIHJlY2VpdmVyLWd1YXJkOmd1YXJkXFxuICAgICAgIGFtb3VudDpkZWNpbWFsXFxuICAgICApXFxuICAgICBAZG9jIFxcXCIgVHJhbnNmZXIgQU1PVU5UIGJldHdlZW4gYWNjb3VudHMgU0VOREVSIGFuZCBSRUNFSVZFUi4gXFxcXFxcbiAgICAgICAgICBcXFxcIEZhaWxzIGlmIFNFTkRFUiBkb2VzIG5vdCBleGlzdC4gSWYgUkVDRUlWRVIgZXhpc3RzLCBndWFyZCBcXFxcXFxuICAgICAgICAgIFxcXFwgbXVzdCBtYXRjaCBleGlzdGluZyB2YWx1ZS4gSWYgUkVDRUlWRVIgZG9lcyBub3QgZXhpc3QsIFxcXFxcXG4gICAgICAgICAgXFxcXCBSRUNFSVZFUiBhY2NvdW50IGlzIGNyZWF0ZWQgdXNpbmcgUkVDRUlWRVItR1VBUkQuIFxcXFxcXG4gICAgICAgICAgXFxcXCBTdWJqZWN0IHRvIG1hbmFnZW1lbnQgYnkgVFJBTlNGRVIgY2FwYWJpbGl0eS5cXFwiXFxuICAgICBAbW9kZWwgWyAocHJvcGVydHkgKD4gYW1vdW50IDAuMCkpXFxuICAgICAgICAgICAgICAocHJvcGVydHkgKCE9IHNlbmRlciBcXFwiXFxcIikpXFxuICAgICAgICAgICAgICAocHJvcGVydHkgKCE9IHJlY2VpdmVyIFxcXCJcXFwiKSlcXG4gICAgICAgICAgICAgIChwcm9wZXJ0eSAoIT0gc2VuZGVyIHJlY2VpdmVyKSlcXG4gICAgICAgICAgICBdXFxuICAgICApXFxuXFxuICAgKGRlZnBhY3QgdHJhbnNmZXItY3Jvc3NjaGFpbjpzdHJpbmdcXG4gICAgICggc2VuZGVyOnN0cmluZ1xcbiAgICAgICByZWNlaXZlcjpzdHJpbmdcXG4gICAgICAgcmVjZWl2ZXItZ3VhcmQ6Z3VhcmRcXG4gICAgICAgdGFyZ2V0LWNoYWluOnN0cmluZ1xcbiAgICAgICBhbW91bnQ6ZGVjaW1hbFxcbiAgICAgKVxcbiAgICAgQGRvYyBcXFwiIDItc3RlcCBwYWN0IHRvIHRyYW5zZmVyIEFNT1VOVCBmcm9tIFNFTkRFUiBvbiBjdXJyZW50IGNoYWluIFxcXFxcXG4gICAgICAgICAgXFxcXCB0byBSRUNFSVZFUiBvbiBUQVJHRVQtQ0hBSU4gdmlhIFNQViBwcm9vZi4gXFxcXFxcbiAgICAgICAgICBcXFxcIFRBUkdFVC1DSEFJTiBtdXN0IGJlIGRpZmZlcmVudCB0aGFuIGN1cnJlbnQgY2hhaW4gaWQuIFxcXFxcXG4gICAgICAgICAgXFxcXCBGaXJzdCBzdGVwIGRlYml0cyBBTU9VTlQgY29pbnMgaW4gU0VOREVSIGFjY291bnQgYW5kIHlpZWxkcyBcXFxcXFxuICAgICAgICAgIFxcXFwgUkVDRUlWRVIsIFJFQ0VJVkVSX0dVQVJEIGFuZCBBTU9VTlQgdG8gVEFSR0VULUNIQUlOLiBcXFxcXFxuICAgICAgICAgIFxcXFwgU2Vjb25kIHN0ZXAgY29udGludWF0aW9uIGlzIHNlbnQgaW50byBUQVJHRVQtQ0hBSU4gd2l0aCBwcm9vZiBcXFxcXFxuICAgICAgICAgIFxcXFwgb2J0YWluZWQgZnJvbSB0aGUgc3B2ICdvdXRwdXQnIGVuZHBvaW50IG9mIENoYWlud2ViLiBcXFxcXFxuICAgICAgICAgIFxcXFwgUHJvb2YgaXMgdmFsaWRhdGVkIGFuZCBSRUNFSVZFUiBpcyBjcmVkaXRlZCB3aXRoIEFNT1VOVCBcXFxcXFxuICAgICAgICAgIFxcXFwgY3JlYXRpbmcgYWNjb3VudCB3aXRoIFJFQ0VJVkVSX0dVQVJEIGFzIG5lY2Vzc2FyeS5cXFwiXFxuICAgICBAbW9kZWwgWyAocHJvcGVydHkgKD4gYW1vdW50IDAuMCkpXFxuICAgICAgICAgICAgICAocHJvcGVydHkgKCE9IHNlbmRlciBcXFwiXFxcIikpXFxuICAgICAgICAgICAgICAocHJvcGVydHkgKCE9IHJlY2VpdmVyIFxcXCJcXFwiKSlcXG4gICAgICAgICAgICAgIChwcm9wZXJ0eSAoIT0gc2VuZGVyIHJlY2VpdmVyKSlcXG4gICAgICAgICAgICAgIChwcm9wZXJ0eSAoIT0gdGFyZ2V0LWNoYWluIFxcXCJcXFwiKSlcXG4gICAgICAgICAgICBdXFxuICAgICApXFxuXFxuICAgKGRlZnVuIGdldC1iYWxhbmNlOmRlY2ltYWxcXG4gICAgICggYWNjb3VudDpzdHJpbmcgKVxcbiAgICAgXFxcIiBHZXQgYmFsYW5jZSBmb3IgQUNDT1VOVC4gRmFpbHMgaWYgYWNjb3VudCBkb2VzIG5vdCBleGlzdC5cXFwiXFxuICAgICApXFxuXFxuICAgKGRlZnVuIGRldGFpbHM6b2JqZWN0e2FjY291bnQtZGV0YWlsc31cXG4gICAgICggYWNjb3VudDogc3RyaW5nIClcXG4gICAgIFxcXCIgR2V0IGFuIG9iamVjdCB3aXRoIGRldGFpbHMgb2YgQUNDT1VOVC4gXFxcXFxcbiAgICAgXFxcXCBGYWlscyBpZiBhY2NvdW50IGRvZXMgbm90IGV4aXN0LlxcXCJcXG4gICAgIClcXG5cXG4gICAoZGVmdW4gcHJlY2lzaW9uOmludGVnZXJcXG4gICAgICgpXFxuICAgICBcXFwiUmV0dXJuIHRoZSBtYXhpbXVtIGFsbG93ZWQgZGVjaW1hbCBwcmVjaXNpb24uXFxcIlxcbiAgICAgKVxcblxcbiAgIChkZWZ1biBlbmZvcmNlLXVuaXQ6Ym9vbFxcbiAgICAgKCBhbW91bnQ6ZGVjaW1hbCApXFxuICAgICBcXFwiIEVuZm9yY2UgbWluaW11bSBwcmVjaXNpb24gYWxsb3dlZCBmb3IgdHJhbnNhY3Rpb25zLlxcXCJcXG4gICAgIClcXG5cXG4gICAoZGVmdW4gY3JlYXRlLWFjY291bnQ6c3RyaW5nXFxuICAgICAoIGFjY291bnQ6c3RyaW5nXFxuICAgICAgIGd1YXJkOmd1YXJkXFxuICAgICApXFxuICAgICBcXFwiIENyZWF0ZSBBQ0NPVU5UIHdpdGggMC4wIGJhbGFuY2UsIHdpdGggR1VBUkQgY29udHJvbGxpbmcgYWNjZXNzLlxcXCJcXG4gICAgIClcXG5cXG4gICAoZGVmdW4gcm90YXRlOnN0cmluZ1xcbiAgICAgKCBhY2NvdW50OnN0cmluZ1xcbiAgICAgICBuZXctZ3VhcmQ6Z3VhcmRcXG4gICAgIClcXG4gICAgIFxcXCIgUm90YXRlIGd1YXJkIGZvciBBQ0NPVU5ULiBUcmFuc2FjdGlvbiBpcyB2YWxpZGF0ZWQgYWdhaW5zdCBcXFxcXFxuICAgICBcXFxcIGV4aXN0aW5nIGd1YXJkIGJlZm9yZSBpbnN0YWxsaW5nIG5ldyBndWFyZC4gXFxcIlxcbiAgICAgKVxcblxcbilcXG5cIn19LFwic2lnbmVyc1wiOltdLFwibWV0YVwiOntcImNyZWF0aW9uVGltZVwiOjAsXCJ0dGxcIjoxNzI4MDAsXCJnYXNMaW1pdFwiOjAsXCJjaGFpbklkXCI6XCJcIixcImdhc1ByaWNlXCI6MCxcInNlbmRlclwiOlwiXCJ9LFwibm9uY2VcIjpcImdlbmVzaXMtMDFcIn0ifQ", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IkxvYWRlZCBpbnRlcmZhY2UgZnVuZ2libGUtdjEifSwicmVxS2V5IjoiNDhUMExqQW5TRnBGV3h2dmFQVi1fNkUtQ2pEQVBoV1lVRldidnlmMmxGcyIsImxvZ3MiOiJRRmEyOHRuOXkydFdMVzdUc3lHdzNOTkhpYzhxYjJ6UGtudXRpWWhnQXc4IiwibWV0YURhdGEiOm51bGwsImNvbnRpbnVhdGlvbiI6bnVsbCwidHhJZCI6MH0") + , (unsafeFromText "{"hash":"XGPEQDk5PIvQkpq0GGkgNTmo-mjki63ZPgER_kovxq4","sigs":[],"cmd":"{\"networkId\":null,\"payload\":{\"exec\":{\"data\":null,\"code\":\"(module coin GOVERNANCE\\n\\n  @doc \\\"'coin' represents the Kadena Coin Contract. This contract provides both the \\\\\\n  \\\\buy/redeem gas support in the form of 'fund-tx', as well as transfer,       \\\\\\n  \\\\credit, debit, coinbase, account creation and query, as well as SPV burn    \\\\\\n  \\\\create. To access the coin contract, you may use its fully-qualified name,  \\\\\\n  \\\\or issue the '(use coin)' command in the body of a module declaration.\\\"\\n\\n  @model\\n    [ (defproperty conserves-mass\\n        (= (column-delta coin-table 'balance) 0.0))\\n\\n      (defproperty valid-account (account:string)\\n        (and\\n          (>= (length account) 3)\\n          (<= (length account) 256)))\\n    ]\\n\\n  (implements fungible-v1)\\n\\n  ; --------------------------------------------------------------------------\\n  ; Schemas and Tables\\n\\n  (defschema coin-schema\\n    @doc \\\"The coin contract token schema\\\"\\n    @model [ (invariant (>= balance 0.0)) ]\\n\\n    balance:decimal\\n    guard:guard)\\n\\n  (deftable coin-table:{coin-schema})\\n\\n  ; --------------------------------------------------------------------------\\n  ; Capabilities\\n\\n  (defcap GOVERNANCE ()\\n    (enforce false \\\"Enforce non-upgradeability\\\"))\\n\\n  (defcap GAS ()\\n    \\\"Magic capability to protect gas buy and redeem\\\"\\n    true)\\n\\n  (defcap COINBASE ()\\n    \\\"Magic capability to protect miner reward\\\"\\n    true)\\n\\n  (defcap GENESIS ()\\n    \\\"Magic capability constraining genesis transactions\\\"\\n    true)\\n\\n  (defcap DEBIT (sender:string)\\n    \\\"Capability for managing debiting operations\\\"\\n    (enforce-guard (at 'guard (read coin-table sender)))\\n    (enforce (!= sender \\\"\\\") \\\"valid sender\\\"))\\n\\n  (defcap CREDIT (receiver:string)\\n    \\\"Capability for managing crediting operations\\\"\\n    (enforce (!= receiver \\\"\\\") \\\"valid receiver\\\"))\\n\\n  (defcap TRANSFER:bool\\n    ( sender:string\\n      receiver:string\\n      amount:decimal\\n    )\\n    @managed amount TRANSFER-mgr\\n    (enforce (!= sender receiver) \\\"same sender and receiver\\\")\\n    (enforce-unit amount)\\n    (enforce (> amount 0.0) \\\"Positive amount\\\")\\n    (compose-capability (DEBIT sender))\\n    (compose-capability (CREDIT receiver))\\n  )\\n\\n  (defun TRANSFER-mgr:decimal\\n    ( managed:decimal\\n      requested:decimal\\n    )\\n\\n    (let ((newbal (- managed requested)))\\n      (enforce (>= newbal 0.0)\\n        (format \\\"TRANSFER exceeded for balance {}\\\" [managed]))\\n      newbal)\\n  )\\n\\n  ; --------------------------------------------------------------------------\\n  ; Constants\\n\\n  (defconst COIN_CHARSET CHARSET_LATIN1\\n    \\\"The default coin contract character set\\\")\\n\\n  (defconst MINIMUM_PRECISION 12\\n    \\\"Minimum allowed precision for coin transactions\\\")\\n\\n  (defconst MINIMUM_ACCOUNT_LENGTH 3\\n    \\\"Minimum account length admissible for coin accounts\\\")\\n\\n  (defconst MAXIMUM_ACCOUNT_LENGTH 256\\n    \\\"Maximum account name length admissible for coin accounts\\\")\\n\\n  ; --------------------------------------------------------------------------\\n  ; Utilities\\n\\n  (defun enforce-unit:bool (amount:decimal)\\n    @doc \\\"Enforce minimum precision allowed for coin transactions\\\"\\n\\n    (enforce\\n      (= (floor amount MINIMUM_PRECISION)\\n         amount)\\n      (format \\\"Amount violates minimum precision: {}\\\" [amount]))\\n    )\\n\\n  (defun validate-account (account:string)\\n    @doc \\\"Enforce that an account name conforms to the coin contract \\\\\\n         \\\\minimum and maximum length requirements, as well as the    \\\\\\n         \\\\latin-1 character set.\\\"\\n\\n    (enforce\\n      (is-charset COIN_CHARSET account)\\n      (format\\n        \\\"Account does not conform to the coin contract charset: {}\\\"\\n        [account]))\\n\\n    (let ((account-length (length account)))\\n\\n      (enforce\\n        (>= account-length MINIMUM_ACCOUNT_LENGTH)\\n        (format\\n          \\\"Account name does not conform to the min length requirement: {}\\\"\\n          [account]))\\n\\n      (enforce\\n        (<= account-length MAXIMUM_ACCOUNT_LENGTH)\\n        (format\\n          \\\"Account name does not conform to the max length requirement: {}\\\"\\n          [account]))\\n      )\\n  )\\n\\n  ; --------------------------------------------------------------------------\\n  ; Coin Contract\\n\\n  (defun gas-only ()\\n    \\\"Predicate for gas-only user guards.\\\"\\n    (require-capability (GAS)))\\n\\n  (defun gas-guard (guard:guard)\\n    \\\"Predicate for gas + single key user guards\\\"\\n    (enforce-one\\n      \\\"Enforce either the presence of a GAS cap or keyset\\\"\\n      [ (gas-only)\\n        (enforce-guard guard)\\n      ]))\\n\\n  (defun buy-gas:string (sender:string total:decimal)\\n    @doc \\\"This function describes the main 'gas buy' operation. At this point \\\\\\n    \\\\MINER has been chosen from the pool, and will be validated. The SENDER   \\\\\\n    \\\\of this transaction has specified a gas limit LIMIT (maximum gas) for    \\\\\\n    \\\\the transaction, and the price is the spot price of gas at that time.    \\\\\\n    \\\\The gas buy will be executed prior to executing SENDER's code.\\\"\\n\\n    @model [ (property (> total 0.0))\\n             (property (valid-account sender))\\n           ]\\n\\n    (validate-account sender)\\n\\n    (enforce-unit total)\\n    (enforce (> total 0.0) \\\"gas supply must be a positive quantity\\\")\\n\\n    (require-capability (GAS))\\n    (with-capability (DEBIT sender)\\n      (debit sender total))\\n    )\\n\\n  (defun redeem-gas:string (miner:string miner-guard:guard sender:string total:decimal)\\n    @doc \\\"This function describes the main 'redeem gas' operation. At this    \\\\\\n    \\\\point, the SENDER's transaction has been executed, and the gas that      \\\\\\n    \\\\was charged has been calculated. MINER will be credited the gas cost,    \\\\\\n    \\\\and SENDER will receive the remainder up to the limit\\\"\\n\\n    @model [ (property (> total 0.0))\\n             (property (valid-account sender))\\n             (property (valid-account miner))\\n           ]\\n\\n    (validate-account sender)\\n    (validate-account miner)\\n    (enforce-unit total)\\n\\n    (require-capability (GAS))\\n    (let*\\n      ((fee (read-decimal \\\"fee\\\"))\\n       (refund (- total fee)))\\n\\n      (enforce-unit fee)\\n      (enforce (>= fee 0.0)\\n        \\\"fee must be a non-negative quantity\\\")\\n\\n      (enforce (>= refund 0.0)\\n        \\\"refund must be a non-negative quantity\\\")\\n\\n        ; directly update instead of credit\\n      (with-capability (CREDIT sender)\\n        (if (> refund 0.0)\\n          (with-read coin-table sender\\n            { \\\"balance\\\" := balance }\\n            (update coin-table sender\\n              { \\\"balance\\\": (+ balance refund) }))\\n\\n          \\\"noop\\\"))\\n\\n      (with-capability (CREDIT miner)\\n        (if (> fee 0.0)\\n          (credit miner miner-guard fee)\\n          \\\"noop\\\"))\\n      )\\n\\n    )\\n\\n  (defun create-account:string (account:string guard:guard)\\n    @model [ (property (valid-account account)) ]\\n\\n    (validate-account account)\\n\\n    (insert coin-table account\\n      { \\\"balance\\\" : 0.0\\n      , \\\"guard\\\"   : guard\\n      })\\n    )\\n\\n  (defun get-balance:decimal (account:string)\\n    (with-read coin-table account\\n      { \\\"balance\\\" := balance }\\n      balance\\n      )\\n    )\\n\\n  (defun details:object{fungible-v1.account-details}\\n    ( account:string )\\n    (with-read coin-table account\\n      { \\\"balance\\\" := bal\\n      , \\\"guard\\\" := g }\\n      { \\\"account\\\" : account\\n      , \\\"balance\\\" : bal\\n      , \\\"guard\\\": g })\\n    )\\n\\n  (defun rotate:string (account:string new-guard:guard)\\n\\n    (with-read coin-table account\\n      { \\\"guard\\\" := old-guard }\\n\\n      (enforce-guard old-guard)\\n      (enforce-guard new-guard)\\n\\n      (update coin-table account\\n        { \\\"guard\\\" : new-guard }\\n        )))\\n\\n\\n  (defun precision:integer\\n    ()\\n    MINIMUM_PRECISION)\\n\\n  (defun transfer:string (sender:string receiver:string amount:decimal)\\n    @model [ (property conserves-mass)\\n             (property (> amount 0.0))\\n             (property (valid-account sender))\\n             (property (valid-account receiver))\\n             (property (!= sender receiver)) ]\\n\\n    (enforce (!= sender receiver)\\n      \\\"sender cannot be the receiver of a transfer\\\")\\n\\n    (validate-account sender)\\n    (validate-account receiver)\\n\\n    (enforce (> amount 0.0)\\n      \\\"transfer amount must be positive\\\")\\n\\n    (enforce-unit amount)\\n\\n    (with-capability (TRANSFER sender receiver amount)\\n      (debit sender amount)\\n      (with-read coin-table receiver\\n        { \\\"guard\\\" := g }\\n\\n        (credit receiver g amount))\\n      )\\n    )\\n\\n  (defun transfer-create:string\\n    ( sender:string\\n      receiver:string\\n      receiver-guard:guard\\n      amount:decimal )\\n\\n    @model [ (property conserves-mass) ]\\n\\n    (enforce (!= sender receiver)\\n      \\\"sender cannot be the receiver of a transfer\\\")\\n\\n    (validate-account sender)\\n    (validate-account receiver)\\n\\n    (enforce (> amount 0.0)\\n      \\\"transfer amount must be positive\\\")\\n\\n    (enforce-unit amount)\\n\\n    (with-capability (TRANSFER sender receiver amount)\\n      (debit sender amount)\\n      (credit receiver receiver-guard amount))\\n    )\\n\\n  (defun coinbase:string (account:string account-guard:guard amount:decimal)\\n    @doc \\\"Internal function for the initial creation of coins.  This function \\\\\\n    \\\\cannot be used outside of the coin contract.\\\"\\n\\n    @model [ (property (valid-account account)) ]\\n\\n    (validate-account account)\\n    (enforce-unit amount)\\n\\n    (require-capability (COINBASE))\\n    (with-capability (CREDIT account)\\n      (credit account account-guard amount))\\n    )\\n\\n  (defpact fund-tx (sender:string miner:string miner-guard:guard total:decimal)\\n    @doc \\\"'fund-tx' is a special pact to fund a transaction in two steps,     \\\\\\n    \\\\with the actual transaction transpiring in the middle:                   \\\\\\n    \\\\                                                                         \\\\\\n    \\\\  1) A buying phase, debiting the sender for total gas and fee, yielding \\\\\\n    \\\\     TX_MAX_CHARGE.                                                      \\\\\\n    \\\\  2) A settlement phase, resuming TX_MAX_CHARGE, and allocating to the   \\\\\\n    \\\\     coinbase account for used gas and fee, and sender account for bal-  \\\\\\n    \\\\     ance (unused gas, if any).\\\"\\n\\n    @model [ (property (> total 0.0))\\n             (property (valid-account sender))\\n             (property (valid-account miner))\\n             ;(property conserves-mass) not supported yet\\n           ]\\n\\n    (step (buy-gas sender total))\\n    (step (redeem-gas miner miner-guard sender total))\\n    )\\n\\n  (defun debit:string (account:string amount:decimal)\\n    @doc \\\"Debit AMOUNT from ACCOUNT balance\\\"\\n\\n    @model [ (property (> amount 0.0))\\n             (property (valid-account account))\\n           ]\\n\\n    (validate-account account)\\n\\n    (enforce (> amount 0.0)\\n      \\\"debit amount must be positive\\\")\\n\\n    (enforce-unit amount)\\n\\n    (require-capability (DEBIT account))\\n    (with-read coin-table account\\n      { \\\"balance\\\" := balance }\\n\\n      (enforce (<= amount balance) \\\"Insufficient funds\\\")\\n\\n      (update coin-table account\\n        { \\\"balance\\\" : (- balance amount) }\\n        ))\\n    )\\n\\n\\n  (defun credit:string (account:string guard:guard amount:decimal)\\n    @doc \\\"Credit AMOUNT to ACCOUNT balance\\\"\\n\\n    @model [ (property (> amount 0.0))\\n             (property (valid-account account))\\n           ]\\n\\n    (validate-account account)\\n\\n    (enforce (> amount 0.0) \\\"credit amount must be positive\\\")\\n    (enforce-unit amount)\\n\\n    (require-capability (CREDIT account))\\n    (with-default-read coin-table account\\n      { \\\"balance\\\" : 0.0, \\\"guard\\\" : guard }\\n      { \\\"balance\\\" := balance, \\\"guard\\\" := retg }\\n      ; we don't want to overwrite an existing guard with the user-supplied one\\n      (enforce (= retg guard)\\n        \\\"account guards do not match\\\")\\n\\n      (write coin-table account\\n        { \\\"balance\\\" : (+ balance amount)\\n        , \\\"guard\\\"   : retg\\n        })\\n      ))\\n\\n\\n  (defschema crosschain-schema\\n    @doc \\\"Schema for yielded value in cross-chain transfers\\\"\\n    receiver:string\\n    receiver-guard:guard\\n    amount:decimal)\\n\\n  (defpact transfer-crosschain:string\\n    ( sender:string\\n      receiver:string\\n      receiver-guard:guard\\n      target-chain:string\\n      amount:decimal )\\n\\n    @model [ (property (> amount 0.0))\\n             (property (!= receiver \\\"\\\"))\\n             (property (valid-account sender))\\n             (property (valid-account receiver))\\n           ]\\n\\n    (step\\n      (with-capability (DEBIT sender)\\n\\n        (validate-account sender)\\n        (validate-account receiver)\\n\\n        (enforce (!= \\\"\\\" target-chain) \\\"empty target-chain\\\")\\n        (enforce (!= (at 'chain-id (chain-data)) target-chain)\\n          \\\"cannot run cross-chain transfers to the same chain\\\")\\n\\n        (enforce (> amount 0.0)\\n          \\\"transfer quantity must be positive\\\")\\n\\n        (enforce-unit amount)\\n\\n        ;; step 1 - debit delete-account on current chain\\n        (debit sender amount)\\n\\n        (let\\n          ((crosschain-details:object{crosschain-schema}\\n            { \\\"receiver\\\" : receiver\\n            , \\\"receiver-guard\\\" : receiver-guard\\n            , \\\"amount\\\" : amount\\n            }))\\n          (yield crosschain-details target-chain)\\n          )))\\n\\n    (step\\n      (resume\\n        { \\\"receiver\\\" := receiver\\n        , \\\"receiver-guard\\\" := receiver-guard\\n        , \\\"amount\\\" := amount\\n        }\\n\\n        ;; step 2 - credit create account on target chain\\n        (with-capability (CREDIT receiver)\\n          (credit receiver receiver-guard amount))\\n        ))\\n    )\\n\\n\\n  ; --------------------------------------------------------------------------\\n  ; Coin allocations\\n\\n  (defschema allocation-schema\\n    @doc \\\"Genesis allocation registry\\\"\\n    ;@model [ (invariant (>= balance 0.0)) ]\\n\\n    balance:decimal\\n    date:time\\n    guard:guard\\n    redeemed:bool)\\n\\n  (deftable allocation-table:{allocation-schema})\\n\\n  (defun create-allocation-account\\n    ( account:string\\n      date:time\\n      keyset-ref:string\\n      amount:decimal\\n    )\\n\\n    @doc \\\"Add an entry to the coin allocation table. This function \\\\\\n         \\\\also creates a corresponding empty coin contract account \\\\\\n         \\\\of the same name and guard. Requires GENESIS capability. \\\"\\n\\n    @model [ (property (valid-account account)) ]\\n\\n    (require-capability (GENESIS))\\n\\n    (validate-account account)\\n    (enforce (>= amount 0.0)\\n      \\\"allocation amount must be non-negative\\\")\\n\\n    (enforce-unit amount)\\n\\n    (let\\n      ((guard:guard (keyset-ref-guard keyset-ref)))\\n\\n      (create-account account guard)\\n\\n      (insert allocation-table account\\n        { \\\"balance\\\" : amount\\n        , \\\"date\\\" : date\\n        , \\\"guard\\\" : guard\\n        , \\\"redeemed\\\" : false\\n        })))\\n\\n  (defun release-allocation\\n    ( account:string )\\n\\n    @doc \\\"Release funds associated with allocation ACCOUNT into main ledger.   \\\\\\n         \\\\ACCOUNT must already exist in main ledger. Allocation is deactivated \\\\\\n         \\\\after release.\\\"\\n    @model [ (property (valid-account account)) ]\\n\\n    (validate-account account)\\n\\n    (with-read allocation-table account\\n      { \\\"balance\\\" := balance\\n      , \\\"date\\\" := release-time\\n      , \\\"redeemed\\\" := redeemed\\n      , \\\"guard\\\" := guard\\n      }\\n\\n      (let ((curr-time:time (at 'block-time (chain-data))))\\n\\n        (enforce (not redeemed)\\n          \\\"allocation funds have already been redeemed\\\")\\n\\n        (enforce\\n          (>= curr-time release-time)\\n          (format \\\"funds locked until {}. current time: {}\\\" [release-time curr-time]))\\n\\n        (enforce-guard guard)\\n\\n        (with-capability (CREDIT account)\\n          (credit account guard balance)\\n\\n          (update allocation-table account\\n            { \\\"redeemed\\\" : true\\n            , \\\"balance\\\" : 0.0\\n            })\\n\\n          \\\"Allocation successfully released to main ledger\\\")\\n    )))\\n\\n)\\n\\n(create-table coin-table)\\n(create-table allocation-table)\\n\"}},\"signers\":[],\"meta\":{\"creationTime\":0,\"ttl\":172800,\"gasLimit\":0,\"chainId\":\"\",\"gasPrice\":0,\"sender\":\"\"},\"nonce\":\"genesis-01\"}"}", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IlRhYmxlQ3JlYXRlZCJ9LCJyZXFLZXkiOiJYR1BFUURrNVBJdlFrcHEwR0drZ05UbW8tbWpraTYzWlBnRVJfa292eHE0IiwibG9ncyI6IkpvNXhfMEl0dnZLVTY5dHJFbjh1a3FCeWpSTy1XaEZJZFlmMDV5dDlfN0UiLCJtZXRhRGF0YSI6bnVsbCwiY29udGludWF0aW9uIjpudWxsLCJ0eElkIjoxfQ") + , (unsafeFromText "eyJoYXNoIjoiU0IzVzVFTGl6azl4elNWWk9MX3dsem5VNjh5aUhPQzlwWUhreHBVXzBnbyIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6bnVsbCxcImNvZGVcIjpcIihpbnRlcmZhY2UgZ2FzLXBheWVyLXYxXFxuXFxuICAoZGVmY2FwIEdBU19QQVlFUjpib29sXFxuICAgICggdXNlcjpzdHJpbmdcXG4gICAgICBsaW1pdDppbnRlZ2VyXFxuICAgICAgcHJpY2U6ZGVjaW1hbFxcbiAgICApXFxuICAgIEBkb2NcXG4gICAgXFxcIiBQcm92aWRlIGEgY2FwYWJpbGl0eSBpbmRpY2F0aW5nIHRoYXQgZGVjbGFyaW5nIG1vZHVsZSBzdXBwb3J0cyBcXFxcXFxuICAgIFxcXFwgZ2FzIHBheW1lbnQgZm9yIFVTRVIgZm9yIGdhcyBMSU1JVCBhbmQgUFJJQ0UuIEZ1bmN0aW9uYWxpdHkgXFxcXFxcbiAgICBcXFxcIHNob3VsZCByZXF1aXJlIGNhcGFiaWxpdHkgKGNvaW4uRlVORF9UWCksIGFuZCBzaG91bGQgdmFsaWRhdGUgXFxcXFxcbiAgICBcXFxcIHRoZSBzcGVuZCBvZiAobGltaXQgKiBwcmljZSksIHBvc3NpYmx5IHVwZGF0aW5nIHNvbWUgZGF0YWJhc2UgXFxcXFxcbiAgICBcXFxcIGVudHJ5LiBcXFxcXFxuICAgIFxcXFwgU2hvdWxkIGNvbXBvc2UgY2FwYWJpbGl0eSByZXF1aXJlZCBmb3IgJ2NyZWF0ZS1nYXMtcGF5ZXItZ3VhcmQnLlxcXCJcXG4gICAgQG1vZGVsXFxuICAgIFsgKHByb3BlcnR5ICh1c2VyICE9IFxcXCJcXFwiKSlcXG4gICAgICAocHJvcGVydHkgKGxpbWl0ID4gMCkpXFxuICAgICAgKHByb3BlcnR5IChwcmljZSA-IDAuMCkpXFxuICAgIF1cXG4gIClcXG5cXG4gIChkZWZ1biBjcmVhdGUtZ2FzLXBheWVyLWd1YXJkOmd1YXJkICgpXFxuICAgIEBkb2NcXG4gICAgXFxcIiBQcm92aWRlIGEgZ3VhcmQgc3VpdGFibGUgZm9yIGNvbnRyb2xsaW5nIGEgY29pbiBhY2NvdW50IHRoYXQgY2FuIFxcXFxcXG4gICAgXFxcXCBwYXkgZ2FzIHZpYSBHQVNfUEFZRVIgbWVjaGFuaWNzLiBHZW5lcmFsbHkgdGhpcyBpcyBhY2NvbXBsaXNoZWQgXFxcXFxcbiAgICBcXFxcIGJ5IGhhdmluZyBHQVNfUEFZRVIgY29tcG9zZSBhbiB1bnBhcmFtZXRlcml6ZWQsIHVubWFuYWdlZCBjYXBhYmlsaXR5IFxcXFxcXG4gICAgXFxcXCB0aGF0IGlzIHJlcXVpcmVkIGluIHRoaXMgZ3VhcmQuIFRodXMsIGlmIGNvaW4gY29udHJhY3QgaXMgYWJsZSB0byBcXFxcXFxuICAgIFxcXFwgc3VjY2Vzc2Z1bGx5IGFjcXVpcmUgR0FTX1BBWUVSLCB0aGUgY29tcG9zZWQgJ2Fub255bW91cycgY2FwIHJlcXVpcmVkIFxcXFxcXG4gICAgXFxcXCBoZXJlIHdpbGwgYmUgaW4gc2NvcGUsIGFuZCBnYXMgYnV5IHdpbGwgc3VjY2VlZC5cXFwiXFxuICApXFxuXFxuKVxcblwifX0sXCJzaWduZXJzXCI6W10sXCJtZXRhXCI6e1wiY3JlYXRpb25UaW1lXCI6MCxcInR0bFwiOjE3MjgwMCxcImdhc0xpbWl0XCI6MCxcImNoYWluSWRcIjpcIlwiLFwiZ2FzUHJpY2VcIjowLFwic2VuZGVyXCI6XCJcIn0sXCJub25jZVwiOlwiZ2VuZXNpcy0wMVwifSJ9", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IkxvYWRlZCBpbnRlcmZhY2UgZ2FzLXBheWVyLXYxIn0sInJlcUtleSI6IlNCM1c1RUxpems5eHpTVlpPTF93bHpuVTY4eWlIT0M5cFlIa3hwVV8wZ28iLCJsb2dzIjoiZlZuSFlta19QNmJSY3VjeVg1RDdLamNLYkVsVDlEcU9vZW9yUFEtUXdsMCIsIm1ldGFEYXRhIjpudWxsLCJjb250aW51YXRpb24iOm51bGwsInR4SWQiOjJ9") + , (unsafeFromText "eyJoYXNoIjoia2ZMd2Y2a0FzdEVnc0NLYnZPOHR2YTNnWktBWXgzT0dHYTZYRURMaU9hMCIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6e1wibnMtYWRtaW4ta2V5c2V0XCI6W1wiMzY4ODIwZjgwYzMyNGJiYzdjMmIwNjEwNjg4YTdkYTQzZTM5ZjkxZDExODczMjY3MWNkOWM3NTAwZmY0M2NjYVwiXSxcIm5zLW9wZXJhdGUta2V5c2V0XCI6W1wiMzY4ODIwZjgwYzMyNGJiYzdjMmIwNjEwNjg4YTdkYTQzZTM5ZjkxZDExODczMjY3MWNkOWM3NTAwZmY0M2NjYVwiXSxcIm5zLWdlbmVzaXMta2V5c2V0XCI6e1wicHJlZFwiOlwiPVwiLFwia2V5c1wiOltdfX0sXCJjb2RlXCI6XCJcXG4oZGVmaW5lLWtleXNldCAnbnMtYWRtaW4ta2V5c2V0IChyZWFkLWtleXNldCAnbnMtYWRtaW4ta2V5c2V0KSlcXG4oZGVmaW5lLWtleXNldCAnbnMtb3BlcmF0ZS1rZXlzZXQgKHJlYWQta2V5c2V0ICducy1nZW5lc2lzLWtleXNldCkpXFxuXFxuKG1vZHVsZSBucyBHT1ZFUk5BTkNFXFxuICBcXFwiQWRtaW5pc3RlcnMgZGVmaW5pdGlvbiBvZiBuZXcgbmFtZXNwYWNlcyBpbiBDaGFpbndlYi5cXFwiXFxuXFxuICAoZGVmc2NoZW1hIHJlZy1lbnRyeVxcbiAgICBhZG1pbi1ndWFyZDpndWFyZFxcbiAgICBhY3RpdmU6Ym9vbClcXG5cXG4gIChkZWZ0YWJsZSByZWdpc3RyeTp7cmVnLWVudHJ5fSlcXG5cXG4gIChkZWZjYXAgR09WRVJOQU5DRSAoKVxcbiAgICAoZW5mb3JjZS1rZXlzZXQgJ25zLWFkbWluLWtleXNldCkpXFxuXFxuICAoZGVmY2FwIE9QRVJBVEUgKClcXG4gICAgKGVuZm9yY2Uta2V5c2V0ICducy1vcGVyYXRlLWtleXNldCkpXFxuXFxuICAoZGVmY29uc3QgR1VBUkRfU1VDQ0VTUyAoY3JlYXRlLXVzZXItZ3VhcmQgKHN1Y2Nlc3MpKSlcXG4gIChkZWZjb25zdCBHVUFSRF9GQUlMVVJFIChjcmVhdGUtdXNlci1ndWFyZCAoZmFpbHVyZSkpKVxcblxcbiAgKGRlZnVuIHN1Y2Nlc3MgKClcXG4gICAgdHJ1ZSlcXG4gIChkZWZ1biBmYWlsdXJlICgpXFxuICAgIChlbmZvcmNlIGZhbHNlIFxcXCJEaXNhYmxlZFxcXCIpKVxcblxcbiAgKGRlZnVuIHZhbGlkYXRlLW5hbWUgKG5hbWUpXFxuICAgIChlbmZvcmNlICghPSBcXFwiXFxcIiBuYW1lKSBcXFwiRW1wdHkgbmFtZSBub3QgYWxsb3dlZFxcXCIpXFxuICAgIChlbmZvcmNlICg8IChsZW5ndGggbmFtZSkgNjQpIFxcXCJOYW1lIG11c3QgYmUgbGVzcyB0aGFuIDY0IGNoYXJhY3RlcnMgbG9uZ1xcXCIpXFxuICAgIChlbmZvcmNlIChpcy1jaGFyc2V0IENIQVJTRVRfTEFUSU4xIG5hbWUpXFxuICAgICAgICAgICAgIFxcXCJOYW1lIG11c3QgYmUgaW4gbGF0aW4xIGNoYXJzZXRcXFwiKSlcXG5cXG4gIChkZWZ1biB2YWxpZGF0ZTpib29sXFxuICAgICAgKCBucy1uYW1lOnN0cmluZ1xcbiAgICAgICAgbnMtYWRtaW46Z3VhcmRcXG4gICAgICAgIClcXG4gICAgXFxcIiBNYW5hZ2VzIG5hbWVzcGFjZSBpbnN0YWxsIGZvciBDaGFpbndlYi4gUmVxdWlyZXMgYWN0aXZlIHJvdyBpbiByZWdpc3RyeSBcXFxcXFxuICAgIFxcXFwgZm9yIE5TLU5BTUUgd2l0aCBndWFyZCBtYXRjaGluZyBOUy1BRE1JTi5cXFwiXFxuXFxuICAgICh2YWxpZGF0ZS1uYW1lIG5zLW5hbWUpXFxuXFxuICAgICh3aXRoLWRlZmF1bHQtcmVhZCByZWdpc3RyeSBucy1uYW1lXFxuICAgICAgeyAnYWRtaW4tZ3VhcmQgOiBucy1hZG1pblxcbiAgICAgICwgJ2FjdGl2ZSA6IGZhbHNlIH1cXG4gICAgICB7ICdhZG1pbi1ndWFyZCA6PSBhZ1xcbiAgICAgICwgJ2FjdGl2ZSA6PSBpcy1hY3RpdmUgfVxcblxcbiAgICAgICAgKGVuZm9yY2UgaXMtYWN0aXZlIFxcXCJJbmFjdGl2ZSBvciB1bnJlZ2lzdGVyZWQgbmFtZXNwYWNlXFxcIilcXG4gICAgICAgIChlbmZvcmNlICg9IG5zLWFkbWluIGFnKSBcXFwiQWRtaW4gZ3VhcmQgbXVzdCBtYXRjaCBndWFyZCBpbiByZWdpc3RyeVxcXCIpXFxuXFxuICAgICAgICB0cnVlKSlcXG5cXG4gIChkZWZ1biB3cml0ZS1yZWdpc3RyeTpzdHJpbmdcXG4gICAgICAoIG5zLW5hbWU6c3RyaW5nXFxuICAgICAgICBndWFyZDpndWFyZFxcbiAgICAgICAgYWN0aXZlOmJvb2xcXG4gICAgICAgIClcXG4gICAgXFxcIiBXcml0ZSBlbnRyeSB3aXRoIEdVQVJEIGFuZCBBQ1RJVkUgaW50byByZWdpc3RyeSBmb3IgTkFNRS4gXFxcXFxcbiAgICBcXFxcIEd1YXJkZWQgYnkgb3BlcmF0ZSBrZXlzZXQuIFxcXCJcXG5cXG4gICAgKHdpdGgtY2FwYWJpbGl0eSAoT1BFUkFURSlcXG5cXG4gICAgICAodmFsaWRhdGUtbmFtZSBucy1uYW1lKVxcblxcbiAgICAgICh3cml0ZSByZWdpc3RyeSBucy1uYW1lXFxuICAgICAgICB7ICdhZG1pbi1ndWFyZDogZ3VhcmRcXG4gICAgICAgICwgJ2FjdGl2ZTogYWN0aXZlIH0pXFxuXFxuICAgICAgXFxcIlJlZ2lzdGVyIGVudHJ5IHdyaXR0ZW5cXFwiKSlcXG5cXG4gIChkZWZ1biBxdWVyeTpvYmplY3R7cmVnLWVudHJ5fVxcbiAgICAgICggbnMtbmFtZTpzdHJpbmcgKVxcbiAgICAocmVhZCByZWdpc3RyeSBucy1uYW1lKSlcXG5cXG4gIClcXG5cXG4oY3JlYXRlLXRhYmxlIHJlZ2lzdHJ5KVxcblxcbih3cml0ZS1yZWdpc3RyeSBcXFwia2FkZW5hXFxcIlxcbiAgKGtleXNldC1yZWYtZ3VhcmQgJ25zLW9wZXJhdGUta2V5c2V0KSB0cnVlKVxcbih3cml0ZS1yZWdpc3RyeSBcXFwidXNlclxcXCIgR1VBUkRfRkFJTFVSRSB0cnVlKVxcbih3cml0ZS1yZWdpc3RyeSBcXFwiZnJlZVxcXCIgR1VBUkRfRkFJTFVSRSB0cnVlKVxcblxcbihkZWZpbmUtbmFtZXNwYWNlIFxcXCJrYWRlbmFcXFwiXFxuICAoa2V5c2V0LXJlZi1ndWFyZCAnbnMtb3BlcmF0ZS1rZXlzZXQpXFxuICAoa2V5c2V0LXJlZi1ndWFyZCAnbnMtb3BlcmF0ZS1rZXlzZXQpKVxcblxcbihkZWZpbmUtbmFtZXNwYWNlIFxcXCJ1c2VyXFxcIiBHVUFSRF9TVUNDRVNTIEdVQVJEX0ZBSUxVUkUpXFxuKGRlZmluZS1uYW1lc3BhY2UgXFxcImZyZWVcXFwiIEdVQVJEX1NVQ0NFU1MgR1VBUkRfRkFJTFVSRSlcXG47O3JvdGF0ZSB0byByZWFsIG9wZXJhdGUga2V5c2V0XFxuKGRlZmluZS1rZXlzZXQgJ25zLW9wZXJhdGUta2V5c2V0IChyZWFkLWtleXNldCAnbnMtb3BlcmF0ZS1rZXlzZXQpKVxcblwifX0sXCJzaWduZXJzXCI6W10sXCJtZXRhXCI6e1wiY3JlYXRpb25UaW1lXCI6MCxcInR0bFwiOjE3MjgwMCxcImdhc0xpbWl0XCI6MCxcImNoYWluSWRcIjpcIlwiLFwiZ2FzUHJpY2VcIjowLFwic2VuZGVyXCI6XCJcIn0sXCJub25jZVwiOlwibG9hZC1ucy1kZXZuZXQtc2VuZGVyMDBcIn0ifQ", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IktleXNldCBkZWZpbmVkIn0sInJlcUtleSI6ImtmTHdmNmtBc3RFZ3NDS2J2Tzh0dmEzZ1pLQVl4M09HR2E2WEVETGlPYTAiLCJsb2dzIjoiZ1BaazFQZzY5UDZUYXlTSEpxZE9QeEthdHc5WTV3eHZXdjNBRFRqMHVRdyIsIm1ldGFEYXRhIjpudWxsLCJjb250aW51YXRpb24iOm51bGwsInR4SWQiOjN9") + , (unsafeFromText "eyJoYXNoIjoiYWlyX21QQWdzOW1hOWlaMS1FdHhYcjV5X1E0bm9xZ0k1T245MjZsWVRqOCIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6e1wiYWxsb2NhdGlvbjAyXCI6W1wiZDI1MDJjZGFiMDhlZjg2NDkwOTBlOTI0ZDBmZDY5Y2U4OWMyYzMxNGZiNTJiZjU0NDA1NzY5YWE1YjQyNWVlYlwiXSxcImFsbG9jYXRpb24wMFwiOltcImY4YTFjNDI1Y2I2NTg5OWFlMGJmMGRkNGM2YTk4MjU1NGYyY2U4OTE0YzUwODQ0NDZmNDk5OWMzYzAwMTk5OTBcIl0sXCJhbGxvY2F0aW9uMDFcIjpbXCJlNTU2MTI0MDFmYjY3MTkwZDNjMGE5MDU3NDkyNjQ4MDZhM2I2NmM5YjlkMDQ4Yjg0Yjk2ZTEyMGU0YTM4N2RjXCJdfSxcImNvZGVcIjpcIihkZWZpbmUta2V5c2V0IFxcXCJhbGxvY2F0aW9uMDBcXFwiIChyZWFkLWtleXNldCBcXFwiYWxsb2NhdGlvbjAwXFxcIikpXFxuKGRlZmluZS1rZXlzZXQgXFxcImFsbG9jYXRpb24wMVxcXCIgKHJlYWQta2V5c2V0IFxcXCJhbGxvY2F0aW9uMDFcXFwiKSlcXG4oZGVmaW5lLWtleXNldCBcXFwiYWxsb2NhdGlvbjAyXFxcIiAocmVhZC1rZXlzZXQgXFxcImFsbG9jYXRpb24wMlxcXCIpKVwifX0sXCJzaWduZXJzXCI6W10sXCJtZXRhXCI6e1wiY3JlYXRpb25UaW1lXCI6MCxcInR0bFwiOjE3MjgwMCxcImdhc0xpbWl0XCI6MCxcImNoYWluSWRcIjpcIlwiLFwiZ2FzUHJpY2VcIjowLFwic2VuZGVyXCI6XCJcIn0sXCJub25jZVwiOlwidGVzdG5ldDA2LWtleXNldHMtTlwifSJ9", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IktleXNldCBkZWZpbmVkIn0sInJlcUtleSI6ImFpcl9tUEFnczltYTlpWjEtRXR4WHI1eV9RNG5vcWdJNU9uOTI2bFlUajgiLCJsb2dzIjoiMmtJc0VCSUlQU0c5MmZuRW96aGVtR1k1dmQ5Vlowb3ZCSXhNYmhJUUlUdyIsIm1ldGFEYXRhIjpudWxsLCJjb250aW51YXRpb24iOm51bGwsInR4SWQiOjR9") + , (unsafeFromText "eyJoYXNoIjoiRjlTLUpKX0Zrdm04dlQtZXdlSkRZZlZTMnVSR0lRcUVHdElUTHRyeDJZbyIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6bnVsbCxcImNvZGVcIjpcIihjb2luLmNyZWF0ZS1hbGxvY2F0aW9uLWFjY291bnQgXFxcImFsbG9jYXRpb24wMFxcXCIgKHRpbWUgXFxcIjE5MDAtMTAtMTVUMTg6MDA6MDBaXFxcIikgXFxcImFsbG9jYXRpb24wMFxcXCIgMTAwMDAwMC4wKVxcbihjb2luLmNyZWF0ZS1hbGxvY2F0aW9uLWFjY291bnQgXFxcImFsbG9jYXRpb24wMVxcXCIgKHRpbWUgXFxcIjIwMjYtMDEtMzFUMTg6MDA6MDBaXFxcIikgXFxcImFsbG9jYXRpb24wMVxcXCIgMTAwMDAwMC4wKVxcbihjb2luLmNyZWF0ZS1hbGxvY2F0aW9uLWFjY291bnQgXFxcImFsbG9jYXRpb24wMlxcXCIgKHRpbWUgXFxcIjIwMjYtMDYtMzFUMTg6MDA6MDBaXFxcIikgXFxcImFsbG9jYXRpb24wMlxcXCIgMTAwMDAwMC4wKVwifX0sXCJzaWduZXJzXCI6W10sXCJtZXRhXCI6e1wiY3JlYXRpb25UaW1lXCI6MCxcInR0bFwiOjE3MjgwMCxcImdhc0xpbWl0XCI6MCxcImNoYWluSWRcIjpcIlwiLFwiZ2FzUHJpY2VcIjowLFwic2VuZGVyXCI6XCJcIn0sXCJub25jZVwiOlwidGVzdG5ldC1hbGxvY2F0aW9ucy0wXCJ9In0", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IldyaXRlIHN1Y2NlZWRlZCJ9LCJyZXFLZXkiOiJGOVMtSkpfRmt2bTh2VC1ld2VKRFlmVlMydVJHSVFxRUd0SVRMdHJ4MllvIiwibG9ncyI6Ik8tWExZNzBQWDhHRzZiU3hTdk8xZUZNVDg1VUkxUnBFZ1BKVVJTMFJzTVkiLCJtZXRhRGF0YSI6bnVsbCwiY29udGludWF0aW9uIjpudWxsLCJ0eElkIjo1fQ") + , (unsafeFromText "eyJoYXNoIjoiVmk3cUJXZHFSdkllX3ZmNVN5ckVKcmtzQkFGTnRfaHRSVjJqcld6YUp6RSIsInNpZ3MiOltdLCJjbWQiOiJ7XCJuZXR3b3JrSWRcIjpudWxsLFwicGF5bG9hZFwiOntcImV4ZWNcIjp7XCJkYXRhXCI6e1wic2VuZGVyMFwiOltcImMwM2U3NmM4MWE1NmY4MzFhMWVhOTFiOTM1NDVkNmI0OWRhZmZjNTNjNzQxNmUwNjVmY2ZmYjAyNWZhZWU0N2VcIl19LFwiY29kZVwiOlwiKGNvaW4uY29pbmJhc2UgXFxcInNlbmRlcjBcXFwiICAocmVhZC1rZXlzZXQgXFxcInNlbmRlcjBcXFwiKSAxLjApXCJ9fSxcInNpZ25lcnNcIjpbXSxcIm1ldGFcIjp7XCJjcmVhdGlvblRpbWVcIjowLFwidHRsXCI6MTcyODAwLFwiZ2FzTGltaXRcIjowLFwiY2hhaW5JZFwiOlwiXCIsXCJnYXNQcmljZVwiOjAsXCJzZW5kZXJcIjpcIlwifSxcIm5vbmNlXCI6XCJ0ZXN0bmV0MDYtZ3JhbnRzLU5cIn0ifQ", unsafeFromText "eyJnYXMiOjAsInJlc3VsdCI6eyJzdGF0dXMiOiJzdWNjZXNzIiwiZGF0YSI6IldyaXRlIHN1Y2NlZWRlZCJ9LCJyZXFLZXkiOiJWaTdxQldkcVJ2SWVfdmY1U3lyRUpya3NCQUZOdF9odFJWMmpyV3phSnpFIiwibG9ncyI6Imhod3hCWjY0OTV1UFUtVExCV1Q5LWNIeHh0dkE4Y2lUYkJUdHo0TDY0eUkiLCJtZXRhRGF0YSI6bnVsbCwiY29udGludWF0aW9uIjpudWxsLCJ0eElkIjo2fQ") + ] diff --git a/src/Chainweb/Version/Registry.hs b/src/Chainweb/Version/Registry.hs index f0653cdf37..4a0f95261f 100644 --- a/src/Chainweb/Version/Registry.hs +++ b/src/Chainweb/Version/Registry.hs @@ -49,6 +49,7 @@ import Chainweb.Version.Development import Chainweb.Version.RecapDevelopment import Chainweb.Version.Mainnet import Chainweb.Version.Testnet04 +import Chainweb.Version.Testnet06 import Chainweb.Utils.Rule -- temporarily left off because it doesn't validate @@ -56,7 +57,7 @@ import Chainweb.Utils.Rule versionMap :: IORef (HashMap ChainwebVersionCode ChainwebVersion) versionMap = unsafePerformIO $ do traverse_ validateVersion knownVersions - newIORef $ HM.fromList [(_versionCode v, v) | v <- [mainnet, testnet04]] + newIORef $ HM.fromList [(_versionCode v, v) | v <- [mainnet, testnet04, testnet06]] -- | Register a version into our registry by code, ensuring it contains no -- errors and there are no others registered with that code. @@ -74,8 +75,8 @@ registerVersion v = do -- | Unregister a version from the registry. This is ONLY for testing versions. unregisterVersion :: HasCallStack => ChainwebVersion -> IO () unregisterVersion v = do - if elem (_versionCode v) (_versionCode <$> [mainnet, testnet04]) - then error "You cannot unregister mainnet or testnet04 versions" + if elem (_versionCode v) (_versionCode <$> [mainnet, testnet06]) + then error "You cannot unregister mainnet or testnet06 versions" else atomicModifyIORef' versionMap $ \m -> (HM.delete (_versionCode v) m, ()) validateVersion :: HasCallStack => ChainwebVersion -> IO () @@ -146,6 +147,7 @@ lookupVersionByName :: HasCallStack => ChainwebVersionName -> ChainwebVersion lookupVersionByName name | name == _versionName mainnet = mainnet | name == _versionName testnet04 = testnet04 + | name == _versionName testnet06 = testnet06 | otherwise = lookupVersion & versionName .~ name where lookupVersion = unsafeDupablePerformIO $ do @@ -163,12 +165,12 @@ fabricateVersionWithName name = -- | Versions known to us by name. knownVersions :: [ChainwebVersion] -knownVersions = [mainnet, testnet04, recapDevnet, devnet] +knownVersions = [mainnet, testnet04, testnet06, recapDevnet, devnet] -- | Look up a known version by name, usually with `m` instantiated to some -- configuration parser monad. findKnownVersion :: MonadFail m => ChainwebVersionName -> m ChainwebVersion findKnownVersion vn = case find (\v -> _versionName v == vn) knownVersions of - Nothing -> fail $ T.unpack (getChainwebVersionName vn) <> " is not a known version: try development, mainnet01, or testnet04" + Nothing -> fail $ T.unpack (getChainwebVersionName vn) <> " is not a known version: try development, mainnet01, or testnet04/06 " Just v -> return v diff --git a/src/Chainweb/Version/Testnet04.hs b/src/Chainweb/Version/Testnet04.hs index 7a4da9db56..369c9604b6 100644 --- a/src/Chainweb/Version/Testnet04.hs +++ b/src/Chainweb/Version/Testnet04.hs @@ -5,6 +5,8 @@ {-# language QuasiQuotes #-} {-# language ViewPatterns #-} +-- Note: Testnet04 is deprecated + module Chainweb.Version.Testnet04(testnet04, pattern Testnet04) where import Control.Lens diff --git a/src/Chainweb/Version/Testnet06.hs b/src/Chainweb/Version/Testnet06.hs new file mode 100644 index 0000000000..975c0cefe8 --- /dev/null +++ b/src/Chainweb/Version/Testnet06.hs @@ -0,0 +1,127 @@ +{-# language LambdaCase #-} +{-# language NumericUnderscores #-} +{-# language OverloadedStrings #-} +{-# language PatternSynonyms #-} +{-# language QuasiQuotes #-} +{-# language ViewPatterns #-} + +module Chainweb.Version.Testnet06(testnet06, pattern Testnet06) where + +import qualified Data.HashMap.Strict as HM +import qualified Data.Set as Set + +import Chainweb.BlockCreationTime +import Chainweb.BlockHeight +import Chainweb.ChainId +import Chainweb.Difficulty +import Chainweb.Graph +import Chainweb.Time +import Chainweb.Utils +import Chainweb.Utils.Rule +import Chainweb.Version +import P2P.BootstrapNodes + +import Pact.Types.Verifier + +import qualified Chainweb.Pact.Transactions.OtherTransactions as CoinV2 +import qualified Chainweb.Pact.Transactions.CoinV3Transactions as CoinV3 +import qualified Chainweb.Pact.Transactions.CoinV4Transactions as CoinV4 +import qualified Chainweb.Pact.Transactions.CoinV5Transactions as CoinV5 +import qualified Chainweb.Pact.Transactions.CoinV6Transactions as CoinV6 + +import qualified Chainweb.BlockHeader.Genesis.Testnet060Payload as TST0 +import qualified Chainweb.BlockHeader.Genesis.Testnet061to19Payload as TSTN + +pattern Testnet06 :: ChainwebVersion +pattern Testnet06 <- ((== testnet06) -> True) where + Testnet06 = testnet06 + +testnet06 :: ChainwebVersion +testnet06 = ChainwebVersion + { _versionCode = ChainwebVersionCode 0x00000008 + , _versionName = ChainwebVersionName "testnet06" + + , _versionForks = tabulateHashMap $ \case + SlowEpoch -> AllChains $ ForkAtBlockHeight $ BlockHeight 0 + Vuln797Fix -> AllChains $ ForkAtBlockHeight $ BlockHeight 0 + PactBackCompat_v16 -> AllChains $ ForkAtBlockHeight $ BlockHeight 0 + OldTargetGuard -> AllChains $ ForkAtBlockHeight $ BlockHeight 0 + SkipFeatureFlagValidation -> AllChains $ ForkAtBlockHeight $ BlockHeight 0 + SkipTxTimingValidation -> AllChains $ ForkAtBlockHeight $ BlockHeight 2 + ModuleNameFix -> AllChains $ ForkAtBlockHeight $ BlockHeight 2 + ModuleNameFix2 -> AllChains $ ForkAtBlockHeight $ BlockHeight 2 + CoinV2 -> onChains $ [(unsafeChainId 0, ForkAtBlockHeight $ BlockHeight 3)] <> [(unsafeChainId i, ForkAtBlockHeight $ BlockHeight 4) | i <- [1..19]] + OldDAGuard -> AllChains $ ForkAtBlockHeight $ BlockHeight 13 + PactEvents -> AllChains $ ForkAtBlockHeight $ BlockHeight 40 + SPVBridge -> AllChains $ ForkAtBlockHeight $ BlockHeight 50 + Pact4Coin3 -> AllChains $ ForkAtBlockHeight $ BlockHeight 80 + Pact42 -> AllChains $ ForkAtBlockHeight $ BlockHeight 90 + EnforceKeysetFormats -> AllChains $ ForkAtBlockHeight $ BlockHeight 100 + CheckTxHash -> AllChains $ ForkAtBlockHeight $ BlockHeight 110 + Chainweb213Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 95 + Chainweb214Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 115 + Chainweb215Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 165 + Pact44NewTrans -> AllChains $ ForkAtBlockHeight $ BlockHeight 185 + Chainweb216Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 215 + Chainweb217Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 470 + Chainweb218Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 500 + Chainweb219Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 550 + Chainweb220Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 560 + Chainweb221Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 580 + Chainweb222Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 590 + Chainweb223Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 600 + Chainweb224Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 610 + Chainweb225Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 620 + Pact5Fork -> AllChains $ ForkAtBlockHeight $ BlockHeight 640 + Chainweb228Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 650 + Chainweb230Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 680 + Chainweb231Pact -> AllChains $ ForkAtBlockHeight $ BlockHeight 690 + Chainweb31 -> AllChains $ ForkAtBlockHeight $ BlockHeight 700 + MigratePlatformShare -> AllChains $ ForkNever + + , _versionUpgrades = foldr (chainZip HM.union) (AllChains mempty) + [ indexByForkHeights testnet06 + [ (CoinV2, AllChains (Pact4Upgrade CoinV2.transactions False)) + , (Pact4Coin3, AllChains (Pact4Upgrade CoinV3.transactions True)) + , (Chainweb214Pact, AllChains (Pact4Upgrade CoinV4.transactions True)) + , (Chainweb215Pact, AllChains (Pact4Upgrade CoinV5.transactions True)) + , (Chainweb223Pact, AllChains (Pact4Upgrade CoinV6.transactions False)) + ] + ] + + , _versionGraphs = Bottom (minBound, twentyChainGraph) + , _versionBlockDelay = BlockDelay 30_000_000 + , _versionWindow = WindowWidth 120 + , _versionHeaderBaseSizeBytes = 318 - 110 + , _versionBootstraps = domainAddr2PeerInfo testnet06BootstrapHosts + , _versionGenesis = VersionGenesis + -- TODO Setup properly here + { _genesisBlockTarget = onChains $ concat + [ [(unsafeChainId i, HashTarget $ maxBound `div` 100_000) | i <- [0..19]] + ] + -- TODO Setup Genesis time properly + , _genesisTime = AllChains $ BlockCreationTime [timeMicrosQQ| 2019-07-17T18:28:37.613832 |] + , _genesisBlockPayload = onChains $ concat + [ [(unsafeChainId 0, TST0.payloadBlock)] + , [(unsafeChainId i, TSTN.payloadBlock) | i <- [1..19]] + ] + } + + , _versionMaxBlockGasLimit = Bottom (minBound, Just 180_000) + , _versionSpvProofRootValidWindow = Bottom (minBound, Nothing) + , _versionCheats = VersionCheats + { _disablePow = False + , _fakeFirstEpochStart = False + , _disablePact = False + } + , _versionDefaults = VersionDefaults + { _disablePeerValidation = False + , _disableMempoolSync = False + } + , _versionVerifierPluginNames = AllChains $ + (600, Set.fromList $ map VerifierName ["hyperlane_v3_message"]) `Above` + Bottom (minBound, mempty) + , _versionQuirks = noQuirks + , _versionForkNumber = 0 + , _versionForkVoteCastingLength = 120 * 119 -- 5 days + } diff --git a/src/P2P/BootstrapNodes.hs b/src/P2P/BootstrapNodes.hs index 0e5ccbafde..72747880a7 100644 --- a/src/P2P/BootstrapNodes.hs +++ b/src/P2P/BootstrapNodes.hs @@ -13,6 +13,8 @@ module P2P.BootstrapNodes ( mainnetBootstrapHosts , testnet04BootstrapHosts +, testnet06BootstrapHosts +, ) where -- internal modules @@ -57,3 +59,18 @@ testnet04BootstrapHosts = [] -- , "ap1.testnet.chainweb.com:443" -- , "ap2.testnet.chainweb.com:443" -- ] + +-- -------------------------------------------------------------------------- -- +-- | Testnet06 bootstrap nodes. +-- +-- Nodes in this list need a public DNS name and a corresponding TLS +-- certificate. Operators of the nodes are expected to guarantee long term +-- availability of the nodes. +-- +-- Please make a pull request, if you like to see your node being included here. +-- +testnet06BootstrapHosts :: [HostAddress] +testnet06BootstrapHosts = map unsafeHostAddressFromText + [ "testnet06-1.chainweb-community.org:443" + , "testnet06-2.chainweb-community.org:443" + ]